xctf攻防世界 CRYPTO高手进阶区 streamgame2

0x01. 进入环境,下载附件

题目给出的一个python文件和key,大致看了看代码,key其实就是生成的文件,我们要利用key的格式去想办法还原flag

0x02. 代码分析

0x02_1. flag的格式
from flag import flag
assert flag.startswith("flag{")
assert flag.endswith("}")
assert len(flag)==27

这段代码第一行表示从flag文件中,导入了flag。这不是废话么!
剩下的2-4行代码我们可以知道,flag的形式为flag{***},且flag的长度为27位,那么括号中的长度为21位。

0x02_2. 参数分析
R=int(flag[5:-1],2)
mask=0x100002

R取了flag中间部分并且以二进制形式转换为数字,那么我们就知道了flag括号内的东西一定是0或者1,我们可以调出idle测试一下

代码中的 R R R 取值来自flag,且int(a,2)函数参数表示字符参数a是2进制的情况下,转成10进制,如下:

print(int('101',2)) # 输出为 5

那么可以推出来,flag一定为0101的字符存在形式,也就是说,一共有21位的0和1组合字符串,那么可能性为 2 21 2^{21} 221 种可能性。

0x02_3. 加密函数
def lfsr(R,mask):
    output = (R << 1) & 0xffffff # 字符R逻辑左移一位并与0xffffff做与操作
    i=(R&mask)&0xffffff # R和mask做与操作后与0xffffff做与操作
    lastbit=0
    while i!=0:
        lastbit^=(i&1) # 求i和1的与操作再与lastbit异或
        i=i>>1
    output^=lastbit
    return (output,lastbit)

回头看lfsr函数,最终目的得到一个加密输出

0x02_4. key文件的由来
f=open("key","ab")
for i in range(12):
    tmp=0
    for j in range(8):
        (R,out)=lfsr(R,mask)
        tmp=(tmp << 1)^out
    f.write(chr(tmp))
f.close()

第一行打开了个key文件并可写,最终i循环12次,每一次执行8个位加密和异或操作,得到一个字节的tmp,最终将其写入文件f中。

那么我们可以知道key文件的长度一定为12位。尝试用winhex打开key文件可以瞅瞅,如图:
在这里插入图片描述
果然是12字节的数据。

0x03. 解题代码

参考大佬们的wp:https://blog.csdn.net/weixin_44604541/article/details/111357774?utm_medium=distribute.pc_relevant.none-task-blog-2defaultbaidujs_baidulandingword~default-0.pc_relevant_aa&spm=1001.2101.3001.4242.1&utm_relevant_index=2

# 原始加密函数
def lfsr(R, mask):
    output = (R << 1) & 0xffffff
    i = (R & mask) & 0xffffff
    lastbit = 0
    while i != 0:
        lastbit ^= (i & 1)
        i = i >> 1
    output ^= lastbit
    return output, lastbit

# 检测函数,判断加密后的结果list2是否和原始list1相同
def check(list1, list2):
    for i in range(12):
        if list1[i] != list2[i]:
            return False
    return True

# 拿到key文件的数据
with open('pic/key', 'rb') as file:
    f = file.read()
s_list = []
for c in f:
    s_list.append(c)

mask = 0x100002

# 一共有2^21种排列组合
for i in range(1 << 21):
    print(i)
    tmp_list = []
    R = i
    for j in range(12):
        tmp = 0
        for k in range(8):
            (R, out) = lfsr(R, mask)
            tmp = (tmp << 1) ^ out
        tmp_list.append(tmp)
	# 如果key和最终加密的结果相同,那么表示找到了flag
    if check(s_list, tmp_list):
        print(bin(i))
        break

最终答案为:flag{110111100101001101001}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

l8947943

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值