还是决定把进阶区以前写过也都记录一下,也算是一个温故知新的过程吧。另外希望自己能早点把进阶区百人以上的题目刷完,加油!(2020.6.2)。
啊,进阶区的题目经常会有变动,前面的题可能会被调到后面,所以顺序可能会不太对。
拿到程序,是个jar文件,拖进jd-jui查看,可以直接看到主函数。可以看到,当判断成立时,经过异或运算后会直接输出flag
my_number的数值已经直接给出了,直接运算向下取整作为输入
或者也可以按照程序给的数值直接运算出flag,这个代码我目前实在不会写,直接照着别人的代码写的,等有自己的理解的再来更
a = '4b64ca12ace755516c178f72d05d7061'
b = 'ecd44646cfe5994ebeb35bf922e25dba'
c = bytes.fromhex(a)
d = bytes.fromhex(b)
flag = [0] * 16
for i in range(len(c)):
flag[i] = c[i] ^ d[i]
flag[i] = hex(flag[i])
print(flag)
直接搜字符串发现有个flag,提交发现是错的,然后对字符串right一路交叉引用过去找到主要判断函数
注意一下输入是倒序就行,然后这个题的反汇编是有一定错误的,v2指向不明,不过不影响做题,因为好猜,读一下发现不难理解
.text:00401100 loc_401100: ; CODE XREF: sub_401080+8E↓j
.text:00401100 mov al, [esi]
.text:00401102 lea esi, [esi-1]
.text:00401105 mov byte_40336C[edx], al
.text:0040110B inc edx
.text:0040110C cmp edx, ecx
.text:0040110E jl short loc_401100
.text:00401110 pop esi
.text:00401111
.text:00401111 loc_401111: ; CODE XREF: sub_401080+6F↑j
.text:00401111 xor edx, edx
.text:00401113
.text:00401113 loc_401113: ; CODE XREF: sub_401080+A6↓j
.text:00401113 mov al, byte_40336C[edx]
.text:00401119 inc al
.text:0040111B xor al, 6
.text:0040111D mov byte_40336C[edx], al
.text:00401123 inc edx
.text:00401124 cmp edx, ecx
.text:00401126 jb short loc_401113
.text:00401128 mov ecx, offset unk_402124
.text:0040112D mov eax, offset byte_40336C
.text:00401132
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-m3Ce91ow-1593054216962)(https://i.loli.net/2020/06/03/huBINj1t6pECA8m.png)]
直接上代码:
cm = [0x78, 0x49, 0x72, 0x43, 0x6A, 0x7E, 0x3C, 0x72, 0x7C, 0x32,
0x74, 0x57, 0x73, 0x76, 0x33, 0x50, 0x74, 0x49, 0x7F, 0x7A,
0x6E, 0x64, 0x6B, 0x61]
flag = ''
for i in range(len(cm)):
flag += chr((cm[i] ^ 6) - 1)
print(flag[::-1])
签到水瓶,根据题目描述,可知主函数给出的一长列字符就是要找到flag
cm = [83,69,67,67,79,78,123,87,101,108,99,111,109,101,32,116,111,32,116,104,101,32,83,69,67,67,79,78,32,50,48,49,52,32,67,84,70,33,125]
flag = ''
for i in range(len(cm)):
flag += chr(cm[i])
print(flag)
打开发现无法反编译MIPS,可以考虑A:去官网下载插件,或者B:直接读汇编。但是这里我选C,我赌出题人的枪里没有子弹。
cm = 'cbtcqLUBChERV[[Nh@_X^D]X_YPV[CJ'
flag = ''
for i in range(len(cm)):
flag += chr(ord(cm[i]) ^ 0x37)
print(flag)
一开始我以为又是直接硬塞flag,把数字转成字符就OK了,但验证发现并不对
然后仔细看了下主函数,发现有一部把输入进行MD5加密的过程,所以需要解MD5,直接在线解就可以了
给的是c代码,从结果进行反推
首先is_ok必须为1,那么下图两个字符串必须相等
int is_ok = 1;
for (i = 0; i < FLAG_LEN; i++) {
if (generated_string[i] != just_a_string[i]) //必须相等
return 0;
}
return 1;
要想相等则result必须为0
for (i = 0; i < FLAG_LEN; i++) {
pthread_join(*(thread+i), &result);
generated_string[i] = *(char *)result + just_a_string[i]; //ressul必须为0
free(result);
free(arguments[i]);
}
则checking函数反回为0
char *arguments[20];
for (i = 0; i < FLAG_LEN; i++) {
arguments[i] = (char *)malloc(3*sizeof(char));
arguments[i][0] = first_letter; //生成的随机数
arguments[i][1] = differences[i]; //给定的字符串
arguments[i][2] = user_string[i]; //你的输入
pthread_create((pthread_t*)(thread+i), NULL, checking, arguments[i]);
}
则**(argument[0]+argument[1])与argument[2]必须相同**
void * checking(void *arg) {
char *result = malloc(sizeof(char));
char *argument = (char *)arg;
*result = (argument[0]+argument[1]) ^ argument[2]; //相同则异或为0
return result;
}
然后我们知道arg[0]是产生的随机数,范围在97到97+26之间,开始爆破
a = [0, 9, -9, -1, 13, -13, -4, -11, -9, -1, -7, 6, -13, 13, 3, 9, -13, -11, 6, -7]
for i in range(97,123):
flag = ''
for j in range(len(a)):
flag += chr(i + a[j])
print(flag)
文件夹里三个文件,exe跑一下发现闪退,kali下跑如下图
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IHkHQWMN-1593054216969)(https://i.loli.net/2020/06/01/Utxuy2CkMQPe4X7.png)]
给的这也不知道是啥,IDA查看一下,对应上图可以看出输出逻辑,但好像并没有什么用
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sewwWYAT-1593054216970)(https://i.loli.net/2020/06/01/MBA7rUWn8GE3Cja.png)]
好的没思路这个时候是时候shift+f12走一波了
发现了一个存在但是没有输出显示的黑色神秘星系,然后一路交叉引用
好家伙,都是赋值操作,建议彻查,这里记录下结尾处的地址,然后去OD中调试
这里额外说一下(因为当初这个问题困扰了我),如何快速查看地址,如下图进入text view就OK了
选中,右键-在数据窗口中跟随-内存地址,然后运行
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-ttEXD9oa-1593054216974)(https://i.loli.net/2020/06/01/apnPxU5dftAQ2wJ.png)]
字符串直接提交即可。
其实挺简单一个题,但刚开始自己没想到,一通胡乱分析猛如虎,之后然后看了题解才明白,捞的不谈。
直接进入winmain函数,有一些套话废话看不懂问题不大,直接分析主要矛盾就OK。
使之判断成功就行,注意V11的定义。
这道题刚开始我错误的理解了atoi和itoa导致纠结了不少时间,建议仔细读一下函数说明。然后注意这个string给的定义是char
主要注意事项如下图:
大体流程:程序把输入先进行一系列判断,然后将输入打乱顺序后和给定的字符串进行比较,一些对解题没用的if语句可以选择性跳过,详见注释
int __cdecl __noreturn main(int argc, const char **argv, const char **envp)
{
__pid_t v3; // eax
size_t v4; // rax
ssize_t v5; // rbx
bool v6; // al
bool bCheckPtrace; // [rsp+13h] [rbp-1BDh]
ssize_t numRead; // [rsp+18h] [rbp-1B8h]
ssize_t numReada; // [rsp+18h] [rbp-1B8h]
char bufWrite[200]; // [rsp+20h] [rbp-1B0h]
char bufParentRead[200]; // [rsp+F0h] [rbp-E0h]
unsigned __int64 v12; // [rsp+1B8h] [rbp-18h]
v12 = __readfsqword(0x28u);
bCheckPtrace = detectDebugging();
if ( pipe(pParentWrite) == -1 )
exit(1);
if ( pipe(pParentRead) == -1 )
exit(1);
v3 = fork();
if ( v3 != -1 )
{
if ( v3 )
{
close(pParentWrite[0]);
close(pParentRead[1]);
while ( 1 )
{
printf("Input key : ");
memset(bufWrite, 0, 0xC8uLL);
gets(bufWrite); // 输入
v4 = strlen(bufWrite);
v5 = write(pParentWrite[1], bufWrite, v4);// 把bufwrite写入pparentwrite
if ( v5 != strlen(bufWrite) )
printf("parent - partial/failed write");// v5==bufwrite长度则表示无差错写入
do
{
memset(bufParentRead, 0, 0xC8uLL);
numReada = read(pParentRead[0], bufParentRead, 0xC8uLL);// 从pParentRead中读取长度为0xc8长度的数据到bufParentRead
v6 = bCheckPtrace || checkDebuggerProcessRunning();
if ( v6 )
{
puts("Wrong !!!\n"); // 反反编译
}
else if ( !checkStringIsNumber(bufParentRead) )
{
puts("Wrong !!!\n");
}
else
{
if ( atoi(bufParentRead) ) // 非空&可以转换为整形
{
puts("True");
if ( close(pParentWrite[1]) == -1 )
exit(1);
exit(0);
}
puts("Wrong !!!\n");
}
}
while ( numReada == -1 );
}
}
close(pParentWrite[1]);
close(pParentRead[0]);
while ( 1 )
{
memset(bufParentRead, 0, 0xC8uLL);
numRead = read(pParentWrite[0], bufParentRead, 0xC8uLL);
if ( numRead == -1 )
break;
if ( numRead )
{
if ( childCheckDebugResult() )
{
responseFalse();
}
else if ( bufParentRead[0] == '{' )
{
if ( strlen(bufParentRead) == 42 )
{
if ( !strncmp(&bufParentRead[1], "53fc275d81", 0xAuLL) )
{
if ( bufParentRead[strlen(bufParentRead) - 1] == '}' )
{
if ( !strncmp(&bufParentRead[31], "4938ae4efd", 0xAuLL) )
{
if ( !confuseKey(bufParentRead, 42) )// 关键乱序函数
{
responseFalse();
}
else if ( !strncmp(bufParentRead, "{daf29f59034938ae4efd53fc275d81053ed5be8c}", 0x2AuLL) )
{
responseTrue();
}
else
{
responseFalse();
}
}
else
{
responseFalse();
}
}
else
{
responseFalse();
}
}
else
{
responseFalse();
}
}
else
{
responseFalse();
}
}
else
{
responseFalse();
}
}
}
exit(1);
}
exit(1);
}
关键key混淆函数:
strncpy(szPart1, szKey + 1, 0xAuLL); // 除开首位的'{'和'}',其余每10位一组
strncpy(szPart2, szKey + 11, 0xAuLL);
strncpy(szPart3, szKey + 21, 0xAuLL);
strncpy(szPart4, szKey + 31, 0xAuLL);
memset(szKey, 0, 0x2AuLL);
*szKey = '{'; // 加上首部
strcat(szKey, szPart3); // 顺序为3,4,1,2。还原就可以了
strcat(szKey, szPart4);
strcat(szKey, szPart1);
strcat(szKey, szPart2);
szKey[41] = '}'; // 加上尾部
直接看程序
printf("Gimme: ", argv, envp);
__isoc99_scanf((__int64)"%d", (__int64)&v4); // 给一个输入,为int
not_the_flag(v4); // 关键判断
not_the_flag
if ( a1 == 42 )
puts("Cipher from Bill \nSubmit without any tags\n#kdudpeh");// 注意格式
else
puts("YOUSUCK"); // 你爬
如果输入为42,就会输出来自bill的密码,又说不要任何标签,所以判断密码就是kdudpeh,然后提交会发现错误,这个时候就体现读题的重要性了,来自sha1的密码,OK,sha1加密
程序运行测试:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-hgSDfAfr-1593054216979)(C:\Users\yi\AppData\Roaming\Typora\typora-user-images\image-20200607181210187.png)]
加密脚本(也可选择在线加密):
import hashlib
a='kdudpeh'
b=hashlib.sha1(a.encode('utf-8'))
print(b.hexdigest())
先说题目大概思路,先一个if判断,成功就输出**“Access granted”**,同时执行puts下面的函数。
进入sub_8048414函数,程序还是很简单的,直接手撕,得到a2:isengard
继续分析,进入sub_8048538,函数的参数用到了上面我们刚得到的a2,大体上这是一个输出过程,猜测输出的就是flag,那对unk_8048760里的数据进行数据提取,然后进行简单的异或后应该
数据提取:IDA里选中数据按shift+e可以进行数据提取,提取的数据有无用的部分,需要进行再提取,比如int四个字节那每四个取第一个就OK,脚本如下(因为python学艺不精,所以用的c++)
#include
#include
using namespace std;
int main()
{
int a[] = { 0x0F, 0x00, 0x00, 0x00, 0x1F, 0x00, 0x00, 0x00, 0x04, 0x00,
0x00, 0x00, 0x09, 0x00, 0x00, 0x00, 0x1C, 0x00, 0x00, 0x00,
0x12, 0x00, 0x00, 0x00, 0x42, 0x00, 0x00, 0x00, 0x09, 0x00,
0x00, 0x00, 0x0C, 0x00, 0x00, 0x00, 0x44, 0x00, 0x00, 0x00,
0x0D, 0x00, 0x00, 0x00, 0x07, 0x00, 0x00, 0x00, 0x09, 0x00,
0x00, 0x00, 0x06, 0x00, 0x00, 0x00, 0x2D, 0x00, 0x00, 0x00,
0x37, 0x00, 0x00, 0x00, 0x59, 0x00, 0x00, 0x00, 0x1E, 0x00,
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x59, 0x00, 0x00, 0x00,
0x0F, 0x00, 0x00, 0x00, 0x08, 0x00, 0x00, 0x00, 0x1C, 0x00,
0x00, 0x00, 0x23, 0x00, 0x00, 0x00, 0x36, 0x00, 0x00, 0x00,
0x07, 0x00, 0x00, 0x00, 0x55, 0x00, 0x00, 0x00, 0x02, 0x00,
0x00, 0x00, 0x0C, 0x00, 0x00, 0x00, 0x08, 0x00, 0x00, 0x00,
0x41, 0x00, 0x00, 0x00, 0x0A, 0x00, 0x00, 0x00, 0x14, 0x00,
0x00, 0x00};
int j = 0;
int x = sizeof(a)/16;
int b[x];
cout<<"length is:"<
提取到了数据,开始写flag脚本
a = [105,115,101,110,103,97,114,100]
b = [15,31,4,9,28,18,66,9,12,68,13,7,9,6,45,55,89,30,0,89,15,8,28,35,54,7,85,2,12,8,65,10,20]
flag = ''
for i in range(len(b)):
flag += chr(b[i] ^ a[i % 8])
print(flag)
或者更简单的方法是,既然判断成功就会输出,并且我们已经知道输入了,那直接在kali中运行
这题说是逆向确实有些勉强了,但是了解到了一点xxd指令(皮毛)
首先用exeinfope或者file指令查看文件类型:ASCII text, with CRLF line terminators
拖进sublime查看一下
00400080 68 66 6C 00 00 48 BF 01 00 00 00 00 00 00 00 48
00400090 8D 34 24 48 BA 02 00 00 00 00 00 00 00 48 B8 01
004000A0 00 00 00 00 00 00 00 0F 05 68 61 67 00 00 48 BF
004000B0 01 00 00 00 00 00 00 00 48 8D 34 24 48 BA 02 00
004000C0 00 00 00 00 00 00 48 B8 01 00 00 00 00 00 00 00
004000D0 0F 05 68 7B 70 00 00 48 BF 01 00 00 00 00 00 00
004000E0 00 48 8D 34 24 48 BA 02 00 00 00 00 00 00 00 48
004000F0 B8 01 00 00 00 00 00 00 00 0F 05 68 6F 70 00 00
00400100 48 BF 01 00 00 00 00 00 00 00 48 8D 34 24 48 BA
00400110 02 00 00 00 00 00 00 00 48 B8 01 00 00 00 00 00
00400120 00 00 0F 05 68 70 6F 00 00 48 BF 01 00 00 00 00
00400130 00 00 00 48 8D 34 24 48 BA 02 00 00 00 00 00 00
00400140 00 48 B8 01 00 00 00 00 00 00 00 0F 05 68 70 72
00400150 00 00 48 BF 01 00 00 00 00 00 00 00 48 8D 34 24
00400160 48 BA 02 00 00 00 00 00 00 00 48 B8 01 00 00 00
00400170 00 00 00 00 0F 05 68 65 74 00 00 48 BF 01 00 00
00400180 00 00 00 00 00 48 8D 34 24 48 BA 02 00 00 00 00
00400190 00 00 00 48 B8 01 00 00 00 00 00 00 00 0F 05 68
004001A0 7D 0A 00 00 48 BF 01 00 00 00 00 00 00 00 48 8D
004001B0 34 24 48 BA 02 00 00 00 00 00 00 00 48 B8 01 00
004001C0 00 00 00 00 00 00 0F 05 48 31 FF 48 B8 3C 00 00
004001D0 00 00 00 00 00 0F 05
这里可以写脚本把16进制转为ASCII码,这里介绍学到的用xxd指令转换
关于xxd指令的内容,我没有找到一篇说写的很好的,这里放两篇勉强能冲的:
1.https://www.sohu.com/a/341144249_610671
2.https://blog.csdn.net/xingyyn78/article/details/79438853
到这里可以直接进行分析了,规律还是很明显的,或者嫌看着难受可以先选出可打印字符:
a = "@�hflH�H@��4$H�H�@�hagH�@�H�4$H�@�H�@�h{pH�@�H�4$H�H@��hop@H�H�4$H�@H�@ hpoH�@0H�4$H�@@H�hpr@PH�H�4$@`H�H�@phetH�@�H�4$H�@�H�h@�}H�H�@�4$H�H�@�H1�H�<@� "
flag = ''
for i in range(len(a)):
if(ord(a[i]) >= 32 and ord(a[i]) <= 125):
flag += a[i]
print(flag)
得到字符串:
@hflHH@4$HH@hagH@H4$H@H@h{pH@H4$HH@hop@HH4$H@H@ hpoH@0H4$H@@Hhpr@PHH4$@`HH@phetH@H4$H@Hh@}HH@4$HH@H1H<@
很明显可以观察到我们需要的是h后面H之前(有时是@)的字符
得到flag:
flag{poppopret}
然后交了很多次发现不对,去掉flag{}即可
这里额外介绍一个一条指令秒的题解,连接https://github.com/ctfs/write-ups-2014/tree/master/tinyctf-2014/tt3441810
交叉引用字符串pass!找到关键函数,函数逻辑是对输入的字符按ascii码值进行变换,变换后的结果作为字符串abcd…XYZ的索引。
脚本:
a = 'abcdefghiABCDEFGHIJKLMNjklmn0123456789opqrstuvwxyzOPQRSTUVWXYZ'
b = 'KanXueCTF2019JustForhappy'
print(len(b))
print(len(a))
f = []
for i in range(len(b)):
for j in range(len(a)):
if(a[j] == b[i]):
f.append(j)
for i in range(len(f)):
if(f[i]>=10 and f[i] <= 35):
f[i] += 87
elif(f[i]>=0 and f[i] <= 9):
f[i] += 48
elif(f[i]>=36 and f[i] <= 61):
f[i] += 29
flag = ''
for i in range(len(f)):
flag += chr(f[i])
print(flag)
写脚本的时候我犯了个错,在连续判断的if语句那,我第一次没有用elif而是连用了三个if,这样的后果就是进行第一个if判断后,修>改后的值可能会满足剩下的if判断条件,造成多次判断成功的情况,当时一直没发现,这里感谢好兄弟ld1ng的指正。
一觉起来打开一看,首页莫名其妙又多了个没写的,本来不想写的,强迫症,没办法
逻辑倒是简单,直接上脚本
key = 0x12
flag = ''
cmpa = [0x69, 0x7A, 0x77, 0x68, 0x72, 0x6F, 0x7A, 0x22, 0x22, 0x77,
0x22, 0x76, 0x2E, 0x4B, 0x22, 0x2E, 0x4E, 0x69]
for i in range(0,6):
x = chr((cmpa[3 * i] ^ key) - 6) #注意异或的优先级
y = chr((cmpa[3 * i + 1] ^ key) + 6)
z = chr((cmpa[3 * i + 2] ^ key) ^ 6)
flag += x
flag += y
flag += z
print(flag)
这里注意到异或的优先级,我刚开始没加上括号,然后结果不对,之后我才意识到异或的优先级好像比减和加要低。
这里先感谢两位师傅ld1ng和iyzyi对我写这道题提供的帮助,有你们真好☺️。
这个题让我很疑惑看不懂的地方就在于其独特的参数表示方式,闻所未闻(主要是孤陋寡闻)。
首先这个题是有壳的,不过upx-d可以直接脱,但是脱壳之后的程序运行就闪退,原因是程序开启了ASLR ,但是使用的确实绝对地址,所以把ASLR关掉之后就可以了。这里我也是借鉴的网上大佬的博客,链接:
https://blog.csdn.net/qq_43547885/article/details/104412656
以及http://blog.eonew.cn/archives/749
我们直接进入关键函数,要读懂程序必须读懂** * v1 = byte_402FF8[(char)v1[(_DWORD)v4]] ** 这句代码,值得注意的是不管v1,v4还是a1全都是指针形式,对于v4 = (a1 - v1),可以把式子改写为(v1 + v4) = a1,而* (v1[v4])可以理解为* (v1 + v4),所以* (v1 + v4) 指向了 a1,这样随着while循环中v1的递增就完成了对 a1 的递增。把递增得到的输入的字符作为索引,在byte_402ff8里查表转换,转换后的值存入v1。
unsigned int __cdecl sub_401000(const char *a1)
{
_BYTE *v1; // ecx
unsigned int v2; // edi
unsigned int result; // eax
const char *v4; // ebx
v2 = 0;
result = strlen(a1);
if ( result )
{
v4 = (const char *)(a1 - v1); // 改写为:v1 + v4 = a1
// 随着v1的递增,遍历整个输入
do
{
*v1 = byte_402FF8[(char)v1[(_DWORD)v4]]; // *(v1[v4]) 相当于 *(v1 + v4)
// 把输入的字符作为索引在byte_402ff8里查表转换
++v2;
++v1;
result = strlen(a1);
}
while ( v2 < result );
}
return result;
}
然后对于byte_402ff8里存的字符,但是点进去发现是这样:
刚开始有点懵,因为byte_402ff8开头的那后面跟了一堆不知道是啥的东西,但是可以发现地址没变,应该是一些声明啥啥啥的,然后尝试直接选取进行数据提取发现没问题 ,如图。
然后网上有另一种思路,ASCII码的可见字符在32以后,开始地址为0x00402ff8,加上32恰好为0x403018,这样就可以直接看到数据了。
这样对输入的转换就完成了,下一步就是字符串比较,这里要注意到,调用函数的时候传入的参数只有v6,但比较的时候却是和v4比较,程序看不出来端倪,我们看一下汇编,在调用函数之前,把v4的地址送如了ecx,如下图:
然后进入被调用的函数,发现v1恰好就是ecx,而转换后的值被保存在了v1,而v1又是v4,这样转换后的值就保存在了v4,如下图:
到这里剩下的部分就简单了,把v4和给定的字符串进行判定,逆推一下就是找到字符串DDCTF{reverseME}在byte_402ff8中所对应的位置,把位置索引转换为字符,即为输入,脚本如下:
a = [0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x4E, 0xE6,
0x40, 0xBB, 0xB1, 0x19, 0xBF, 0x44, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF, 0xFF, 0xFF, 0xFE, 0xFF, 0xFF, 0xFF, 0x01, 0x00,
0x00, 0x00, 0x7E, 0x7D, 0x7C, 0x7B, 0x7A, 0x79, 0x78, 0x77,
0x76, 0x75, 0x74, 0x73, 0x72, 0x71, 0x70, 0x6F, 0x6E, 0x6D,
0x6C, 0x6B, 0x6A, 0x69, 0x68, 0x67, 0x66, 0x65, 0x64, 0x63,
0x62, 0x61, 0x60, 0x5F, 0x5E, 0x5D, 0x5C, 0x5B, 0x5A, 0x59,
0x58, 0x57, 0x56, 0x55, 0x54, 0x53, 0x52, 0x51, 0x50, 0x4F,
0x4E, 0x4D, 0x4C, 0x4B, 0x4A, 0x49, 0x48, 0x47, 0x46, 0x45,
0x44, 0x43, 0x42, 0x41, 0x40, 0x3F, 0x3E, 0x3D, 0x3C, 0x3B,
0x3A, 0x39, 0x38, 0x37, 0x36, 0x35, 0x34, 0x33, 0x32, 0x31,
0x30, 0x2F, 0x2E, 0x2D, 0x2C, 0x2B, 0x2A, 0x29, 0x28, 0x27,
0x26, 0x25, 0x24, 0x23, 0x22, 0x21, 0x20, 0x00, 0x00, 0x00,
0x00, 0x00]
b = 'DDCTF{reverseME}'
flag = ''
for i in range(len(b)):
flag += chr(a[ord(b[i])])
print(flag)