赞
踩
给了源码, 修改一下decode, 可以解出flag
// Damn.cpp : 定义控制台应用程序的入口点。 // #include "stdafx.h" #include<Windows.h> #include<iostream> #include<ctime> #include<cstdio> #include<cstring> #include<cstdlib> using namespace std; int map[100], c = 2; int key[7] = { 'b', 'k', 'd', 'c', 'e', 'w' }; unsigned int ks[6] = { 0x8c2c133a, 0xf74cb3f6, 0xfedfa6f2, 0xab293e3b, 0x26cf8a2a, 0x88a1f279 };//{0x8c2cecc5,0xf74cb3f6,0xfedf590d,0xab293e3b,0x26cf75d5,0x88a1f279}; FARPROC proc = NULL; int initHook() { HMODULE hModule = LoadLibraryA("Kernel32.dll"); if (hModule) { proc = GetProcAddress(hModule, "IsDebuggerPresent"); if (proc == NULL) return -1; } return 0; } PDWORD update() { if (initHook() != 0) exit(-1); HANDLE hProcess = GetModuleHandle(NULL); PIMAGE_DOS_HEADER dos_header = (PIMAGE_DOS_HEADER)hProcess; PIMAGE_NT_HEADERS nt_header = (PIMAGE_NT_HEADERS)(dos_header->e_lfanew + (DWORD)hProcess); IMAGE_OPTIONAL_HEADER* opt_header = &(nt_header->OptionalHeader); PIMAGE_IMPORT_DESCRIPTOR iat = (PIMAGE_IMPORT_DESCRIPTOR)((DWORD)hProcess + opt_header->DataDirectory[IMAGE_DIRECTORY_ENTRY_IMPORT].VirtualAddress); while (iat->FirstThunk) { PIMAGE_THUNK_DATA data = (PIMAGE_THUNK_DATA)(iat->FirstThunk + (DWORD)hProcess); while (data->u1.Function) { if (IMAGE_SNAP_BY_ORDINAL(data->u1.AddressOfData)) { data++; continue; } if ((DWORD)proc == data->u1.Function) return &data->u1.Function; data++; } iat++; } return NULL; } PDWORD table_addr = NULL; int writeAddr(DWORD addr) { if (table_addr == NULL) table_addr = update(); DWORD dwOldProtect; MEMORY_BASIC_INFORMATION mbi_thunk; VirtualQuery(table_addr, &mbi_thunk, sizeof(MEMORY_BASIC_INFORMATION)); VirtualProtect(mbi_thunk.BaseAddress, mbi_thunk.RegionSize, PAGE_READWRITE, &mbi_thunk.Protect); *table_addr = (DWORD)addr; VirtualProtect(mbi_thunk.BaseAddress, mbi_thunk.RegionSize, mbi_thunk.Protect, &dwOldProtect); return 0; } bool WINAPI CallBackProc() { writeAddr((DWORD)proc); map[0] = 2; c--; key[0] = 'a'; key[1] = 'k'; key[2] = 'e'; key[3] = 'y'; key[4] = 'e'; key[5] = 'z'; if (IsDebuggerPresent()) { MessageBoxW(NULL, L"U R Using Debugger!", L"U Suck!", MB_OK); return true; } return false; } void sleep() { } int checkDebug() { if (*((unsigned char *)(*(DWORD*)(__readfsdword(0x18) + 0x30)) + 0x2)) sleep(); return 0; } int doit = writeAddr((DWORD)CallBackProc), dd = checkDebug(); bool encode(char* ur_flag) { unsigned int k = 0, bk = 0; for (int i = 0; i < strlen(ur_flag); i += 4) { k = (((int)ur_flag[i]) << 24) | (((int)ur_flag[i + 1]) << 16) | (((int)ur_flag[i + 2]) << 8) | ((int)ur_flag[i + 3]); k = (k >> key[i / 4]) | (k << (32 - key[i / 4])); _asm { call sub2 _emit 0xEB jmp label2 sub2 : add dword ptr[esp], 1 retn label2 : } k = ((~(k >> 16)) & 0x0000ffff) | (k << 16); k = (1 << key[i / 4]) ^ k; _asm { call sub7 _emit 0xE8 jmp label7 sub7 : add dword ptr[esp], 1 retn label7 : } if (i > 0) k ^= bk; bk = k; if (k != ks[i / 4]) return false; } return true; } void genKey() { int len = 20, keylen = 6, maxium = 0; int before; _asm { call sub1 _emit 0xE8 jmp label1 sub1 : add dword ptr[esp], 1 retn label1 : } srand(time(NULL)); for (int i = 1; i <= len; i++) { map[i] = map[i - 1] + rand() % 5; _asm { call sub3 _emit 0xE8 jmp label3 sub3 : add dword ptr[esp], 1 retn label3 : } maxium = maxium > map[i] ? maxium : map[i]; } _asm { call sub4 _emit 0xE8 jmp label4 sub4 : add dword ptr[esp], 1 retn label4 : } before = time(NULL); _asm { call sub5 _emit 0xE8 jmp label5 sub5 : add dword ptr[esp], 1 retn label5 : } for (int i = 0; i < keylen; i++) { int step = 0; long long t = time(NULL); int delta = t - before; _asm { call sub6 _emit 0xE8 jmp label6 sub6 : add dword ptr[esp], 1 retn label6 : } if (delta > maxium) return; for (int j = 0; j <= len; j++) if (delta <= map[j]) { step = map[j]; _asm { call sub8 _emit 0xE8 jmp label8 sub8 : add dword ptr[esp], 1 retn label8 : } break; } _asm { call sub9 _emit 0xE8 jmp label9 sub9 : add dword ptr[esp], 1 retn label9 : } key[i] = (key[i] * c + step + i * 3) % 32; _asm { call sub10 _emit 0xE8 jmp label10 sub10 : add dword ptr[esp], 1 retn label10 : } before = t; } } void decode() { unsigned int k[] = { 0x03, 0x10, 0x0d, 0x04, 0x13, 0x0b }; for (int i = 5; i >= 0; i--) { if (i > 0) ks[i] ^= ks[i - 1]; ks[i] = (1 << k[i]) ^ ks[i]; ks[i] = ((ks[i] >> 16)) | ((~(ks[i] << 16)) & 0xffff0000); ks[i] = ((ks[i] << k[i])) | (ks[i] >> (32 - k[i])); printf("%X\n", ks[i]); } } int main() { char ur_flag[50]; cout << "please input your flag:" << endl; cin >> ur_flag; //if (IsDebuggerPresent()) //{ // cout << "U suck! 233333" << endl; // Sleep(2000); // exit(0); //} if (strlen(ur_flag) != 24) { cout << "Wrong!" << endl; Sleep(2000); exit(0); } genKey(); bool flag = encode(ur_flag); decode(); //if (flag) //{ // cout << "U did it!" << endl << "GJ!" << endl; // system("pause"); // exit(0); //} //cout << "Wrong!" << endl; //Sleep(2000); getchar(); getchar(); return 0; }
拼接起来
from Crypto.Util.number import long_to_bytes flags = [ 0x6573747D, 0x6F725F74, 0x72655F66, 0x6132795F, 0x7B615F33, 0x666C6167 ] flag = b'' for i in range(len(flags) - 1, -1, -1): flag += long_to_bytes(flags[i]) print(flag)
int __cdecl main(int argc, const char **argv, const char **envp) { int v4; // ebx int v5; // ebx char v7_20[20]; // [esp+1Ch] [ebp-48h] BYREF int v7[3]; // [esp+30h] [ebp-34h] BYREF size_t v8; // [esp+3Ch] [ebp-28h] int v9; // [esp+40h] [ebp-24h] int v10; // [esp+44h] [ebp-20h] int func_ptr[3]; // [esp+48h] [ebp-1Ch] int v12; // [esp+54h] [ebp-10h] size_t length; // [esp+58h] [ebp-Ch] FILE *file_ptr; // [esp+5Ch] [ebp-8h] __main(); func_ptr[0] = (int)func0; func_ptr[1] = (int)func1; func_ptr[2] = (int)func2; v7[0] = 0; v7[1] = 1; v7[2] = 2; v8 = 3; v9 = 3; v10 = 4; file_ptr = fopen("data", "rb"); if ( !file_ptr ) return -1; fseek(file_ptr, 0, 2); length = ftell(file_ptr); fseek(file_ptr, 0, 0); v12 = ftell(file_ptr); if ( v12 ) { puts("something wrong"); return 0; } else { for ( i = 0; i < (int)length; ++i ) { v4 = i; v7_20[v4] = fgetc(file_ptr); } if ( strlen(v7_20) <= length ) { length = v8; i = 0; v12 = v10; while ( i <= 2 ) { v5 = i + 1; v7[v5] = ((int (__cdecl *)(int *, int, int))func_ptr[i])(v7, v9, v10);// get v9,v10,v11 v9 = ++i; v10 = i + 1; } if ( v8 ) { return -1; } else { get_key(length, v12); system("PAUSE"); return 0; } } else { return -1; } } } signed int __cdecl func0(int a1, int a2, int a3) { int temp; // [esp+Ch] [ebp-4h] temp = *(_DWORD *)(4 * a2 + a1); *(_DWORD *)(a1 + 4 * a2) = *(_DWORD *)(4 * a3 + a1); *(_DWORD *)(a1 + 4 * a3) = temp; return 1; } unsigned int __cdecl func1(int a1, int a2, int a3) { return abs32(*(_DWORD *)(4 * a2 + a1) + *(_DWORD *)(4 * a3 + a1)) - abs32(*(_DWORD *)(4 * a3 + a1)) - abs32(*(_DWORD *)(4 * a2 + a1)) + 2; } // return v11 unsigned int __cdecl func2(int a1, int a2, int a3) { return abs32(*(_DWORD *)(4 * a3 + a1)) - abs32(*(_DWORD *)(4 * a3 + a1) + *(_DWORD *)(4 * a2 + a1)) + abs32(*(_DWORD *)(4 * a2 + a1)) + 2; }
func0
相当于X <--> Y
func1
相当于|X + Y| - |X| - |Y| + 2
func2
相当于|X| + |Y| - |X + Y| + 2
目标是程序读入data后, 使得func2
的返回值为0, 但是不可能(因为数学上|X| + |Y| >= |X + Y|), 不过func1
的值可以等于0, 所以这里先用func0交换func1和func2的函数指针, 这样func1返回0时, 就可以进入打印flag的函数get_key
构造data
(这道题更像PWN
UPX脱壳后拖入IDA
int __cdecl main(int argc, const char **argv, const char **envp) { int i; // [sp+8h] [-58h] _BYTE v5[32]; // [sp+10h] [-50h] BYREF char v6[24]; // [sp+30h] [-30h] BYREF int v7; // [sp+48h] [-18h] const char **v8; // [sp+4Ch] [-14h] v7 = argc; v8 = argv; strcpy(v6, "EHK}kanqxgarqygtre"); printf("Insert password : ", argv, envp); _isoc99_scanf("%31s", v5); for ( i = 0; i < 21; ++i ) v6[i] -= 2; if ( strcmp(v5, v6) ) puts("Wrong password."); else puts("Password is correct. Submit this as the flag."); return 0; }
给定字符串, 每个字符-2
, 直接逆
s = 'EHK}kanqxgarqygtre'
flag = ''
for i in range(len(s)):
flag += chr(ord(s[i]) - 2)
print(flag)
APK逆向, JEB打开, 四处游走一下搜索关键代码
package com.example.blink; import android.graphics.BitmapFactory; import android.os.Bundle; import android.support.v7.app.AppCompatActivity; import android.util.Base64; import android.widget.ImageView; public class r2d2 extends AppCompatActivity { @Override // android.support.v7.app.AppCompatActivity protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); this.setContentView(0x7F0A001C); // layout:activity_r2d2 ImageView image = (ImageView)this.findViewById(0x7F080054); // id:imageView byte[] arr_b = Base64.decode(" ".split(",")[1], 0); image.setImageBitmap(BitmapFactory.decodeByteArray(arr_b, 0, arr_b.length)); } }
这里的base64很长, 解码看有没有重要信息, 仔细看发现是图片编码为base64
一眼控制流平坦化, deflat一波, 脚本链接
int __cdecl main(int argc, const char **argv, const char **envp) { int v3; // eax int v4; // eax int v5; // ecx int v6; // eax int v7; // eax int v8; // eax int v9; // eax int v10; // eax char v11; // al char v12; // al char v13; // al char v14; // al char v15; // al int v16; // ecx int v17; // eax int v18; // eax int v19; // eax int v20; // eax int v21; // eax int v22; // ecx char v24; // al char v25; // al char v26; // al char v27; // al int v28; // [rsp+9Ch] [rbp-94h] char v29[8]; // [rsp+A0h] [rbp-90h] BYREF char v30[8]; // [rsp+A8h] [rbp-88h] BYREF char v31[8]; // [rsp+B0h] [rbp-80h] BYREF char v32[8]; // [rsp+B8h] [rbp-78h] BYREF char v33[8]; // [rsp+C0h] [rbp-70h] BYREF char v34[7]; // [rsp+C8h] [rbp-68h] BYREF char v35; // [rsp+CFh] [rbp-61h] int v36; // [rsp+D0h] [rbp-60h] int v37; // [rsp+D4h] [rbp-5Ch] int v38; // [rsp+D8h] [rbp-58h] int v39; // [rsp+DCh] [rbp-54h] char s; // [rsp+E0h] [rbp-50h] BYREF char v41[23]; // [rsp+E1h] [rbp-4Fh] BYREF char v42[8]; // [rsp+F8h] [rbp-38h] BYREF char v43[8]; // [rsp+100h] [rbp-30h] BYREF char v44[8]; // [rsp+108h] [rbp-28h] BYREF char v45[4]; // [rsp+110h] [rbp-20h] BYREF int v46; // [rsp+114h] [rbp-1Ch] const char **v47; // [rsp+118h] [rbp-18h] int v48; // [rsp+120h] [rbp-10h] int v49; // [rsp+124h] [rbp-Ch] int v50; // [rsp+128h] [rbp-8h] bool v51; // [rsp+12Eh] [rbp-2h] bool v52; // [rsp+12Fh] [rbp-1h] v49 = 0; v48 = argc; v47 = argv; v46 = time(0LL); puts("func(?)=\"01abfc750a0c942167651c40d088531d\"?"); s = getchar(); fgets(v41, 21, stdin); v39 = time(0LL); v38 = v39 - v46; v50 = v39 - v46; v28 = 1883240069; while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( 1 ) { while ( v28 == -2090540314 ) { v35 = v38 ^ v41[v36 - 1]; v34[0] = main::$_0::operator()(v44, (unsigned int)v35); v33[0] = main::$_1::operator()(v42, (unsigned int)*(&s + v38 + v36 - 1)); v11 = main::$_1::operator() const(char)::{lambda(int)#1}::operator()(v33, 7LL); v35 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()( v34, (unsigned int)v11); v32[0] = main::$_2::operator()(v45, (unsigned int)v35); v31[0] = main::$_2::operator()(v45, (unsigned int)*(&s + v38 + v36 - 1)); v12 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()(v31, 18LL); v30[0] = main::$_3::operator()(v43, (unsigned int)v12); v13 = main::$_3::operator() const(char)::{lambda(char)#1}::operator()(v30, 3LL); v29[0] = main::$_0::operator()(v44, (unsigned int)v13); v14 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()(v29, 2LL); v15 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()( v32, (unsigned int)v14); v16 = 1299792285; v35 = v15; v52 = enc[v36 - 1] != v15; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v16 = -424557443; v28 = v16; } if ( v28 != -1957245689 ) break; v28 = 1587023630; } if ( v28 != -1884297895 ) break; v28 = -984930794; puts("You win"); } if ( v28 != -1852837876 ) break; v18 = 1375414539; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v18 = 1154698238; v28 = v18; } if ( v28 != -1220297252 ) break; v21 = -1884297895; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v21 = -984930794; v28 = v21; } if ( v28 != -984930794 ) break; puts("You win"); v22 = -1884297895; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v22 = 456293525; v28 = v22; } if ( v28 != -740226431 ) break; v4 = 2137069843; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v4 = 739060228; v28 = v4; } if ( v28 == -459161563 ) exit(0); if ( v28 != -424557443 ) break; v17 = 1856799435; if ( v52 ) v17 = -1852837876; v28 = v17; } if ( v28 != -350248402 ) break; v28 = -55540564; } if ( v28 != -294402024 ) break; v6 = 1721328217; if ( v51 ) v6 = -459161563; v28 = v6; } if ( v28 != -226137905 ) break; v28 = 24093646; } if ( v28 != -55540564 ) break; v20 = -350248402; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v20 = -226137905; v28 = v20; } if ( v28 != 24093646 ) break; ++v36; v28 = 1587023630; } if ( v28 == 456293525 ) break; switch ( v28 ) { case 506113758: puts("Let the silent second hand take the place of my doubt..."); exit(0); case 623475433: v36 = 1; v28 = 1132336453; break; case 739060228: v37 = strlen(&s); v51 = v37 != 21; v5 = 2137069843; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v5 = -294402024; v28 = v5; break; case 1011555671: v10 = 1299792285; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v10 = -2090540314; v28 = v10; break; case 1132336453: v8 = 623475433; v36 = 1; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v8 = -1957245689; v28 = v8; break; case 1154698238: exit(0); case 1299792285: v35 = v38 ^ v41[v36 - 1]; v34[0] = main::$_0::operator()(v44, (unsigned int)v35); v33[0] = main::$_1::operator()(v42, (unsigned int)*(&s + v38 + v36 - 1)); v24 = main::$_1::operator() const(char)::{lambda(int)#1}::operator()(v33, 7LL); v35 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()(v34, (unsigned int)v24); v32[0] = main::$_2::operator()(v45, (unsigned int)v35); v31[0] = main::$_2::operator()(v45, (unsigned int)*(&s + v38 + v36 - 1)); v25 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()(v31, 18LL); v30[0] = main::$_3::operator()(v43, (unsigned int)v25); v26 = main::$_3::operator() const(char)::{lambda(char)#1}::operator()(v30, 3LL); v29[0] = main::$_0::operator()(v44, (unsigned int)v26); v27 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()(v29, 2LL); v35 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()(v32, (unsigned int)v27); v28 = -2090540314; break; case 1375414539: exit(0); case 1587023630: v9 = -1220297252; if ( v36 < 21 ) v9 = 1011555671; v28 = v9; break; case 1721328217: v7 = 623475433; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v7 = 1132336453; v28 = v7; break; case 1856799435: v19 = -350248402; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) v19 = -55540564; v28 = v19; break; case 1883240069: v3 = -740226431; if ( v50 > 0 ) v3 = 506113758; v28 = v3; break; default: v37 = strlen(&s); v28 = 739060228; break; } } return 0; }
得到去平坦化的代码, 勉强可以读了
int __cdecl main(int argc, const char **argv, const char **envp) { char v3; // al char v4; // al char v5; // al char v6; // al char v8; // al char v9; // al char v10; // al char v11; // al char v12[8]; // [rsp+A0h] [rbp-90h] BYREF char v13[8]; // [rsp+A8h] [rbp-88h] BYREF char v14[8]; // [rsp+B0h] [rbp-80h] BYREF char v15[8]; // [rsp+B8h] [rbp-78h] BYREF char v16[8]; // [rsp+C0h] [rbp-70h] BYREF char v17[7]; // [rsp+C8h] [rbp-68h] BYREF char v18; // [rsp+CFh] [rbp-61h] int v19; // [rsp+D0h] [rbp-60h] int v20; // [rsp+D4h] [rbp-5Ch] int v21; // [rsp+D8h] [rbp-58h] int v22; // [rsp+DCh] [rbp-54h] char s; // [rsp+E0h] [rbp-50h] BYREF char v24[23]; // [rsp+E1h] [rbp-4Fh] BYREF char v25[8]; // [rsp+F8h] [rbp-38h] BYREF char v26[8]; // [rsp+100h] [rbp-30h] BYREF char v27[8]; // [rsp+108h] [rbp-28h] BYREF char v28[4]; // [rsp+110h] [rbp-20h] BYREF int v29; // [rsp+114h] [rbp-1Ch] const char **v30; // [rsp+118h] [rbp-18h] int v31; // [rsp+120h] [rbp-10h] int v32; // [rsp+124h] [rbp-Ch] int v33; // [rsp+128h] [rbp-8h] bool v34; // [rsp+12Eh] [rbp-2h] v32 = 0; v31 = argc; v30 = argv; v29 = time(0LL); puts("func(?)=\"01abfc750a0c942167651c40d088531d\"?"); s = getchar(); fgets(v24, 21, stdin); v22 = time(0LL); v21 = v22 - v29; v33 = v22 - v29; if ( y >= 10 && ((((_BYTE)x - 1) * (_BYTE)x) & 1) != 0 ) goto LABEL_13; while ( 1 ) { v20 = strlen(&s); v34 = v20 != 21; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) break; LABEL_13: v20 = strlen(&s); } while ( 1 ) { v19 = 1; if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) break; v19 = 1; } while ( v19 < 21 ) { if ( y >= 10 && ((((_BYTE)x - 1) * (_BYTE)x) & 1) != 0 ) { v18 = v21 ^ v24[v19 - 1]; v17[0] = main::$_0::operator()(v27, (unsigned int)v18); v16[0] = main::$_1::operator()(v25, (unsigned int)*(&s + v21 + v19 - 1)); v8 = main::$_1::operator() const(char)::{lambda(int)#1}::operator()(v16, 7LL); v18 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()(v17, (unsigned int)v8); v15[0] = main::$_2::operator()(v28, (unsigned int)v18); v14[0] = main::$_2::operator()(v28, (unsigned int)*(&s + v21 + v19 - 1)); v9 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()(v14, 18LL); v13[0] = main::$_3::operator()(v26, (unsigned int)v9); v10 = main::$_3::operator() const(char)::{lambda(char)#1}::operator()(v13, 3LL); v12[0] = main::$_0::operator()(v27, (unsigned int)v10); v11 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()(v12, 2LL); v18 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()(v15, (unsigned int)v11); } do { v18 = v21 ^ v24[v19 - 1]; v17[0] = main::$_0::operator()(v27, (unsigned int)v18); v16[0] = main::$_1::operator()(v25, (unsigned int)*(&s + v21 + v19 - 1)); v3 = main::$_1::operator() const(char)::{lambda(int)#1}::operator()(v16, 7LL); v18 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()(v17, (unsigned int)v3); v15[0] = main::$_2::operator()(v28, (unsigned int)v18); v14[0] = main::$_2::operator()(v28, (unsigned int)*(&s + v21 + v19 - 1)); v4 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()(v14, 18LL); v13[0] = main::$_3::operator()(v26, (unsigned int)v4); v5 = main::$_3::operator() const(char)::{lambda(char)#1}::operator()(v13, 3LL); v12[0] = main::$_0::operator()(v27, (unsigned int)v5); v6 = main::$_0::operator() const(char)::{lambda(char)#1}::operator()(v12, 2LL); v18 = main::$_2::operator() const(char)::{lambda(char)#1}::operator()(v15, (unsigned int)v6); } while ( enc[v19 - 1] != v18 ); while ( y >= 10 && ((((_BYTE)x - 1) * (_BYTE)x) & 1) != 0 ) ; ++v19; } if ( y >= 10 && ((((_BYTE)x - 1) * (_BYTE)x) & 1) != 0 ) goto LABEL_16; while ( 1 ) { puts("You win"); if ( y < 10 || ((((_BYTE)x - 1) * (_BYTE)x) & 1) == 0 ) break; LABEL_16: puts("You win"); } return 0; }
分析之后, 可以写出等价的处理逻辑
((lastchar % 7) + nextchar) ^ (lastchar & 18 * 3 + 2) == enc[i]
每相邻两个字符进行加密操作,最后跟enc的字符相比较
这个字符串是#
字符的md5值, 所以第一个字符是#
, 这样就可以反推全部字符
checklist = [
243, 46, 24, 54, 225, 76, 34, 209, 249, 140,
64, 118, 244, 14, 0, 5, 163, 144, 14, 165
]
flag = '#'
for i in range(len(checklist)):
lastc = ord(flag[-1])
enci = checklist[i]
nextc = ((enci ^ (((lastc ^ 18) * 3) + 2)) & 0xFF) - (lastc % 7)
flag += chr(nextc)
print(flag)
至此BUU前100个逆向题打完, 后续的WP可能只出有价值的题目, 并不再按顺序打
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。