1
0
mirror of https://github.com/trezor/trezor-firmware.git synced 2025-01-01 19:10:58 +00:00
trezor-firmware/micropython/bootloader/messages.c

274 lines
8.0 KiB
C
Raw Normal View History

#include <string.h>
#include <pb_decode.h>
2017-06-14 18:40:31 +00:00
#include <pb_encode.h>
#include "messages.pb.h"
2017-04-15 15:44:19 +00:00
#include "usb.h"
#include "version.h"
#include "messages.h"
2017-06-14 18:40:31 +00:00
bool msg_parse_header(const uint8_t *buf, uint16_t *msg_id, uint32_t *msg_size)
{
if (buf[0] != '?' || buf[1] != '#' || buf[2] != '#') {
return false;
}
*msg_id = (buf[3] << 8) + buf[4];
*msg_size = (buf[5] << 24) + (buf[6] << 16) + (buf[7] << 8) + buf[8];
return true;
}
2017-06-14 21:18:00 +00:00
typedef struct {
uint8_t iface_num;
uint8_t packet_index;
uint8_t packet_pos;
uint8_t buf[USB_PACKET_SIZE];
} usb_write_state;
static bool _usb_write(pb_ostream_t *stream, const pb_byte_t *buf, size_t count)
2017-06-14 18:40:31 +00:00
{
2017-06-14 21:18:00 +00:00
usb_write_state *state = (usb_write_state *)(stream->state);
size_t written = 0;
// while we have data left
while (written < count) {
size_t remaining = count - written;
// if all remaining data fit into our packet
if (state->packet_pos + remaining <= USB_PACKET_SIZE) {
// append data from buf to state->buf
memcpy(state->buf + state->packet_pos, buf + written, remaining);
// advance position
state->packet_pos += remaining;
// and return
return true;
} else {
// append data that fits
memcpy(state->buf + state->packet_pos, buf + written, USB_PACKET_SIZE - state->packet_pos);
written += USB_PACKET_SIZE - state->packet_pos;
// send packet
usb_hid_write_blocking(state->iface_num, state->buf, USB_PACKET_SIZE, 1);
// prepare new packet
state->packet_index++;
memset(state->buf, 0, USB_PACKET_SIZE);
state->buf[0] = '?';
2017-06-16 13:40:27 +00:00
state->packet_pos = MSG_HEADER2_LEN;
2017-06-14 21:18:00 +00:00
}
2017-06-14 18:40:31 +00:00
}
2017-06-14 21:18:00 +00:00
return true;
}
static void _usb_write_flush(usb_write_state *state)
2017-06-14 21:18:00 +00:00
{
// if packet is not filled up completely
if (state->packet_pos < USB_PACKET_SIZE) {
// pad it with zeroes
memset(state->buf + state->packet_pos, 0, USB_PACKET_SIZE - state->packet_pos);
}
// send packet
usb_hid_write_blocking(state->iface_num, state->buf, USB_PACKET_SIZE, 1);
2017-06-14 18:40:31 +00:00
}
2017-06-14 21:18:00 +00:00
static bool _send_msg(uint8_t iface_num, uint16_t msg_id, const pb_field_t fields[], const void *msg)
2017-06-14 18:40:31 +00:00
{
2017-06-14 21:18:00 +00:00
// determine message size by serializing it into a dummy stream
pb_ostream_t sizestream = {
.callback = NULL,
.state = NULL,
.max_size = SIZE_MAX,
.bytes_written = 0,
.errmsg = NULL};
2017-06-14 18:40:31 +00:00
if (!pb_encode(&sizestream, fields, msg)) {
return false;
}
2017-06-14 21:18:00 +00:00
const uint32_t msg_size = sizestream.bytes_written;
usb_write_state state = {
.iface_num = iface_num,
.packet_index = 0,
2017-06-16 13:40:27 +00:00
.packet_pos = MSG_HEADER1_LEN,
2017-06-14 21:18:00 +00:00
.buf = {
'?', '#', '#',
(msg_id >> 8) & 0xFF, msg_id & 0xFF,
(msg_size >> 24) & 0xFF, (msg_size >> 16) & 0xFF, (msg_size >> 8) & 0xFF, msg_size & 0xFF,
},
};
pb_ostream_t stream = {
.callback = &_usb_write,
.state = &state,
.max_size = SIZE_MAX,
.bytes_written = 0,
.errmsg = NULL
};
2017-06-14 18:40:31 +00:00
if (!pb_encode(&stream, fields, msg)) {
return false;
}
_usb_write_flush(&state);
2017-06-14 18:40:31 +00:00
return true;
}
#define MSG_SEND_INIT(TYPE) TYPE msg_send = TYPE##_init_default
#define MSG_SEND_ASSIGN_VALUE(FIELD, VALUE) do { msg_send.has_##FIELD = true; msg_send.FIELD = VALUE; } while (0)
// FIXME: strcpy -> strncpy
#define MSG_SEND_ASSIGN_STRING(FIELD, VALUE) do { msg_send.has_##FIELD = true; strcpy(msg_send.FIELD, VALUE); } while (0)
#define MSG_SEND(TYPE) do { _send_msg(iface_num, MessageType_MessageType_##TYPE, TYPE##_fields, &msg_send); } while (0)
typedef struct {
uint8_t iface_num;
uint8_t packet_index;
uint8_t packet_pos;
uint8_t *buf;
} usb_read_state;
2017-06-14 18:40:31 +00:00
static bool _usb_read(pb_istream_t *stream, uint8_t *buf, size_t count)
2017-06-14 18:40:31 +00:00
{
usb_read_state *state = (usb_read_state *)(stream->state);
size_t read = 0;
// while we have data left
while (read < count) {
size_t remaining = count - read;
// if all remaining data fit into our packet
if (state->packet_pos + remaining <= USB_PACKET_SIZE) {
// append data from buf to state->buf
memcpy(buf + read, state->buf + state->packet_pos, remaining);
// advance position
state->packet_pos += remaining;
// and return
return true;
} else {
// append data that fits
memcpy(buf + read, state->buf + state->packet_pos, USB_PACKET_SIZE - state->packet_pos);
read += USB_PACKET_SIZE - state->packet_pos;
// read next packet
usb_hid_write_blocking(state->iface_num, state->buf, USB_PACKET_SIZE, 1);
// prepare next packet
state->packet_index++;
state->packet_pos = MSG_HEADER2_LEN;
2017-06-16 13:40:27 +00:00
}
}
return true;
}
static void _usb_read_flush(usb_read_state *state)
{
(void)state;
}
static bool _recv_msg(uint8_t iface_num, uint32_t msg_size, uint8_t *buf, const pb_field_t fields[], void *msg)
{
usb_read_state state = {
.iface_num = iface_num,
.packet_index = 0,
.packet_pos = MSG_HEADER1_LEN,
.buf = buf
};
pb_istream_t stream = {
.callback = &_usb_read,
.state = &state,
.bytes_left = msg_size,
.errmsg = NULL
};
if (!pb_decode_noinit(&stream, fields, msg)) {
return false;
}
_usb_read_flush(&state);
return true;
2017-06-16 13:40:27 +00:00
}
/*
static bool _decode(pb_istream_t *stream, const pb_field_t *field, void **arg)
{
pb_byte_t *buf = *arg;
memset(buf, 0, 1024);
if (stream->bytes_left > 1024 - 1) {
return false;
}
if (!pb_read(stream, buf, stream->bytes_left)) {
return false;
}
return true;
}
*/
#define MSG_RECV_INIT(TYPE) TYPE msg_recv = TYPE##_init_default
#define MSG_RECV(TYPE) do { _recv_msg(iface_num, msg_size, buf, TYPE##_fields, &msg_recv); } while(0)
2017-06-16 13:40:27 +00:00
void process_msg_Initialize(uint8_t iface_num, uint32_t msg_size, uint8_t *buf)
{
MSG_RECV_INIT(Initialize);
MSG_RECV(Initialize);
MSG_SEND_INIT(Features);
MSG_SEND_ASSIGN_STRING(vendor, "trezor.io");
MSG_SEND_ASSIGN_VALUE(major_version, VERSION_MAJOR);
MSG_SEND_ASSIGN_VALUE(minor_version, VERSION_MINOR);
MSG_SEND_ASSIGN_VALUE(patch_version, VERSION_PATCH);
MSG_SEND_ASSIGN_VALUE(bootloader_mode, true);
2017-06-14 18:40:31 +00:00
// TODO: properly detect firmware
MSG_SEND_ASSIGN_VALUE(firmware_present, false);
2017-06-14 18:40:31 +00:00
MSG_SEND(Features);
}
2017-06-16 13:40:27 +00:00
void process_msg_Ping(uint8_t iface_num, uint32_t msg_size, uint8_t *buf)
2017-06-14 18:40:31 +00:00
{
MSG_RECV_INIT(Ping);
MSG_RECV(Ping);
2017-06-16 13:40:27 +00:00
MSG_SEND_INIT(Success);
MSG_SEND_ASSIGN_STRING(message, msg_recv.message);
2017-06-14 18:40:31 +00:00
MSG_SEND(Success);
}
2017-06-16 13:40:27 +00:00
void process_msg_FirmwareErase(uint8_t iface_num, uint32_t msg_size, uint8_t *buf)
2017-06-14 18:40:31 +00:00
{
MSG_RECV_INIT(FirmwareErase);
MSG_RECV(FirmwareErase);
2017-06-16 13:40:27 +00:00
MSG_SEND_INIT(FirmwareRequest);
MSG_SEND_ASSIGN_VALUE(offset, 0);
MSG_SEND_ASSIGN_VALUE(length, FIRMWARE_CHUNK_SIZE);
MSG_SEND(FirmwareRequest);
2017-06-14 18:40:31 +00:00
}
2017-06-16 13:40:27 +00:00
void process_msg_FirmwareUpload(uint8_t iface_num, uint32_t msg_size, uint8_t *buf)
2017-06-14 18:40:31 +00:00
{
MSG_RECV_INIT(FirmwareUpload);
MSG_RECV(FirmwareUpload);
/*
MSG_SEND_INIT(FirmwareRequest);
MSG_SEND_ASSIGN_VALUE(offset, FIRMWARE_CHUNK_SIZE);
MSG_SEND_ASSIGN_VALUE(length, FIRMWARE_CHUNK_SIZE);
MSG_SEND(FirmwareRequest);
*/
MSG_SEND_INIT(Success);
MSG_SEND(Success);
2017-06-14 18:40:31 +00:00
}
2017-06-16 13:40:27 +00:00
void process_msg_unknown(uint8_t iface_num, uint32_t msg_size, uint8_t *buf)
2017-06-14 18:40:31 +00:00
{
// consume remaining message
int remaining_chunks = (msg_size - (USB_PACKET_SIZE - MSG_HEADER1_LEN)) / (USB_PACKET_SIZE - MSG_HEADER2_LEN);
for (int i = 0; i < remaining_chunks; i++) {
int r = usb_hid_read_blocking(USB_IFACE_NUM, buf, USB_PACKET_SIZE, 100);
if (r != USB_PACKET_SIZE) {
break;
}
}
2017-06-16 13:40:27 +00:00
MSG_SEND_INIT(Failure);
MSG_SEND_ASSIGN_VALUE(code, FailureType_Failure_UnexpectedMessage);
MSG_SEND_ASSIGN_STRING(message, "Unexpected message");
2017-06-14 18:40:31 +00:00
MSG_SEND(Failure);
2017-04-15 15:44:19 +00:00
}