trezor-core/embed/bootloader/main.c

383 lines
12 KiB
C
Raw Normal View History

2018-02-26 05:06:10 -08:00
/*
* This file is part of the TREZOR project, https://trezor.io/
*
* Copyright (c) SatoshiLabs
*
* 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/>.
*/
#include <string.h>
#include <sys/types.h>
#include "common.h"
#include "image.h"
2017-03-28 04:04:54 -07:00
#include "flash.h"
#include "mini_printf.h"
#include "rng.h"
#include "secbool.h"
#include "touch.h"
2017-03-28 04:04:54 -07:00
#include "usb.h"
#include "version.h"
2017-03-20 07:41:21 -07:00
2017-12-16 16:09:45 -08:00
#include "bootui.h"
#include "messages.h"
// #include "mpu.h"
const uint8_t BOOTLOADER_KEY_M = 2;
const uint8_t BOOTLOADER_KEY_N = 3;
static const uint8_t * const BOOTLOADER_KEYS[] = {
2017-10-12 03:17:04 -07:00
#if PRODUCTION
2017-10-04 11:20:01 -07:00
(const uint8_t *)"\xc2\xc8\x7a\x49\xc5\xa3\x46\x09\x77\xfb\xb2\xec\x9d\xfe\x60\xf0\x6b\xd6\x94\xdb\x82\x44\xbd\x49\x81\xfe\x3b\x7a\x26\x30\x7f\x3f",
2017-10-04 06:55:33 -07:00
(const uint8_t *)"\x80\xd0\x36\xb0\x87\x39\xb8\x46\xf4\xcb\x77\x59\x30\x78\xde\xb2\x5d\xc9\x48\x7a\xed\xcf\x52\xe3\x0b\x4f\xb7\xcd\x70\x24\x17\x8a",
(const uint8_t *)"\xb8\x30\x7a\x71\xf5\x52\xc6\x0a\x4c\xbb\x31\x7f\xf4\x8b\x82\xcd\xbf\x6b\x6b\xb5\xf0\x4c\x92\x0f\xec\x7b\xad\xf0\x17\x88\x37\x51",
#else
(const uint8_t *)"\xd7\x59\x79\x3b\xbc\x13\xa2\x81\x9a\x82\x7c\x76\xad\xb6\xfb\xa8\xa4\x9a\xee\x00\x7f\x49\xf2\xd0\x99\x2d\x99\xb8\x25\xad\x2c\x48",
(const uint8_t *)"\x63\x55\x69\x1c\x17\x8a\x8f\xf9\x10\x07\xa7\x47\x8a\xfb\x95\x5e\xf7\x35\x2c\x63\xe7\xb2\x57\x03\x98\x4c\xf7\x8b\x26\xe2\x1a\x56",
(const uint8_t *)"\xee\x93\xa4\xf6\x6f\x8d\x16\xb8\x19\xbb\x9b\xeb\x9f\xfc\xcd\xfc\xdc\x14\x12\xe8\x7f\xee\x6a\x32\x4c\x2a\x99\xa1\xe0\xe6\x71\x48",
2017-10-04 06:55:33 -07:00
#endif
};
#define USB_IFACE_NUM 0
2017-12-15 06:26:36 -08:00
static void usb_init_all(void) {
2017-03-28 04:04:54 -07:00
static const usb_dev_info_t dev_info = {
.device_class = 0x00,
.device_subclass = 0x00,
.device_protocol = 0x00,
.vendor_id = 0x1209,
.product_id = 0x53C0,
.release_num = 0x0200,
.manufacturer = "SatoshiLabs",
.product = "TREZOR",
.serial_number = "000000000000000000000000",
.interface = "TREZOR Interface",
.usb21_enabled = sectrue,
2017-03-28 04:04:54 -07:00
};
static uint8_t rx_buffer[USB_PACKET_SIZE];
static const usb_webusb_info_t webusb_info = {
.iface_num = USB_IFACE_NUM,
.ep_in = USB_EP_DIR_IN | 0x01,
.ep_out = USB_EP_DIR_OUT | 0x01,
.subclass = 0,
.protocol = 0,
.max_packet_len = sizeof(rx_buffer),
.rx_buffer = rx_buffer,
.polling_interval = 1,
};
2017-03-28 04:04:54 -07:00
usb_init(&dev_info);
ensure(usb_webusb_add(&webusb_info), NULL);
usb_start();
}
2017-12-16 16:09:45 -08:00
static secbool bootloader_usb_loop(const vendor_header * const vhdr, const image_header * const hdr)
2017-04-06 06:53:47 -07:00
{
2017-10-20 07:03:27 -07:00
usb_init_all();
2017-04-27 10:18:57 -07:00
uint8_t buf[USB_PACKET_SIZE];
for (;;) {
int r = usb_webusb_read_blocking(USB_IFACE_NUM, buf, USB_PACKET_SIZE, USB_TIMEOUT);
if (r != USB_PACKET_SIZE) {
continue;
}
uint16_t msg_id;
uint32_t msg_size;
2017-10-26 18:32:26 -07:00
if (sectrue != msg_parse_header(buf, &msg_id, &msg_size)) {
2017-06-14 11:40:31 -07:00
// invalid header -> discard
continue;
}
switch (msg_id) {
case 0: // Initialize
2017-12-16 16:09:45 -08:00
process_msg_Initialize(USB_IFACE_NUM, msg_size, buf, vhdr, hdr);
break;
case 1: // Ping
2017-06-16 06:40:27 -07:00
process_msg_Ping(USB_IFACE_NUM, msg_size, buf);
break;
2017-10-16 11:24:28 -07:00
case 5: // WipeDevice
2018-01-25 04:21:04 -08:00
ui_fadeout();
ui_screen_wipe_confirm();
2018-01-25 04:21:04 -08:00
ui_fadein();
int response = ui_user_input(INPUT_CONFIRM | INPUT_CANCEL);
if (INPUT_CANCEL == response) {
2018-01-25 04:21:04 -08:00
ui_fadeout();
ui_screen_info(secfalse, vhdr, hdr);
2018-01-25 04:21:04 -08:00
ui_fadein();
send_user_abort(USB_IFACE_NUM, "Wipe cancelled");
break;
}
2018-01-25 03:33:09 -08:00
ui_fadeout();
2017-12-16 16:09:45 -08:00
ui_screen_wipe();
2018-01-25 03:33:09 -08:00
ui_fadein();
2017-10-16 11:24:28 -07:00
r = process_msg_WipeDevice(USB_IFACE_NUM, msg_size, buf);
if (r < 0) { // error
2018-01-25 04:21:04 -08:00
ui_fadeout();
2017-12-16 16:09:45 -08:00
ui_screen_fail();
2018-01-25 04:21:04 -08:00
ui_fadein();
usb_stop();
usb_deinit();
return secfalse; // shutdown
2017-10-16 11:24:28 -07:00
} else { // success
2018-01-25 04:21:04 -08:00
ui_fadeout();
2018-01-25 05:44:03 -08:00
ui_screen_done(0, sectrue);
2018-01-25 04:21:04 -08:00
ui_fadein();
usb_stop();
usb_deinit();
return secfalse; // shutdown
2017-10-16 11:24:28 -07:00
}
break;
case 6: // FirmwareErase
2017-06-16 06:40:27 -07:00
process_msg_FirmwareErase(USB_IFACE_NUM, msg_size, buf);
break;
case 7: // FirmwareUpload
2017-10-16 09:03:04 -07:00
r = process_msg_FirmwareUpload(USB_IFACE_NUM, msg_size, buf);
if (r < 0 && r != -4) { // error, but not user abort (-4)
2018-01-25 04:21:04 -08:00
ui_fadeout();
2017-12-16 16:09:45 -08:00
ui_screen_fail();
2018-01-25 04:21:04 -08:00
ui_fadein();
usb_stop();
usb_deinit();
return secfalse; // shutdown
2017-10-16 09:03:04 -07:00
} else
if (r == 0) { // last chunk received
ui_screen_install_progress_upload(1000);
2018-01-25 04:21:04 -08:00
ui_fadeout();
2018-01-25 05:44:03 -08:00
ui_screen_done(4, sectrue);
2018-01-25 04:21:04 -08:00
ui_fadein();
2018-01-25 05:44:03 -08:00
ui_screen_done(3, secfalse);
2018-01-25 04:21:04 -08:00
hal_delay(1000);
2018-01-25 05:44:03 -08:00
ui_screen_done(2, secfalse);
2018-01-25 04:21:04 -08:00
hal_delay(1000);
2018-01-25 05:44:03 -08:00
ui_screen_done(1, secfalse);
2018-01-25 04:21:04 -08:00
hal_delay(1000);
usb_stop();
usb_deinit();
2017-12-16 16:09:45 -08:00
ui_fadeout();
return sectrue; // jump to firmware
2017-10-16 09:03:04 -07:00
}
break;
case 55: // GetFeatures
process_msg_GetFeatures(USB_IFACE_NUM, msg_size, buf, vhdr, hdr);
break;
default:
2017-06-16 06:40:27 -07:00
process_msg_unknown(USB_IFACE_NUM, msg_size, buf);
break;
}
}
2017-04-06 06:53:47 -07:00
}
2017-12-15 06:26:36 -08:00
secbool load_vendor_header_keys(const uint8_t * const data, vendor_header * const vhdr)
{
return load_vendor_header(data, BOOTLOADER_KEY_M, BOOTLOADER_KEY_N, BOOTLOADER_KEYS, vhdr);
}
#define OTP_BLOCK_VENDOR_KEYS_LOCK 2
static secbool check_vendor_keys_lock(const vendor_header * const vhdr) {
uint8_t lock[FLASH_OTP_BLOCK_SIZE];
ensure(flash_otp_read(OTP_BLOCK_VENDOR_KEYS_LOCK, 0, lock, FLASH_OTP_BLOCK_SIZE), NULL);
if (0 == memcmp(lock, "\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF\xFF", FLASH_OTP_BLOCK_SIZE)) {
return sectrue;
}
uint8_t hash[32];
vendor_keys_hash(vhdr, hash);
return sectrue * (0 == memcmp(lock, hash, 32));
}
// protection against bootloader downgrade
2017-12-15 06:26:36 -08:00
#if PRODUCTION
2017-12-13 14:08:15 -08:00
#define OTP_BLOCK_BOOTLOADER_VERSION 1
2017-12-15 06:26:36 -08:00
static void check_bootloader_version(void)
{
uint8_t bits[FLASH_OTP_BLOCK_SIZE];
for (int i = 0; i < FLASH_OTP_BLOCK_SIZE * 8; i++) {
if (i < VERSION_MONOTONIC) {
bits[i / 8] &= ~(1 << (7 - (i % 8)));
} else {
bits[i / 8] |= (1 << (7 - (i % 8)));
}
}
2017-12-13 14:08:15 -08:00
ensure(flash_otp_write(OTP_BLOCK_BOOTLOADER_VERSION, 0, bits, FLASH_OTP_BLOCK_SIZE), NULL);
uint8_t bits2[FLASH_OTP_BLOCK_SIZE];
2017-12-13 14:08:15 -08:00
ensure(flash_otp_read(OTP_BLOCK_BOOTLOADER_VERSION, 0, bits2, FLASH_OTP_BLOCK_SIZE), NULL);
ensure(sectrue * (0 == memcmp(bits, bits2, FLASH_OTP_BLOCK_SIZE)), "Bootloader downgraded");
}
2017-12-15 06:26:36 -08:00
#endif
2017-12-13 14:08:15 -08:00
int main(void)
2017-03-20 07:41:21 -07:00
{
main_start:
#if PRODUCTION
check_bootloader_version();
#endif
touch_init();
2017-03-28 04:04:54 -07:00
2017-10-16 11:24:28 -07:00
// delay to detect touch
uint32_t touched = 0;
for (int i = 0; i < 100; i++) {
touched = touch_read();
if (touched) {
break;
}
hal_delay(1);
2017-06-14 11:40:31 -07:00
}
vendor_header vhdr;
2017-12-16 16:09:45 -08:00
image_header hdr;
secbool firmware_present;
// detect whether the devices contains a valid firmware
2018-01-25 03:33:09 -08:00
2017-12-16 16:09:45 -08:00
firmware_present = load_vendor_header_keys((const uint8_t *)FIRMWARE_START, &vhdr);
if (sectrue == firmware_present) {
firmware_present = check_vendor_keys_lock(&vhdr);
}
2017-12-16 16:09:45 -08:00
if (sectrue == firmware_present) {
firmware_present = load_image_header((const uint8_t *)(FIRMWARE_START + vhdr.hdrlen), FIRMWARE_IMAGE_MAGIC, FIRMWARE_IMAGE_MAXSIZE, vhdr.vsig_m, vhdr.vsig_n, vhdr.vpub, &hdr);
}
if (sectrue == firmware_present) {
firmware_present = check_image_contents(&hdr, IMAGE_HEADER_SIZE + vhdr.hdrlen, firmware_sectors, FIRMWARE_SECTORS_COUNT);
}
2017-12-16 16:09:45 -08:00
// start the bootloader if no or broken firmware found ...
2018-01-24 09:28:10 -08:00
if (firmware_present != sectrue) {
2018-01-25 03:33:09 -08:00
// show intro animation
// no ui_fadeout(); - we already start from black screen
2018-01-24 09:28:10 -08:00
ui_screen_first();
ui_fadein();
2018-01-25 03:33:09 -08:00
2018-01-24 09:28:10 -08:00
hal_delay(1000);
2018-01-25 03:33:09 -08:00
ui_fadeout();
2018-01-24 09:28:10 -08:00
ui_screen_second();
ui_fadein();
2018-01-25 03:33:09 -08:00
2018-01-24 09:28:10 -08:00
hal_delay(1000);
2018-01-25 03:33:09 -08:00
ui_fadeout();
2018-01-24 09:28:10 -08:00
ui_screen_third();
ui_fadein();
2017-12-16 16:09:45 -08:00
2018-01-25 03:33:09 -08:00
// and start the usb loop
2018-01-24 09:28:10 -08:00
if (bootloader_usb_loop(NULL, NULL) != sectrue) {
return 1;
}
} else
2018-01-25 03:33:09 -08:00
// ... or if user touched the screen on start
2018-01-24 09:28:10 -08:00
if (touched) {
2018-01-25 03:33:09 -08:00
// show firmware info with connect buttons
// no ui_fadeout(); - we already start from black screen
2018-01-24 09:28:10 -08:00
ui_screen_info(sectrue, &vhdr, &hdr);
ui_fadein();
2018-01-25 03:33:09 -08:00
for (;;) {
int response = ui_user_input(INPUT_CONFIRM | INPUT_CANCEL | INPUT_INFO);
ui_fadeout();
2018-01-25 03:33:09 -08:00
// if cancel was pressed -> restart
if (INPUT_CANCEL == response) {
goto main_start;
}
// if confirm was pressed -> jump out
if (INPUT_CONFIRM == response) {
// show firmware info without connect buttons
ui_screen_info(secfalse, &vhdr, &hdr);
ui_fadein();
break;
}
// if info icon was pressed -> show fingerprint
if (INPUT_INFO == response) {
// show fingerprint
ui_screen_info_fingerprint(&hdr);
ui_fadein();
while (INPUT_LONG_CONFIRM != ui_user_input(INPUT_LONG_CONFIRM)) { }
ui_fadeout();
ui_screen_info(sectrue, &vhdr, &hdr);
ui_fadein();
}
}
2018-01-25 03:33:09 -08:00
// and start the usb loop
2018-01-24 09:28:10 -08:00
if (bootloader_usb_loop(&vhdr, &hdr) != sectrue) {
return 1;
}
2017-10-13 09:05:47 -07:00
}
ensure(
load_vendor_header_keys((const uint8_t *)FIRMWARE_START, &vhdr),
2017-10-13 09:05:47 -07:00
"invalid vendor header");
2017-12-13 14:08:15 -08:00
ensure(
check_vendor_keys_lock(&vhdr),
"unauthorized vendor keys");
2017-10-13 09:05:47 -07:00
ensure(
load_image_header((const uint8_t *)(FIRMWARE_START + vhdr.hdrlen), FIRMWARE_IMAGE_MAGIC, FIRMWARE_IMAGE_MAXSIZE, vhdr.vsig_m, vhdr.vsig_n, vhdr.vpub, &hdr),
2017-10-13 09:05:47 -07:00
"invalid firmware header");
ensure(
check_image_contents(&hdr, IMAGE_HEADER_SIZE + vhdr.hdrlen, firmware_sectors, FIRMWARE_SECTORS_COUNT),
"invalid firmware hash");
2017-12-15 11:53:29 -08:00
// if all VTRUST flags are unset = ultimate trust => skip the procedure
2017-12-15 11:53:29 -08:00
if ((vhdr.vtrust & VTRUST_ALL) != VTRUST_ALL) {
2017-12-15 06:26:36 -08:00
2018-01-25 03:33:09 -08:00
// ui_fadeout(); // no fadeout - we start from black screen
2017-12-16 16:09:45 -08:00
ui_screen_boot(&vhdr, &hdr);
2018-01-24 09:28:10 -08:00
ui_fadein();
2017-12-16 16:09:45 -08:00
int delay = (vhdr.vtrust & VTRUST_WAIT) ^ VTRUST_WAIT;
2018-01-28 04:47:03 -08:00
if (delay > 1) {
while (delay > 0) {
ui_screen_boot_wait(delay);
hal_delay(1000);
delay--;
}
} else if (delay == 1) {
2017-12-15 11:53:29 -08:00
hal_delay(1000);
}
if ((vhdr.vtrust & VTRUST_CLICK) == 0) {
2017-12-16 16:09:45 -08:00
ui_screen_boot_click();
2017-12-15 11:53:29 -08:00
touch_click();
}
2017-12-16 16:09:45 -08:00
ui_fadeout();
2017-12-15 11:53:29 -08:00
}
2017-03-29 11:50:45 -07:00
// mpu_config();
// jump_to_unprivileged(FIRMWARE_START + vhdr.hdrlen + IMAGE_HEADER_SIZE);
jump_to(FIRMWARE_START + vhdr.hdrlen + IMAGE_HEADER_SIZE);
2017-03-20 07:41:21 -07:00
return 0;
}