逆向攻防世界CTF系列47-crazy
64位无壳
丢入IDA并远程调试,随便输入,发现要32位长度,也跟踪到几个函数

校验函数:

判断长度和简单处理逻辑

主函数:

到这逻辑理顺了,其实这题的难点主要在眼花缭乱的C++代码
会头看这,应该是把输入赋值给a1+16和a1+48,然后把327a6c4304ad5938eaf0efb6cc3e53dc给了a1+80
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
| unsigned __int64 __fastcall HighTemplar::HighTemplar(DarkTemplar *a1, __int64 a2) { char v3; unsigned __int64 v4;
v4 = __readfsqword(0x28u); DarkTemplar::DarkTemplar(a1); *(_QWORD *)a1 = &off_401EA0; *((_DWORD *)a1 + 3) = 0; std::string::basic_string((__int64)a1 + 16, a2); std::string::basic_string((__int64)a1 + 48, a2); std::allocator<char>::allocator(&v3); std::string::basic_string((char *)a1 + 80, "327a6c4304ad5938eaf0efb6cc3e53dc", &v3); std::allocator<char>::~allocator(&v3); return __readfsqword(0x28u) ^ v4; }
|
要逆向,从后面再看:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17
| __int64 __fastcall HighTemplar::getSerial(HighTemplar *this){ for ( i = 0; (int)i < (unsigned __int64)std::string::length((char *)this + 16); ++i){ v1 = *(_BYTE *)std::string::operator[]((char *)this + 80, (int)i); if ( v1 != *(_BYTE *)std::string::operator[]((char *)this + 16, (int)i)){ v4 = std::operator<<<std::char_traits<char>>(&std::cout, "You did not pass "); v5 = std::ostream::operator<<(v4, i); std::ostream::operator<<(v5, &std::endl<char,std::char_traits<char>>); *((_DWORD *)this + 3) = 1; return *((unsigned int *)this + 3); } v2 = std::operator<<<std::char_traits<char>>(&std::cout, "Pass "); v3 = std::ostream::operator<<(v2, i); std::ostream::operator<<(v3, &std::endl<char,std::char_traits<char>>); } return *((unsigned int *)this + 3); }
|
enc:327a6c4304ad5938eaf0efb6cc3e53dc
看calctue:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18
| bool __fastcall HighTemplar::calculate(HighTemplar *this){ if ( std::string::length((char *)this + 16) != 32 ){ v1 = std::operator<<<std::char_traits<char>>(&std::cout, "Too short or too long"); std::ostream::operator<<(v1, &std::endl<char,std::char_traits<char>>); exit(-1); } for ( i = 0; i <= (unsigned __int64)std::string::length((char *)this + 16); ++i ){ v2 = (_BYTE *)std::string::operator[]((char *)this + 16, i); *v2 = (*(_BYTE *)std::string::operator[]((char *)this + 16, i) ^ 0x50) + 23; } for ( j = 0; ; ++j ) { result = j <= (unsigned __int64)std::string::length((char *)this + 16); if ( !result ) break; v4 = (_BYTE *)std::string::operator[]((char *)this + 16, j); *v4 = (*(_BYTE *)std::string::operator[]((char *)this + 16, j) ^ 0x13) + 11; } return result; }
|
首先满足32长度,然后对每一位^0x50,再加23
然后 ^ 0x13 + 11
解密代码:
1 2 3 4 5 6 7 8 9 10 11 12
| enc = '327a6c4304ad5938eaf0efb6cc3e53dc'
flag1 = '' flag2 = ''
for i in range(len(enc)): flag1 += chr(ord(enc[i]) - 11 ^ 0x13)
for i in range(len(enc)): flag2 += chr(ord(flag1[i]) - 23 ^ 0x50)
print(flag2)
|