2017-03-20 14:41:21 +00:00
|
|
|
#include STM32_HAL_H
|
|
|
|
|
2017-04-01 17:58:58 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
2017-03-21 00:41:49 +00:00
|
|
|
#include "common.h"
|
2017-03-20 14:41:21 +00:00
|
|
|
#include "display.h"
|
2017-04-01 10:57:14 +00:00
|
|
|
#include "image.h"
|
2017-04-05 17:33:50 +00:00
|
|
|
#include "touch.h"
|
2017-03-20 14:41:21 +00:00
|
|
|
|
2017-03-21 15:06:22 +00:00
|
|
|
#define LOADER_FGCOLOR 0xFFFF
|
|
|
|
#define LOADER_BGCOLOR 0x0000
|
|
|
|
|
|
|
|
#define LOADER_PRINT(X) do { display_print(X, -1); display_print_out(LOADER_FGCOLOR, LOADER_BGCOLOR); } while(0)
|
|
|
|
#define LOADER_PRINTLN(X) do { display_print(X "\n", -1); display_print_out(LOADER_FGCOLOR, LOADER_BGCOLOR); } while(0)
|
|
|
|
|
2017-04-05 15:41:10 +00:00
|
|
|
#define IMAGE_MAGIC 0x465A5254 // TRZF
|
|
|
|
#define IMAGE_MAXSIZE (7 * 128 * 1024)
|
|
|
|
|
2017-03-21 14:56:50 +00:00
|
|
|
void pendsv_isr_handler(void) {
|
|
|
|
__fatal_error("pendsv");
|
2017-03-20 14:41:21 +00:00
|
|
|
}
|
|
|
|
|
2017-04-05 13:10:33 +00:00
|
|
|
void display_vendor(const uint8_t *vimg, const char *vstr, uint32_t vstr_len, uint32_t fw_version)
|
2017-04-01 17:58:58 +00:00
|
|
|
{
|
2017-04-01 18:12:48 +00:00
|
|
|
display_clear();
|
2017-04-01 17:58:58 +00:00
|
|
|
if (memcmp(vimg, "TOIf", 4) != 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
uint16_t w = *(uint16_t *)(vimg + 4);
|
|
|
|
uint16_t h = *(uint16_t *)(vimg + 6);
|
2017-04-01 18:12:48 +00:00
|
|
|
if (w != 120 || h != 120) {
|
|
|
|
return;
|
|
|
|
}
|
2017-04-01 17:58:58 +00:00
|
|
|
uint32_t datalen = *(uint32_t *)(vimg + 8);
|
2017-04-05 15:41:10 +00:00
|
|
|
display_image(60, 32, w, h, vimg + 12, datalen);
|
|
|
|
display_text_center(120, 192, vstr, vstr_len, FONT_BOLD, 0xFFFF, 0x0000);
|
|
|
|
char ver_str[] = "v0.0.0.0";
|
|
|
|
// TODO: fixme - the following does not work for values >= 10
|
|
|
|
ver_str[1] += fw_version & 0xFF;
|
|
|
|
ver_str[3] += (fw_version >> 8) & 0xFF;
|
|
|
|
ver_str[5] += (fw_version >> 16) & 0xFF;
|
|
|
|
ver_str[7] += (fw_version >> 24) & 0xFF;
|
|
|
|
display_text_center(120, 215, ver_str, -1, FONT_NORMAL, 0x7BEF, 0x0000);
|
2017-04-01 18:12:48 +00:00
|
|
|
display_refresh();
|
2017-04-01 17:58:58 +00:00
|
|
|
}
|
|
|
|
|
2017-04-01 00:32:05 +00:00
|
|
|
void check_and_jump(void)
|
|
|
|
{
|
2017-04-05 14:24:43 +00:00
|
|
|
LOADER_PRINTLN("checking vendor header");
|
2017-04-01 17:24:41 +00:00
|
|
|
|
|
|
|
vendor_header vhdr;
|
2017-04-05 14:24:43 +00:00
|
|
|
if (vendor_parse_header((const uint8_t *)FIRMWARE_START, &vhdr)) {
|
|
|
|
LOADER_PRINTLN("valid vendor header");
|
|
|
|
} else {
|
2017-04-01 17:24:41 +00:00
|
|
|
LOADER_PRINTLN("invalid vendor header");
|
|
|
|
return;
|
|
|
|
}
|
2017-04-05 14:24:43 +00:00
|
|
|
|
|
|
|
if (vendor_check_signature((const uint8_t *)FIRMWARE_START, &vhdr)) {
|
|
|
|
LOADER_PRINTLN("valid vendor header signature");
|
|
|
|
} else {
|
|
|
|
LOADER_PRINTLN("invalid vendor header signature");
|
2017-04-01 17:24:41 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-04-05 14:24:43 +00:00
|
|
|
LOADER_PRINTLN("checking firmware header");
|
|
|
|
|
2017-04-05 13:10:33 +00:00
|
|
|
image_header hdr;
|
2017-04-05 15:41:10 +00:00
|
|
|
if (image_parse_header((const uint8_t *)(FIRMWARE_START + vhdr.hdrlen), IMAGE_MAGIC, IMAGE_MAXSIZE, &hdr)) {
|
2017-04-05 14:24:43 +00:00
|
|
|
LOADER_PRINTLN("valid firmware header");
|
|
|
|
} else {
|
2017-04-05 13:10:33 +00:00
|
|
|
LOADER_PRINTLN("invalid firmware header");
|
|
|
|
return;
|
|
|
|
}
|
2017-04-05 14:24:43 +00:00
|
|
|
|
2017-04-05 13:10:33 +00:00
|
|
|
if (image_check_signature((const uint8_t *)(FIRMWARE_START + vhdr.hdrlen), &hdr, &vhdr)) {
|
|
|
|
LOADER_PRINTLN("valid firmware signature");
|
2017-04-05 14:24:43 +00:00
|
|
|
|
2017-04-05 13:10:33 +00:00
|
|
|
display_vendor(vhdr.vimg, (const char *)vhdr.vstr, vhdr.vstr_len, hdr.version);
|
2017-04-01 20:30:10 +00:00
|
|
|
HAL_Delay(1000); // TODO: remove?
|
2017-04-01 10:57:14 +00:00
|
|
|
LOADER_PRINTLN("JUMP!");
|
2017-04-01 17:24:41 +00:00
|
|
|
jump_to(FIRMWARE_START + vhdr.hdrlen + HEADER_SIZE);
|
2017-04-05 14:24:43 +00:00
|
|
|
|
2017-04-01 00:32:05 +00:00
|
|
|
} else {
|
2017-04-05 13:10:33 +00:00
|
|
|
LOADER_PRINTLN("invalid firmware signature");
|
2017-04-01 00:32:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-05 17:33:50 +00:00
|
|
|
void mainloop(void)
|
|
|
|
{
|
|
|
|
__fatal_error("touch detected - launch aborted");
|
|
|
|
}
|
|
|
|
|
2017-03-23 15:22:58 +00:00
|
|
|
int main(void)
|
2017-03-20 14:41:21 +00:00
|
|
|
{
|
2017-03-31 22:22:24 +00:00
|
|
|
SCB->VTOR = LOADER_START + HEADER_SIZE;
|
2017-03-23 15:22:58 +00:00
|
|
|
periph_init();
|
2017-03-20 14:41:21 +00:00
|
|
|
|
2017-04-05 17:33:50 +00:00
|
|
|
touch_init();
|
|
|
|
|
2017-03-20 14:41:21 +00:00
|
|
|
display_init();
|
|
|
|
display_clear();
|
|
|
|
display_backlight(255);
|
|
|
|
|
2017-04-01 00:32:05 +00:00
|
|
|
LOADER_PRINTLN("TREZOR Loader");
|
|
|
|
LOADER_PRINTLN("=============");
|
|
|
|
LOADER_PRINTLN("starting loader");
|
|
|
|
|
2017-04-05 17:33:50 +00:00
|
|
|
if (touch_read() != 0) {
|
|
|
|
mainloop();
|
|
|
|
} else {
|
|
|
|
check_and_jump();
|
|
|
|
}
|
2017-03-29 18:50:45 +00:00
|
|
|
|
2017-04-01 00:32:05 +00:00
|
|
|
__fatal_error("halt");
|
2017-03-20 14:41:21 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|