|
| 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