-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathangrit.py
125 lines (106 loc) · 3.89 KB
/
angrit.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
import os
import angr
import claripy
def get_single_active_state(simgr):
active = simgr.stashes['active']
if len(active) == 0:
raise Exception('No active states')
if len(active) != 1:
pcs = [state.regs.pc for state in active]
raise Exception(f'Multiple active states: {pcs}')
state, = active
return state
def step_until_pc(simgr, until):
while True:
state = get_single_active_state(simgr)
pc = state.solver.eval(state.regs.pc)
print(f'pc: 0x{pc:x}')
if pc in until:
return state
simgr.step()
def get_str(state, addr):
s = b''
while True:
b = state.solver.eval(state.mem[addr + len(s)].char.resolved)
if b == 0:
break
s += bytes((b,))
return s.decode()
def step_until_puts(p, simgr, arg):
puts_addr = p.loader.find_symbol('puts').rebased_addr
while True:
state = step_until_pc(simgr, (puts_addr,))
actual = get_str(state, state.regs.a0)
print(f'puts("{actual}")')
if actual == arg:
return state
simgr.step()
def step_until_read(p, simgr):
read_addr = p.loader.find_symbol('read').rebased_addr
return step_until_pc(simgr, (read_addr,))
def step_until_call(simgr):
while True:
state = get_single_active_state(simgr)
pc = state.solver.eval(state.regs.pc)
jk = state.history.jumpkind
print(f'pc: 0x{pc:x}, jk: {jk}')
if jk == 'Ijk_Call':
return state
simgr.step()
def solve(path):
p = angr.Project(path)
read_addr = p.loader.find_symbol('read').rebased_addr
entry_state = p.factory.entry_state()
simgr = p.factory.simgr(entry_state)
step_until_puts(p, simgr, 'Faster > ')
state_at_read = step_until_pc(simgr, (read_addr,))
assert state_at_read.solver.eval(state_at_read.regs.a0) == 0
user_buf_addr = state_at_read.solver.eval(state_at_read.regs.a1)
assert state_at_read.solver.eval(state_at_read.regs.a2) == 0x100
user_bytes = [claripy.BVS(f'x{i}', 8) for i in range(0x100)]
for i, user_byte in enumerate(user_bytes):
state_at_read.mem[user_buf_addr + i].char = user_byte
state_at_read.regs.v0 = 0x100
state_at_read.regs.pc = state_at_read.regs.ra
simgr = p.factory.simgr(state_at_read)
simgr.step()
step_until_call(simgr)
simgr.step()
state_at_checker = step_until_call(simgr)
while True:
pc = state_at_checker.solver.eval(state_at_checker.regs.pc)
print(f'checker pc: 0x{pc:x}')
# some checkers are nonlinear and confuse z3
# try silly solutions first
state_at_checker_silly = state_at_checker.copy()
for i in range(4):
ch_addr = state_at_checker_silly.regs.a0 + i
ch = state_at_checker_silly.mem[ch_addr].char.resolved
state_at_checker_silly.add_constraints(ch < 2)
simgr_silly = p.factory.simgr(state_at_checker_silly)
simgr_silly.step()
simgr_silly.explore(
find=lambda state: state.history.jumpkind == 'Ijk_Call',
avoid=lambda state: state.history.jumpkind == 'Ijk_Ret',
)
if len(simgr_silly.stashes['found']) > 0:
state_at_checker = simgr_silly.stashes['found'][0]
simgr = p.factory.simgr(state_at_checker)
continue
simgr.step()
simgr.explore(
find=lambda state: state.history.jumpkind == 'Ijk_Call',
avoid=lambda state: state.history.jumpkind == 'Ijk_Ret',
)
if len(simgr.stashes['found']) > 0:
state_at_checker, = simgr.stashes['found']
simgr = p.factory.simgr(state_at_checker)
continue
break
return bytes(
state_at_checker.solver.eval(user_byte)
for user_byte in user_bytes
)
if __name__ == '__main__':
for i in os.listdir('samples'):
print(solve(f'samples/{i}').hex())