-
Notifications
You must be signed in to change notification settings - Fork 71
/
Copy pathea_emu_client.py
196 lines (140 loc) · 4.66 KB
/
ea_emu_client.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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
import socket
from api_funcs import *
from cPickle import dumps, loads
from ea_UI import Emulate_UI
from ea_utils import QtWidgets, ea_warning, get_bits, root_dir
from idaapi import *
from os import name
from subprocess import Popen
from time import sleep
# Importing Unicorn Emulator directly into the IDAPython environment causes instability in IDA (random crashes ect.)
# As a result, Unicorn emulator is decoupled from IDA and runs as a seperate process communicating with IDA using a local socket (port 28745)
# The following client code runs within IDAPython and ships emulation requests to ea_emu_server which is a pure Python process
class Hook(DBG_Hooks):
def __init__(self):
DBG_Hooks.__init__(self)
def dbg_bpt(self, tid, ea):
send()
return 0
def dbg_step_into(self):
send()
return 0
def dbg_step_until_ret(self):
send()
return 0
def dbg_step_over(self):
send()
return 0
def send(addr=None, code=None):
if get_process_state() != -1:
ea_warning("Process must be paused/suspended")
else:
if not addr:
flags = None
addr = get_rg("RIP")
bp = get_bp(addr,False)
if bp:
flags = bp.flags
bp.flags = 2
update_bpt(bp)
code = dbg_read_memory(addr & 0xfffffffffffff000, 0x1000)
if flags:
bp.flags = flags
update_bpt(bp)
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
try:
s.connect((TCP_IP, TCP_PORT))
except socket.error:
launch_server()
sleep(0.5)
s.connect((TCP_IP, TCP_PORT))
s.send(dumps(("emu", (addr,code, get_bits(), server_print))))
error = False
while True:
data = s.recv(BUFFER_SIZE)
if not data: break
func, args = loads(data)
if func == "result":
break
if func == "error":
ea_warning(args)
error = True
break
s.send(dumps(globals()[func](*args)))
s.close()
if not error and annotate:
rip = get_rg("RIP")
if rip in args:
del args[rip]
for c, v in args.items():
v = [i for i in v if i[0] not in ("rip", "eip")]
comment = GetCommentEx(c, 0)
if v:
annotation = " ".join(a + "=" + hex(b).replace("L", "") for a, b in v)
if comment and "e:" in comment:
comment = comment[:comment.find("e:")].strip(" ")
MakeComm(c, (comment if comment else "").ljust(10) + " e: " + annotation)
else:
if comment and "e:" in comment:
comment = comment[:comment.find("e:")].strip(" ")
MakeComm(c, (comment if comment else "").ljust(10) + " e: " + "No reg changes")
def launch_server():
# Launch emulation server as a seperate process (see top for details why)
global server_running
Popen("python \"%sea_emu_server.py\"" % root_dir, shell=True if name=="posix" else False)
server_running = True
def close_server(arg):
global server_running
server_running = False
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
s.connect((TCP_IP, TCP_PORT))
s.send(dumps(("quit", (0, 0, 0, 0))))
def ea_emulate():
global form
global a
global server_running
if not server_running:
launch_server()
a = QtWidgets.QFrame()
form = Emulate_UI()
form.setupUi(a)
if hooked:
form.checkBox.click()
form.checkBox.stateChanged.connect(toggle_hooking)
form.pushButton.clicked.connect(a.close)
form.pushButton_2.clicked.connect(send)
form.checkBox_3.stateChanged.connect(set_annotate)
form.checkBox_2.stateChanged.connect(set_server_print)
# a.setWindowFlags(a.windowFlags() | QtCore.Qt.WindowStaysOnTopHint)
a.closeEvent = close_server
a.show()
def toggle_hooking(state):
global h
global hooked
if state:
if not hooked:
h = Hook()
h.hook()
hooked = True
else:
h.unhook()
hooks = False
def set_annotate(state):
global annotate
annotate = True if state else False
def set_server_print(state):
global server_print
server_print = True if state else False
TCP_IP = '127.0.0.1';
TCP_PORT = 28745;
BUFFER_SIZE = 0x4000;
comments = []
file_name = None
h = None
hooked = False
form = None
a = None
bp = None
server_running = False
annotate = True
server_print = True