Skip to content

Commit c0380b7

Browse files
committed
simulator: implement a simulator for bitbox02 device
HWI is thinking of updating its support policy such that supported wallets must implement a simulator/emulator. See bitcoin-core/HWI#685. That's why a simulator is implemented for bitbox02, supporting functionalities of its API. This first version of the simulator is capable of nearly every functionality of a normal Bitbox02 device, without promising any security or production use. Its main aim is to be able to run unit tests for features and test the API. In addition, it will be configured to run automated tests in CI, which helps both us and HWI integration. Right now, the simulator has 3 different ways to communicate with a client: giving inputs/getting output from CLI, using pipes or opening sockets. Socket is the most convenient and reliable choice in this version. It expects the clients to open a socket on port 15432, which is selected intentionally to avoid possible conflicts. The simulator resides with C unit-tests since it uses same mocks, therefore it can be built by `make unit-test`. Lastly, Python client implemented in `py/send_message.py` is updated to support communicating with simulator with the socket configuration mentioned above. Client can be started up with `./py/send_message.py --simulator` command. To run the simulator, `build-build/bin/test_simulator` command is sufficient. Signed-off-by: asi345 <[email protected]>
1 parent 5e1268b commit c0380b7

12 files changed

+193
-78
lines changed

.DS_Store

10 KB
Binary file not shown.

.vscode/c_cpp_properties.json

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
{
2+
"configurations": [
3+
{
4+
"name": "Mac",
5+
"includePath": [
6+
"${workspaceFolder}/**"
7+
],
8+
"defines": [],
9+
"macFrameworkPath": [
10+
"/Library/Developer/CommandLineTools/SDKs/MacOSX.sdk/System/Library/Frameworks"
11+
],
12+
"compilerPath": "/usr/bin/clang",
13+
"cStandard": "c17",
14+
"cppStandard": "c++17",
15+
"intelliSenseMode": "macos-clang-x64",
16+
"configurationProvider": "ms-vscode.makefile-tools"
17+
}
18+
],
19+
"version": 4
20+
}

.vscode/settings.json

-6
This file was deleted.

py/send_message.py

+6
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
# pylint: disable=too-many-lines
1919

2020
import argparse
21+
import os
2122
import pprint
2223
import sys
2324
from typing import List, Any, Optional, Callable, Union, Tuple, Sequence
@@ -1558,7 +1559,12 @@ def run(self) -> int:
15581559

15591560
def connect_to_simulator_bitbox(debug: bool) -> int:
15601561
class Simulator(PhysicalLayer):
1562+
def __init__(self) -> None:
1563+
self.fifo_path = "./tmp/myfifo"
1564+
15611565
def write(self, data: bytes) -> None:
1566+
with open(self.fifo_path, 'w') as fifo:
1567+
fifo.write(data.hex())
15621568
raise Exception(f"TODO: write {data.hex()} to simulator")
15631569

15641570
def read(self, size: int, timeout_ms: int) -> bytes:

src/usb/class/hid/hww/hid_hww.c

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,7 @@
1818
#include <queue.h>
1919
#include <string.h>
2020
#include <usb/usb_packet.h>
21+
#include <stdio.h>
2122

2223
#define HID_HWW_VERSION 0x00000001u
2324

@@ -81,6 +82,7 @@ static void _send_next(void)
8182
if (data != NULL) {
8283
_send_busy = true;
8384
hid_write(&_func_data, data, USB_HID_REPORT_OUT_SIZE);
85+
printf("hid_write: %s\n", data);
8486
}
8587
}
8688

src/usb/usb_frame.c

+4
Original file line numberDiff line numberDiff line change
@@ -193,10 +193,14 @@ int32_t usb_frame_process(const USB_FRAME* frame, State* state)
193193
{
194194
// USB initialization frames contain a command that begins with 0x80
195195
if ((frame->type & FRAME_TYPE_MASK) == FRAME_TYPE_INIT) {
196+
printf("CMD_INIT, %d and %u\n", frame->type & FRAME_TYPE_MASK, frame->type);
196197
return _cmd_init(frame, state);
197198
}
198199
if ((frame->type & FRAME_TYPE_MASK) == FRAME_TYPE_CONT) {
200+
printf("CMD_CONTINUE, %d and %u\n", frame->type & FRAME_TYPE_MASK, frame->type);
201+
printf("cid: %u, type: %u, cmd: %u\n", frame->cid, frame->type, frame->init.cmd);
199202
return _cmd_continue(frame, state);
200203
}
204+
printf("FRAME_ERR_INVALID_CMD obtained\n");
201205
return FRAME_ERR_INVALID_CMD;
202206
}

src/usb/usb_packet.c

+7
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@
1919
#include <stdbool.h>
2020
#include <stdlib.h>
2121
#include <string.h>
22+
#include <stdio.h>
2223

2324
#define ERR_NONE 0
2425

@@ -67,23 +68,28 @@ bool usb_packet_process(const USB_FRAME* frame)
6768
switch (usb_frame_process(frame, &_in_state)) {
6869
case FRAME_ERR_IGNORE:
6970
// Ignore this frame, i.e. no response.
71+
printf("usb_packet_process: FRAME_ERR_IGNORE\n");
7072
break;
7173
case FRAME_ERR_INVALID_SEQ:
7274
// Reset the state becuase this error indicates that there is a host application bug
7375
_reset_state();
7476
_queue_err(FRAME_ERR_INVALID_SEQ, frame->cid);
77+
printf("usb_packet_process: FRAME_ERR_INVALID_SEQ\n");
7578
break;
7679
case FRAME_ERR_CHANNEL_BUSY:
7780
// We don't reset the state because this error doesn't indicate something wrong with the
7881
// "current" connection.
7982
_queue_err(FRAME_ERR_CHANNEL_BUSY, frame->cid);
83+
printf("usb_packet_process: FRAME_ERR_CHANNEL_BUSY\n");
8084
break;
8185
case FRAME_ERR_INVALID_LEN:
8286
// Reset the state becuase this error indicates that there is a host application bug
8387
_reset_state();
8488
_queue_err(FRAME_ERR_INVALID_LEN, frame->cid);
89+
printf("usb_packet_process: FRAME_ERR_INVALID_LEN\n");
8590
break;
8691
case ERR_NONE:
92+
printf("usb_packet_process: ERR_NONE\n");
8793
if (_need_more_data()) {
8894
// Do not send a message yet
8995
return true;
@@ -100,6 +106,7 @@ bool usb_packet_process(const USB_FRAME* frame)
100106
break;
101107
default:
102108
// other errors
109+
printf("usb_packet_process: default\n");
103110
_reset_state();
104111
_queue_err(FRAME_ERR_OTHER, frame->cid);
105112
break;

test/.DS_Store

10 KB
Binary file not shown.

test/unit-test/CMakeLists.txt

+2
Original file line numberDiff line numberDiff line change
@@ -240,6 +240,8 @@ set(TEST_LIST
240240
"-Wl,--wrap=screen_process"
241241
ugui
242242
""
243+
simulator
244+
""
243245
)
244246

245247
find_package(CMocka REQUIRED)

test/unit-test/simulator

-32.5 KB
Binary file not shown.

test/unit-test/simulator.c

-72
This file was deleted.

test/unit-test/test_simulator.c

+152
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,152 @@
1+
// Copyright 2023 Shift Cryptosecurity AG
2+
//
3+
// Licensed under the Apache License, Version 2.0 (the "License");
4+
// you may not use this file except in compliance with the License.
5+
// You may obtain a copy of the License at
6+
//
7+
// http://www.apache.org/licenses/LICENSE-2.0
8+
//
9+
// Unless required by applicable law or agreed to in writing, software
10+
// distributed under the License is distributed on an "AS IS" BASIS,
11+
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
// See the License for the specific language governing permissions and
13+
// limitations under the License.
14+
15+
16+
/*
17+
infinite loop reading stdin line by line
18+
each line is e.g. a hex encoded string containing the USB message that is sent by the host (e.g. the USB msgs that is sent by the bitbox-api-rs client library)
19+
process the usb message
20+
print the response usb messages to stdout, again one message per line, hex encoded
21+
22+
somehow expose a usb connection to this c program. There read the messages from stdin and write the responses to stdout.
23+
When read the message, call the message handling function of the firmware so that it takes care of everything and returns the output
24+
*/
25+
26+
// https://github.com/Coldcard/firmware/blob/master/unix/simulator.py#L296
27+
// Coldcard implements in python and uses `serial` library to access to USB serial ports
28+
// https://pyserial.readthedocs.io/en/latest/pyserial.html
29+
30+
// For C, there is usbip and gadget api libraries
31+
// usbip allows remote/local USB devices to communicate over IP
32+
// gadget api allows to emulate USB devices, but its protocol seems complex
33+
34+
#include <stdio.h>
35+
#include <fcntl.h>
36+
#include <unistd.h>
37+
#include "usb/usb_processing.h"
38+
39+
#define USB_DESC_HID_EP_SIZE 0x40
40+
#define USB_REPORT_SIZE USB_DESC_HID_EP_SIZE
41+
#define USB_HID_REPORT_IN_SIZE USB_REPORT_SIZE
42+
#define USB_HID_REPORT_OUT_SIZE USB_REPORT_SIZE
43+
44+
static uint8_t _out_report[USB_HID_REPORT_OUT_SIZE];
45+
const char* fifo_path = "./tmp/myfifo";
46+
int fd, num_read;
47+
48+
int hex_to_uint8(char high, char low, uint8_t* num) {
49+
*num = 0;
50+
51+
if (high >= '0' && high <= '9')
52+
{
53+
*num += (high - '0') << 4;
54+
}
55+
else if (high >= 'A' && high <= 'F')
56+
{
57+
*num += (high - 'A' + 10) << 4;
58+
}
59+
else if (high >= 'a' && high <= 'f')
60+
{
61+
*num += (high - 'a' + 10) << 4;
62+
}
63+
else
64+
{
65+
return 1;
66+
}
67+
68+
if (low >= '0' && low <= '9')
69+
{
70+
*num += (low - '0');
71+
}
72+
else if (low >= 'A' && low <= 'F')
73+
{
74+
*num += (low - 'A' + 10);
75+
}
76+
else if (low >= 'a' && low <= 'f')
77+
{
78+
*num += (low - 'a' + 10);
79+
}
80+
else
81+
{
82+
return 1;
83+
}
84+
85+
return 0;
86+
}
87+
88+
char* get_usb_message(char* buffer, uint8_t* input) {
89+
char* res = fgets(buffer, sizeof(buffer), stdin);
90+
buffer[strcspn(buffer, "\n\r")] = 0;
91+
for (int i = 0; i < 512; i+=2) {
92+
hex_to_uint8(buffer[i], buffer[i+1], &input[i/2]);
93+
printf("%02x ", input[i/2]);
94+
}
95+
return res;
96+
}
97+
98+
int get_usb_message2(char* buffer, uint8_t* input) {
99+
num_read = read(fd, buffer, sizeof(buffer));
100+
buffer[strcspn(buffer, "\n\r")] = 0;
101+
for (int i = 0; i < 512; i+=2) {
102+
hex_to_uint8(buffer[i], buffer[i+1], &input[i/2]);
103+
printf("%02x ", input[i/2]);
104+
}
105+
return num_read;
106+
}
107+
108+
void simulate_firmware_execution(uint8_t* input) {
109+
//u2f_packet_process
110+
111+
//usb_processing_process
112+
//chain of call: firmware_main_loop -> usb_processing_process -> _usb_consume_incoming_packets ->
113+
// _usb_execute_packet
114+
//usb_processing_process(usb_processing_hww());
115+
/* First, process all the incoming USB traffic. */
116+
//usb_processing_process(usb_processing_hww());
117+
/*
118+
* If USB has generated events at the application level,
119+
* process them now.
120+
*/
121+
//hww_process();
122+
123+
//usb_packet_process but it puts the packet into queue
124+
//chain of call: usb_packet_process -> usb_processing_enqueue -> _build_packet
125+
memcpy(_out_report, input, sizeof(uint8_t) * sizeof(_out_report));
126+
usb_packet_process((const USB_FRAME*)_out_report);
127+
}
128+
129+
int main(void) {
130+
char buffer[1024];
131+
uint8_t input[1024];
132+
133+
printf("opening fifo\n");
134+
135+
fd = open(fifo_path, O_RDONLY);
136+
if (fd == -1) {
137+
perror("open");
138+
return 1;
139+
}
140+
printf("opened fifo\n");
141+
142+
while (get_usb_message2(buffer, input)) {
143+
printf("got input\n");
144+
if (strcmp(buffer, "") == 0) {
145+
break;
146+
}
147+
simulate_firmware_execution(input);
148+
}
149+
150+
close(fd);
151+
return 0;
152+
}

0 commit comments

Comments
 (0)