队伍成绩:
656
pt/4
kill/67
名/203
有解/692
登录/825
报名
个人成绩:346
pt/1
kill/134
名/465
有解/2161
登录/2864
报名
题目比较变态,另外因为头一天还要打河南工控的,所以没看太多的题目,其他题目只能赛后学习了,拿个缝合杂项的二血已经心满意足了。(不过有一说一祥云杯这最后垂直上分属于是老传统了,晚上8点还只有2解的题,9点就10解了,看来大家都很强😅)
Misc
super_electric[二血]
这题主要难度在中间的逆向上,不知道出题人咋想的,非要在外面套一层流量分析,工控Windows是吧(关键是这题Win7还跑不起来,哪有上Win10的工控机子😅)
拿到题目,是个流量分析题,先看下流量分布。
可以看出来,IPv4-TCP-MMS
几乎占了绝大部分,而其他的协议占比也没有超出合理范围,那么这个流量包是让分析MMS
协议基本上是板上钉钉的事了。
那就筛选一下mms
。
发现从头到尾都有好多的confirmed-RequestPDU
(这个消息类型的过滤器是mms.confirmedServiceRequest
),而其他的几个消息类型又似乎是没啥用的那种(用mms && !mms.confirmedServiceRequest
过滤后可以看出确实没啥有用信息)。
那就过滤一下mms.confirmedServiceRequest
呗。
发现除了开头那条是LLN0$CO$FunEna1$SBOw
,后续不是LLN0$CO$FunEna2$Oper
就是LLN0$CO$FunEna1$Oper
,那目标已经很明确了,肯定是后两者咯。
先是mms.itemId == "LLN0$CO$FunEna2$Oper"
看了下,发现从头到尾一直在传同一个值:172.20.1.23
。
所以LLN0$CO$FunEna2$Oper
大概是没啥用了,那就去看LLN0$CO$FunEna1$Oper
呗。
这熟悉的4d5a9000
开头,这不exe
吗?那就tshark
提取一下呗(还是老老实实用cmd
跑tshark
比较好,pwsh
不知道为啥一直会报tshark: "$" was unexpected in this context.
)。
tshark -r .\super_electric.pcapng -Y "mms.itemId == \"LLN0$CO$FunEna1$Oper\"" -T fields -e "mms.octet_string" > data.txt
提取结果是hex
的:
那么写个脚本输出成exe
吧。
data = b""
with open('data.txt', 'r') as f:
for line in f:
data += bytes.fromhex(line.strip())
with open("test.exe", 'wb') as f:
f.write(data)
那就开始逆向呗。
打开发现有窗口,那就找一下文本先。
发现有反调试相关函数名,不过应该问题不大,先去看看crackme
那块有啥调用吧。
应该是在创建窗口或者应用啥的,那么上面的sub_43CB50
应该就是主函数了,跟过去看看。
有个aC96f278c370c43
常量,好像上面看字符串的时候见过,跟回去看看附近还有没有啥。
有一堆不知道是啥的常量,直接A
一下发现就是窗口上的各种文本和消息。(有些文本被ida解析成函数了,需要U
一下然后A
)
那就回去看看哪里触发了那个恭喜
。
所以关键还是那个C96F278C370C43299F1EE98257CCBD8A
,只不过这边dword_449CB0
函数没解析出来是啥,过去看看定义发现应该是Win32 Api
。
那就x32dbg
调试一下看看,因为这玩意我记得能识别Win32 Api
。
到调用点43ccbd
看看,发现这个函数其实是memcmp
。
那有没有可能,输入的就是这一串,因为上面没见到啥复杂的处理逻辑或者其他调用。
于是输入了看看,发现确实是。
但是点确定
之后就退出了,有点离谱...
于是命令行启动看看,猜测会不会是在STDOUT
输出了东西。
结果输入完点确定
后发现,根本不是正常退出,是异常了...
去看了下系统日志,是VC14
翻车了,啥情况啊。然后想起来之前打绿城杯线下有一题也是这个类似的情况(不过那题不是VC14
翻的,但是都是最后环节过几秒钟异常)。
绿城杯那题是因为作者写了个按条件触发的自解压的壳,然后因为我电脑环境问题好像触发了空指针。这题也是触发了空指针,难道这题也是自己写的壳?结合上面pdb
路径里面带了个upacker
,感觉真有可能。
不过当时为了保险起见,这种出现了异常的情况,肯定是先问问赛务比较靠谱。
结果赛务这回答,当时刚开始觉得好有道理,然后管理员运行、兼容性运行都给试了一遍,还是问题依旧。
然后突然醒悟,这回答不是命令行逆向题的标准回答吗?把我当小白了...赛务果然不靠谱。
然后后来我又小怼了一下,看出来回复确实才变成出题人的回复了。
但是看这意思是,出题人觉得正常情况下,点完确认是可以到下一阶段开始命令行交互的,然后大概会有在STDOUT
写东西然后马上关闭?
那我这属于是环境问题导致完全不知道有下一阶段呗...搁这耽误了快两个小时是真离谱。
算了算了,确认有壳就行,找脱壳点吧。回到上面那个恭喜
逻辑。
相比于没输入正确,多了个调用,那就去x32dbg
看看是啥函数的调用。
是个SendMessageA
,然后消息是16
,这不是WM_CLOSE
嘛...
那看来这个窗口应该是没啥脱壳逻辑了,往上追调用堆栈看看后续有没有吧。
然后一路追直接到了start
函数。
其中sub_43CD50
是创建窗体的函数,那么下面还一堆内存处理的,咋看咋像脱壳,然后最后有个内存块当函数调用的,就更像是脱壳完后的入口地址了。
于是懒得看脱壳逻辑了,直接跑到最后这个调用点43d4a6
然后Scylla
一键脱了吧。
结果压根没跑到这个地方,触发啥异常了,应该是哪个地方有反调试插桩。那就从sub_43CD50
的调用结束后下个断点然后单步跟呗。
结果连sub_43CD50
还没结束就触发反调试了。
那就只能把这个异常给过滤掉了。
然后发现在调用43d2a0
会触发另一个异常。
上ida一看,发现是个反调试,直接nop。
然后跟过去发现真有代码。
直接Scylla
一把梭。
不过无论是dump后的还是修复后的都依旧跑不起来,感觉还是踩到那个环境问题了。
不过无所谓,ida直接找到了真实的main函数,这就方便多了。
似乎是需要输入什么东西然后异或0x89
后和byte_42D624
比较,但是这个输入的后续就没有使用了,而且咱们跑不起来,也就无所谓需要输入啥了,反正没地方输入。
然后下面开始循环异或下标,范围是[0, 717)
,那就写个脚本看看这里解出来的结果是啥。
b = bytearray(bytes.fromhex("66736D6E2446747E787D65254F647E677563327A796579656C395B5E4F177772504E505704474F49495A494245274742405E4047145D574450555359365B4C502D612A2B2C652F2A3826383F6C2B222E375B332027302423783F363A3B06646A3D415F5E4442000B090E114C4C0C000B50171E12132E5B4642245A46415D5902A78BE9E6FDA5BBA7EAAEBEEFB5ECB9BFA0A1A3A3A0A6A1BDB2B3BD91F0BDA3BFCCC4CC8BCFC0DF8EA2C4CFD8DFCCC9CA908C92D193F1D997C1D6CF9BD9CBDBCDE0A7A7A6A8E9E6A1ADACA6EBBFA2EEBFB1A1B7A1F4A1BEBEB6F5FA97B5B6BBFF81C18A8C919683C7878FCA888D9F8A9CDCD1BD9D91D5949B978EDA9D8E9293DF6360746A6A62266E662E2A202C6F67617162717A7D3B6379707C6277757B67374840514B484C44095B414B19191B064455481B1D5C504E53515E5F48481517161B7B7373194F2F3168746A2D202C2914656B7F62095F3B322B2A3B3C397D637F0D041110050203474349081218081D47581D525E5419131950141F080F1C191AA9A1A7A3E8ACA6ADA8EAE2F9A4E1AEA2B0FDF7FDBCF8F3E4EBF8FDFEB5BDBBBFCC888E83C1CBC5C8CCC0C4CC8C908E88C5C5D49E8C929FBDD9DCC99B819DFFFA93EFACA6B3EDADA2B1E5EA8A899EE0829F95978C979795FBF8B0ACF2D6ADACB68E95CA818D8B87948B8083C58488968399978BDB959085D99D979989858D8AD76D6471706562632E21200028262724253A3B38393E3F3C3D32333031363734350A0B08090E0F0C0D02030001060704051A1B18191E1F1C1D12131011161714156A6B68696E6F6C6D62636061666764657A7B78797E7F7C7D72737071767774754A4B48494E4F4C4D42434041464744455A5B58595E5F5C5D5253505156575455AAABA8A9AEAFACADA2A3A0A1A6A7A4A5BABBB8B9BEBFBCBDB2B3B0B1B6B7B4B58A8B91C5C6C49093C9CD9DC99B95989886D4868580868F828980838F8E898D8FF2A3F0F2A6F7A4F6FFADA8F9C6"))
for i in range(len(b)):
b[i] = 0xff & (b[i] ^ i)
print(b)
发现蹦出来个python
代码,这就是上面说的You Got it!\nNow, try to resolve this crypto...\n
?这套娃...
这已知key
高位,然后key
又被sha256
了一波扔到m
结尾去了,然后message = message + bytes((l - len(message) % l) * chr(l - len(message) % l), encoding = 'utf-8')
就是padding
操作,最后flag当成iv
用,加密结果只告诉了低位。
这就是ezAES这题改的。
所以咱就按照这个wp的思路来。
首先AES/CBC/PKCS7Padding
的特性有:
- 分段加密,每段16字节,上一段的加密结果会当作下一段的
iv
去使用 - 用来填充的数据每字节内容和其长度相等,如
01
、0202
、030303
这种
那么咱们就可以将加密结果的最后16字节和倒数第二个16字节取出来,即78676e464395199424302b21b2b17db2
和**********************3fba64ad7b
。
然后将**********************3fba64ad7b
当成iv
去解密78676e464395199424302b21b2b17db2
,至于key
,爆破就完事。
至于爆破是否正确的判断,当然是利用padding
特性了,根据题目的加密脚本我们可以知道m
的长度是84+10=94
,也就是说需要填充0202
,那咱们拿上面wp里的脚本改改就行:
import binascii
from Crypto.Cipher import AES
def decrypt(message, passphrase, iv):
aes = AES.new(passphrase, AES.MODE_CBC, iv)
return aes.decrypt(message)
def find_key():
keytmp = '4d9a700010437{}{}{}'
for c1 in "0123456789abcdef":
for c2 in"0123456789abcdef":
for c3 in "0123456789abcdef":
tmp = decrypt(
binascii.unhexlify('78676e464395199424302b21b2b17db2'),
keytmp.format(c1, c2, c3).encode('utf8'),
binascii.unhexlify('0' * 22 + '3fba64ad7b'),
)
if 2 == tmp[-1] == tmp[-2]:
print(keytmp.format(c1, c2, c3))
find_key()
所以key
大概率就是4d9a7000104376fe
了。
然后已知m
和key
求iv
这个属于是密码学题目基本操作了,抄一下上面wp的脚本稍微改一下就行:
import binascii
import hashlib
from Crypto.Cipher import AES
def encrypt(message,passphrase,iv):
aes = AES.new(passphrase, AES.MODE_CBC, iv)
return aes.encrypt(message)
def decrypt(message, passphrase, iv):
aes = AES.new(passphrase, AES.MODE_CBC, iv)
return aes.decrypt(message)
key = b"4d9a7000104376fe"
message = b"Do you ever feel, feel so paper thin, Like a house of cards, One blow from caving in"\
+ binascii.unhexlify(hashlib.sha256(key).hexdigest())[:10]\
+ b"\x02\x02"
IV = b'yellow_submarine'
arbitrary = binascii.hexlify(encrypt(message, key, IV))
encrypted = '******************************************************************************************************************************************************3fba64ad7b78676e464395199424302b21b2b17db2'.replace('*', '0')
encrypted = [encrypted[i:i+32] for i in range(0, len(encrypted), 32)]
arbitrary = [arbitrary[i:i+32] for i in range(0, len(arbitrary), 32)]
def guess_block(flag_block, correct_block, correct_iv):
c1 = decrypt(binascii.unhexlify(flag_block), key, binascii.unhexlify(b'0' * 32))
c2 = decrypt(binascii.unhexlify(correct_block), key, binascii.unhexlify(correct_iv))
result = b''
for i in range(16):
result += bytes([c1[i] ^ c2[i]])
return binascii.hexlify(result)
for i in range(1, len(arbitrary) + 1):
if i == len(arbitrary):
e2 = binascii.hexlify(IV)
else:
e2 = arbitrary[-i - 1]
tmp = guess_block(encrypted[-i], arbitrary[-i], e2)
if i == len(arbitrary):
flag = binascii.unhexlify(tmp)
else:
encrypted[-i - 1] = tmp
print(flag.decode('utf8'))
所以最后flag就出来了。
flag{72713126e9b90eab}