mirror of
https://github.com/trezor/trezor-firmware.git
synced 2024-12-16 11:28:14 +00:00
491647085f
[no changelog]
315 lines
9.1 KiB
C
315 lines
9.1 KiB
C
/*
|
|
* 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 TREZOR_BOARD
|
|
|
|
#include "bootui.h"
|
|
#include "colors.h"
|
|
#include "display.h"
|
|
#include "display_draw.h"
|
|
#include "display_utils.h"
|
|
#include "fonts/fonts.h"
|
|
#ifdef TREZOR_EMULATOR
|
|
#include "emulator.h"
|
|
#else
|
|
#include "mini_printf.h"
|
|
#endif
|
|
#include "rust_ui.h"
|
|
#include "version.h"
|
|
|
|
#define BACKLIGHT_NORMAL 150
|
|
|
|
#define COLOR_BL_BG COLOR_WHITE // background
|
|
#define COLOR_BL_FG COLOR_BLACK // foreground
|
|
|
|
#ifdef RGB16
|
|
#define COLOR_BL_FAIL RGB16(0xFF, 0x00, 0x00) // red
|
|
#define COLOR_BL_DONE RGB16(0x00, 0xAE, 0x0B) // green
|
|
#define COLOR_BL_PROCESS RGB16(0x4A, 0x90, 0xE2) // blue
|
|
#define COLOR_BL_GRAY RGB16(0x99, 0x99, 0x99) // gray
|
|
#else
|
|
#define COLOR_BL_FAIL COLOR_BL_FG
|
|
#define COLOR_BL_DONE COLOR_BL_FG
|
|
#define COLOR_BL_PROCESS COLOR_BL_FG
|
|
#define COLOR_BL_GRAY COLOR_BL_FG
|
|
#endif
|
|
|
|
#if !defined TREZOR_MODEL_R && !defined TREZOR_MODEL_T3B1
|
|
#define BOOT_WAIT_HEIGHT 25
|
|
#define BOOT_WAIT_Y_TOP (DISPLAY_RESY - BOOT_WAIT_HEIGHT)
|
|
#else
|
|
#define BOOT_WAIT_HEIGHT 12
|
|
#define BOOT_WAIT_Y_TOP (DISPLAY_RESY - BOOT_WAIT_HEIGHT)
|
|
#endif
|
|
|
|
#define TOIF_LENGTH(ptr) ((*(uint32_t *)((ptr) + 8)) + 12)
|
|
|
|
// common shared functions
|
|
|
|
static void format_ver(const char *format, uint32_t version, char *buffer,
|
|
size_t buffer_len) {
|
|
mini_snprintf(buffer, buffer_len, format, (int)(version & 0xFF),
|
|
(int)((version >> 8) & 0xFF), (int)((version >> 16) & 0xFF)
|
|
// ignore build field (int)((version >> 24) & 0xFF)
|
|
);
|
|
}
|
|
|
|
// boot UI
|
|
|
|
#ifndef NEW_RENDERING
|
|
static uint16_t boot_background;
|
|
#endif
|
|
|
|
static bool initial_setup = true;
|
|
|
|
void ui_set_initial_setup(bool initial) { initial_setup = initial; }
|
|
|
|
#ifndef NEW_RENDERING
|
|
static void ui_screen_boot_old(const vendor_header *const vhdr,
|
|
const image_header *const hdr) {
|
|
const int show_string = ((vhdr->vtrust & VTRUST_NO_STRING) == 0);
|
|
if ((vhdr->vtrust & VTRUST_NO_RED) != 0) {
|
|
boot_background = COLOR_BLACK;
|
|
} else {
|
|
boot_background = COLOR_BL_FAIL;
|
|
}
|
|
|
|
const uint8_t *vimg = vhdr->vimg;
|
|
const uint32_t fw_version = hdr->version;
|
|
|
|
display_bar(0, 0, DISPLAY_RESX, DISPLAY_RESY, boot_background);
|
|
|
|
#if !defined TREZOR_MODEL_R && !defined TREZOR_MODEL_T3B1
|
|
int image_top = show_string ? 30 : (DISPLAY_RESY - 120) / 2;
|
|
// check whether vendor image is 120x120
|
|
if (memcmp(vimg, "TOIF\x78\x00\x78\x00", 8) == 0) {
|
|
uint32_t datalen = TOIF_LENGTH(vimg);
|
|
display_image((DISPLAY_RESX - 120) / 2, image_top, vimg, datalen);
|
|
}
|
|
|
|
if (show_string) {
|
|
char ver_str[64];
|
|
display_text_center(DISPLAY_RESX / 2, DISPLAY_RESY - 5 - 50, vhdr->vstr,
|
|
vhdr->vstr_len, FONT_NORMAL, COLOR_BL_BG,
|
|
boot_background);
|
|
format_ver("%d.%d.%d", fw_version, ver_str, sizeof(ver_str));
|
|
display_text_center(DISPLAY_RESX / 2, DISPLAY_RESY - 5 - 25, ver_str, -1,
|
|
FONT_NORMAL, COLOR_BL_BG, boot_background);
|
|
}
|
|
#else
|
|
// check whether vendor image is 24x24
|
|
if (memcmp(vimg, "TOIG\x18\x00\x18\x00", 8) == 0) {
|
|
uint32_t datalen = TOIF_LENGTH(vimg);
|
|
display_icon((DISPLAY_RESX - 22) / 2, 0, vimg, datalen, COLOR_BL_BG,
|
|
boot_background);
|
|
}
|
|
|
|
if (show_string) {
|
|
char ver_str[64];
|
|
display_text_center(DISPLAY_RESX / 2, 36, vhdr->vstr, vhdr->vstr_len,
|
|
FONT_NORMAL, COLOR_BL_BG, boot_background);
|
|
format_ver("%d.%d.%d", fw_version, ver_str, sizeof(ver_str));
|
|
display_text_center(DISPLAY_RESX / 2, 46, ver_str, -1, FONT_NORMAL,
|
|
COLOR_BL_BG, boot_background);
|
|
}
|
|
|
|
#endif
|
|
|
|
display_pixeldata_dirty();
|
|
display_refresh();
|
|
}
|
|
#endif
|
|
|
|
#ifndef NEW_RENDERING
|
|
static void ui_screen_boot_wait(int wait_seconds) {
|
|
char wait_str[32];
|
|
mini_snprintf(wait_str, sizeof(wait_str), "starting in %d s", wait_seconds);
|
|
display_bar(0, BOOT_WAIT_Y_TOP, DISPLAY_RESX, BOOT_WAIT_HEIGHT,
|
|
boot_background);
|
|
display_text_center(DISPLAY_RESX / 2, DISPLAY_RESY - 5, wait_str, -1,
|
|
FONT_NORMAL, COLOR_BL_BG, boot_background);
|
|
display_pixeldata_dirty();
|
|
display_refresh();
|
|
}
|
|
#endif
|
|
|
|
#if defined USE_TOUCH
|
|
#include "touch.h"
|
|
|
|
void ui_click(void) {
|
|
// flush touch events if any
|
|
while (touch_get_event()) {
|
|
}
|
|
// wait for TOUCH_START
|
|
while ((touch_get_event() & TOUCH_START) == 0) {
|
|
}
|
|
// wait for TOUCH_END
|
|
while ((touch_get_event() & TOUCH_END) == 0) {
|
|
}
|
|
// flush touch events if any
|
|
while (touch_get_event()) {
|
|
}
|
|
}
|
|
|
|
#elif defined USE_BUTTON
|
|
#include "button.h"
|
|
|
|
void ui_click(void) {
|
|
for (;;) {
|
|
button_read();
|
|
if (button_state_left() != 0 && button_state_right() != 0) {
|
|
break;
|
|
}
|
|
}
|
|
for (;;) {
|
|
button_read();
|
|
if (button_state_left() != 1 && button_state_right() != 1) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
#else
|
|
#error "No input method defined"
|
|
#endif
|
|
|
|
#ifndef NEW_RENDERING
|
|
static void ui_screen_boot_click(void) {
|
|
display_bar(0, BOOT_WAIT_Y_TOP, DISPLAY_RESX, BOOT_WAIT_HEIGHT,
|
|
boot_background);
|
|
bld_continue_label(boot_background);
|
|
display_pixeldata_dirty();
|
|
display_refresh();
|
|
ui_click();
|
|
}
|
|
#endif
|
|
|
|
#ifdef NEW_RENDERING
|
|
void ui_screen_boot(const vendor_header *const vhdr,
|
|
const image_header *const hdr, int wait) {
|
|
bool show_string = ((vhdr->vtrust & VTRUST_NO_STRING) == 0);
|
|
const char *vendor_str = show_string ? vhdr->vstr : NULL;
|
|
const size_t vendor_str_len = show_string ? vhdr->vstr_len : 0;
|
|
bool red_screen = ((vhdr->vtrust & VTRUST_NO_RED) == 0);
|
|
uint32_t vimg_len = TOIF_LENGTH(vhdr->vimg);
|
|
|
|
screen_boot(red_screen, vendor_str, vendor_str_len, hdr->version, vhdr->vimg,
|
|
vimg_len, wait);
|
|
}
|
|
#else // NEW_RENDERING
|
|
|
|
void ui_screen_boot(const vendor_header *const vhdr,
|
|
const image_header *const hdr, int wait) {
|
|
if (wait == 0) {
|
|
ui_screen_boot_old(vhdr, hdr);
|
|
} else if (wait > 0) {
|
|
ui_screen_boot_wait(wait);
|
|
} else {
|
|
ui_screen_boot_click();
|
|
}
|
|
}
|
|
#endif
|
|
|
|
// welcome UI
|
|
|
|
void ui_screen_welcome(void) { screen_welcome(); }
|
|
|
|
uint32_t ui_screen_intro(const vendor_header *const vhdr,
|
|
const image_header *const hdr, bool fw_ok) {
|
|
char bld_ver[32];
|
|
char ver_str[64];
|
|
format_ver("%d.%d.%d", VERSION_UINT32, bld_ver, sizeof(bld_ver));
|
|
format_ver("%d.%d.%d", hdr->version, ver_str, sizeof(ver_str));
|
|
|
|
return screen_intro(bld_ver, vhdr->vstr, vhdr->vstr_len, ver_str, fw_ok);
|
|
}
|
|
|
|
uint32_t ui_screen_menu(secbool firmware_present) {
|
|
return screen_menu(firmware_present);
|
|
}
|
|
|
|
// install UI
|
|
|
|
uint32_t ui_screen_install_confirm(const vendor_header *const vhdr,
|
|
const image_header *const hdr,
|
|
secbool should_keep_seed,
|
|
secbool is_newvendor, secbool is_newinstall,
|
|
int version_cmp) {
|
|
uint8_t fingerprint[32];
|
|
char ver_str[64];
|
|
get_image_fingerprint(hdr, fingerprint);
|
|
format_ver("%d.%d.%d", hdr->version, ver_str, sizeof(ver_str));
|
|
return screen_install_confirm(vhdr->vstr, vhdr->vstr_len, ver_str,
|
|
fingerprint, should_keep_seed == sectrue,
|
|
|
|
is_newvendor == sectrue,
|
|
is_newinstall == sectrue, version_cmp);
|
|
}
|
|
|
|
void ui_screen_install_start() {
|
|
screen_install_progress(0, true, initial_setup);
|
|
}
|
|
|
|
void ui_screen_install_progress_erase(int pos, int len) {
|
|
screen_install_progress(250 * pos / len, false, initial_setup);
|
|
}
|
|
|
|
void ui_screen_install_progress_upload(int pos) {
|
|
screen_install_progress(pos, false, initial_setup);
|
|
}
|
|
|
|
// wipe UI
|
|
|
|
uint32_t ui_screen_wipe_confirm(void) { return screen_wipe_confirm(); }
|
|
|
|
void ui_screen_wipe(void) { screen_wipe_progress(0, true); }
|
|
|
|
void ui_screen_wipe_progress(int pos, int len) {
|
|
screen_wipe_progress(1000 * pos / len, false);
|
|
}
|
|
|
|
// done UI
|
|
void ui_screen_done(uint8_t restart_seconds, secbool full_redraw) {
|
|
screen_install_success(restart_seconds, initial_setup, full_redraw);
|
|
}
|
|
|
|
void ui_screen_boot_stage_1(bool fading) { screen_boot_stage_1(fading); }
|
|
|
|
// error UI
|
|
void ui_screen_fail(void) { screen_install_fail(); }
|
|
|
|
#ifdef USE_OPTIGA
|
|
uint32_t ui_screen_unlock_bootloader_confirm(void) {
|
|
return screen_unlock_bootloader_confirm();
|
|
}
|
|
#else
|
|
void ui_screen_install_restricted(void) { screen_install_fail(); }
|
|
#endif
|
|
|
|
// general functions
|
|
|
|
void ui_fadein(void) { display_fade(0, BACKLIGHT_NORMAL, 1000); }
|
|
|
|
void ui_fadeout(void) {
|
|
display_fade(BACKLIGHT_NORMAL, 0, 500);
|
|
display_clear();
|
|
}
|