2022-01-22 12:00:14 +08:00
|
|
|
#include "usb_protocol.h"
|
2022-01-28 10:12:25 +08:00
|
|
|
#include "health.h"
|
2022-01-22 12:00:14 +08:00
|
|
|
|
|
|
|
extern int _app_start[0xc000]; // Only first 3 sectors of size 0x4000 are used
|
|
|
|
|
|
|
|
// Prototypes
|
|
|
|
void set_safety_mode(uint16_t mode, int16_t param);
|
|
|
|
bool is_car_safety_mode(uint16_t mode);
|
|
|
|
|
|
|
|
int get_health_pkt(void *dat) {
|
|
|
|
COMPILE_TIME_ASSERT(sizeof(struct health_t) <= USBPACKET_MAX_SIZE);
|
|
|
|
struct health_t * health = (struct health_t*)dat;
|
|
|
|
|
|
|
|
health->uptime_pkt = uptime_cnt;
|
|
|
|
health->voltage_pkt = adc_get_voltage();
|
|
|
|
health->current_pkt = current_board->read_current();
|
|
|
|
|
|
|
|
//Use the GPIO pin to determine ignition or use a CAN based logic
|
|
|
|
health->ignition_line_pkt = (uint8_t)(current_board->check_ignition());
|
|
|
|
health->ignition_can_pkt = (uint8_t)(ignition_can);
|
|
|
|
|
|
|
|
health->controls_allowed_pkt = controls_allowed;
|
|
|
|
health->gas_interceptor_detected_pkt = gas_interceptor_detected;
|
|
|
|
health->can_rx_errs_pkt = can_rx_errs;
|
|
|
|
health->can_send_errs_pkt = can_send_errs;
|
|
|
|
health->can_fwd_errs_pkt = can_fwd_errs;
|
|
|
|
health->gmlan_send_errs_pkt = gmlan_send_errs;
|
|
|
|
health->car_harness_status_pkt = car_harness_status;
|
|
|
|
health->usb_power_mode_pkt = usb_power_mode;
|
|
|
|
health->safety_mode_pkt = (uint8_t)(current_safety_mode);
|
|
|
|
health->safety_param_pkt = current_safety_param;
|
2022-01-25 18:35:57 +08:00
|
|
|
health->unsafe_mode_pkt = unsafe_mode;
|
2022-01-22 12:00:14 +08:00
|
|
|
health->power_save_enabled_pkt = (uint8_t)(power_save_status == POWER_SAVE_STATUS_ENABLED);
|
|
|
|
health->heartbeat_lost_pkt = (uint8_t)(heartbeat_lost);
|
2022-01-28 09:46:24 +08:00
|
|
|
health->blocked_msg_cnt_pkt = blocked_msg_cnt;
|
2022-01-22 12:00:14 +08:00
|
|
|
|
|
|
|
health->fault_status_pkt = fault_status;
|
|
|
|
health->faults_pkt = faults;
|
|
|
|
|
|
|
|
return sizeof(*health);
|
|
|
|
}
|
|
|
|
|
|
|
|
int get_rtc_pkt(void *dat) {
|
|
|
|
timestamp_t t = rtc_get_time();
|
|
|
|
(void)memcpy(dat, &t, sizeof(t));
|
|
|
|
return sizeof(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// send on serial, first byte to select the ring
|
|
|
|
void usb_cb_ep2_out(void *usbdata, int len) {
|
|
|
|
uint8_t *usbdata8 = (uint8_t *)usbdata;
|
|
|
|
uart_ring *ur = get_ring_by_number(usbdata8[0]);
|
|
|
|
if ((len != 0) && (ur != NULL)) {
|
|
|
|
if ((usbdata8[0] < 2U) || safety_tx_lin_hook(usbdata8[0] - 2U, &usbdata8[1], len - 1)) {
|
|
|
|
for (int i = 1; i < len; i++) {
|
|
|
|
while (!putc(ur, usbdata8[i])) {
|
|
|
|
// wait
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void usb_cb_ep3_out_complete(void) {
|
|
|
|
if (can_tx_check_min_slots_free(MAX_CAN_MSGS_PER_BULK_TRANSFER)) {
|
|
|
|
usb_outep3_resume_if_paused();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void usb_cb_enumeration_complete(void) {
|
|
|
|
puts("USB enumeration complete\n");
|
|
|
|
is_enumerated = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int usb_cb_control_msg(USB_Setup_TypeDef *setup, uint8_t *resp) {
|
|
|
|
unsigned int resp_len = 0;
|
|
|
|
uart_ring *ur = NULL;
|
|
|
|
timestamp_t t;
|
|
|
|
switch (setup->b.bRequest) {
|
|
|
|
// **** 0xa0: get rtc time
|
|
|
|
case 0xa0:
|
|
|
|
resp_len = get_rtc_pkt(resp);
|
|
|
|
break;
|
|
|
|
// **** 0xa1: set rtc year
|
|
|
|
case 0xa1:
|
|
|
|
t = rtc_get_time();
|
|
|
|
t.year = setup->b.wValue.w;
|
|
|
|
rtc_set_time(t);
|
|
|
|
break;
|
|
|
|
// **** 0xa2: set rtc month
|
|
|
|
case 0xa2:
|
|
|
|
t = rtc_get_time();
|
|
|
|
t.month = setup->b.wValue.w;
|
|
|
|
rtc_set_time(t);
|
|
|
|
break;
|
|
|
|
// **** 0xa3: set rtc day
|
|
|
|
case 0xa3:
|
|
|
|
t = rtc_get_time();
|
|
|
|
t.day = setup->b.wValue.w;
|
|
|
|
rtc_set_time(t);
|
|
|
|
break;
|
|
|
|
// **** 0xa4: set rtc weekday
|
|
|
|
case 0xa4:
|
|
|
|
t = rtc_get_time();
|
|
|
|
t.weekday = setup->b.wValue.w;
|
|
|
|
rtc_set_time(t);
|
|
|
|
break;
|
|
|
|
// **** 0xa5: set rtc hour
|
|
|
|
case 0xa5:
|
|
|
|
t = rtc_get_time();
|
|
|
|
t.hour = setup->b.wValue.w;
|
|
|
|
rtc_set_time(t);
|
|
|
|
break;
|
|
|
|
// **** 0xa6: set rtc minute
|
|
|
|
case 0xa6:
|
|
|
|
t = rtc_get_time();
|
|
|
|
t.minute = setup->b.wValue.w;
|
|
|
|
rtc_set_time(t);
|
|
|
|
break;
|
|
|
|
// **** 0xa7: set rtc second
|
|
|
|
case 0xa7:
|
|
|
|
t = rtc_get_time();
|
|
|
|
t.second = setup->b.wValue.w;
|
|
|
|
rtc_set_time(t);
|
|
|
|
break;
|
|
|
|
// **** 0xb0: set IR power
|
|
|
|
case 0xb0:
|
|
|
|
current_board->set_ir_power(setup->b.wValue.w);
|
|
|
|
break;
|
|
|
|
// **** 0xb1: set fan power
|
|
|
|
case 0xb1:
|
|
|
|
current_board->set_fan_power(setup->b.wValue.w);
|
|
|
|
break;
|
|
|
|
// **** 0xb2: get fan rpm
|
|
|
|
case 0xb2:
|
|
|
|
resp[0] = (fan_rpm & 0x00FFU);
|
|
|
|
resp[1] = ((fan_rpm & 0xFF00U) >> 8U);
|
|
|
|
resp_len = 2;
|
|
|
|
break;
|
|
|
|
// **** 0xb3: set phone power
|
|
|
|
case 0xb3:
|
|
|
|
current_board->set_phone_power(setup->b.wValue.w > 0U);
|
|
|
|
break;
|
|
|
|
// **** 0xc0: get CAN debug info
|
|
|
|
case 0xc0:
|
|
|
|
puts("can tx: "); puth(can_tx_cnt);
|
|
|
|
puts(" txd: "); puth(can_txd_cnt);
|
|
|
|
puts(" rx: "); puth(can_rx_cnt);
|
|
|
|
puts(" err: "); puth(can_err_cnt);
|
|
|
|
puts("\n");
|
|
|
|
break;
|
|
|
|
// **** 0xc1: get hardware type
|
|
|
|
case 0xc1:
|
|
|
|
resp[0] = hw_type;
|
|
|
|
resp_len = 1;
|
|
|
|
break;
|
|
|
|
// **** 0xd0: fetch serial number
|
|
|
|
case 0xd0:
|
|
|
|
// addresses are OTP
|
|
|
|
if (setup->b.wValue.w == 1U) {
|
|
|
|
(void)memcpy(resp, (uint8_t *)DEVICE_SERIAL_NUMBER_ADDRESS, 0x10);
|
|
|
|
resp_len = 0x10;
|
|
|
|
} else {
|
|
|
|
get_provision_chunk(resp);
|
|
|
|
resp_len = PROVISION_CHUNK_LEN;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xd1: enter bootloader mode
|
|
|
|
case 0xd1:
|
|
|
|
// this allows reflashing of the bootstub
|
|
|
|
switch (setup->b.wValue.w) {
|
|
|
|
case 0:
|
|
|
|
// only allow bootloader entry on debug builds
|
|
|
|
#ifdef ALLOW_DEBUG
|
|
|
|
puts("-> entering bootloader\n");
|
|
|
|
enter_bootloader_mode = ENTER_BOOTLOADER_MAGIC;
|
|
|
|
NVIC_SystemReset();
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
puts("-> entering softloader\n");
|
|
|
|
enter_bootloader_mode = ENTER_SOFTLOADER_MAGIC;
|
|
|
|
NVIC_SystemReset();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
puts("Bootloader mode invalid\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xd2: get health packet
|
|
|
|
case 0xd2:
|
|
|
|
resp_len = get_health_pkt(resp);
|
|
|
|
break;
|
|
|
|
// **** 0xd3: get first 64 bytes of signature
|
|
|
|
case 0xd3:
|
|
|
|
{
|
|
|
|
resp_len = 64;
|
|
|
|
char * code = (char*)_app_start;
|
|
|
|
int code_len = _app_start[0];
|
|
|
|
(void)memcpy(resp, &code[code_len], resp_len);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xd4: get second 64 bytes of signature
|
|
|
|
case 0xd4:
|
|
|
|
{
|
|
|
|
resp_len = 64;
|
|
|
|
char * code = (char*)_app_start;
|
|
|
|
int code_len = _app_start[0];
|
|
|
|
(void)memcpy(resp, &code[code_len + 64], resp_len);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xd6: get version
|
|
|
|
case 0xd6:
|
|
|
|
COMPILE_TIME_ASSERT(sizeof(gitversion) <= USBPACKET_MAX_SIZE);
|
|
|
|
(void)memcpy(resp, gitversion, sizeof(gitversion));
|
|
|
|
resp_len = sizeof(gitversion) - 1U;
|
|
|
|
break;
|
|
|
|
// **** 0xd8: reset ST
|
|
|
|
case 0xd8:
|
|
|
|
NVIC_SystemReset();
|
|
|
|
break;
|
|
|
|
// **** 0xd9: set ESP power
|
|
|
|
case 0xd9:
|
|
|
|
if (setup->b.wValue.w == 1U) {
|
|
|
|
current_board->set_gps_mode(GPS_ENABLED);
|
|
|
|
} else if (setup->b.wValue.w == 2U) {
|
|
|
|
current_board->set_gps_mode(GPS_BOOTMODE);
|
|
|
|
} else {
|
|
|
|
current_board->set_gps_mode(GPS_DISABLED);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xda: reset ESP, with optional boot mode
|
|
|
|
case 0xda:
|
|
|
|
current_board->set_gps_mode(GPS_DISABLED);
|
|
|
|
delay(1000000);
|
|
|
|
if (setup->b.wValue.w == 1U) {
|
|
|
|
current_board->set_gps_mode(GPS_BOOTMODE);
|
|
|
|
} else {
|
|
|
|
current_board->set_gps_mode(GPS_ENABLED);
|
|
|
|
}
|
|
|
|
delay(1000000);
|
|
|
|
current_board->set_gps_mode(GPS_ENABLED);
|
|
|
|
break;
|
|
|
|
// **** 0xdb: set GMLAN (white/grey) or OBD CAN (black) multiplexing mode
|
|
|
|
case 0xdb:
|
|
|
|
if(current_board->has_obd){
|
|
|
|
if (setup->b.wValue.w == 1U) {
|
|
|
|
// Enable OBD CAN
|
|
|
|
current_board->set_can_mode(CAN_MODE_OBD_CAN2);
|
|
|
|
} else {
|
|
|
|
// Disable OBD CAN
|
|
|
|
current_board->set_can_mode(CAN_MODE_NORMAL);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (setup->b.wValue.w == 1U) {
|
|
|
|
// GMLAN ON
|
|
|
|
if (setup->b.wIndex.w == 1U) {
|
|
|
|
can_set_gmlan(1);
|
|
|
|
} else if (setup->b.wIndex.w == 2U) {
|
|
|
|
can_set_gmlan(2);
|
|
|
|
} else {
|
|
|
|
puts("Invalid bus num for GMLAN CAN set\n");
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
can_set_gmlan(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// **** 0xdc: set safety mode
|
|
|
|
case 0xdc:
|
|
|
|
set_safety_mode(setup->b.wValue.w, (uint16_t) setup->b.wIndex.w);
|
|
|
|
break;
|
|
|
|
// **** 0xdd: get healthpacket and CANPacket versions
|
|
|
|
case 0xdd:
|
|
|
|
resp[0] = HEALTH_PACKET_VERSION;
|
|
|
|
resp[1] = CAN_PACKET_VERSION;
|
|
|
|
resp_len = 2;
|
|
|
|
break;
|
|
|
|
// **** 0xde: set can bitrate
|
|
|
|
case 0xde:
|
|
|
|
if (setup->b.wValue.w < BUS_CNT) {
|
|
|
|
// TODO: add sanity check, ideally check if value is correct(from array of correct values)
|
|
|
|
bus_config[setup->b.wValue.w].can_speed = setup->b.wIndex.w;
|
|
|
|
bool ret = can_init(CAN_NUM_FROM_BUS_NUM(setup->b.wValue.w));
|
|
|
|
UNUSED(ret);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xdf: set unsafe mode
|
|
|
|
case 0xdf:
|
|
|
|
// you can only set this if you are in a non car safety mode
|
|
|
|
if (!is_car_safety_mode(current_safety_mode)) {
|
|
|
|
unsafe_mode = setup->b.wValue.w;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xe0: uart read
|
|
|
|
case 0xe0:
|
|
|
|
ur = get_ring_by_number(setup->b.wValue.w);
|
|
|
|
if (!ur) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Remove this again and fix boardd code to hande the message bursts instead of single chars
|
|
|
|
if (ur == &uart_ring_gps) {
|
|
|
|
dma_pointer_handler(ur, DMA2_Stream5->NDTR);
|
|
|
|
}
|
|
|
|
|
|
|
|
// read
|
|
|
|
while ((resp_len < MIN(setup->b.wLength.w, USBPACKET_MAX_SIZE)) &&
|
|
|
|
getc(ur, (char*)&resp[resp_len])) {
|
|
|
|
++resp_len;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xe1: uart set baud rate
|
|
|
|
case 0xe1:
|
|
|
|
ur = get_ring_by_number(setup->b.wValue.w);
|
|
|
|
if (!ur) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
uart_set_baud(ur->uart, setup->b.wIndex.w);
|
|
|
|
break;
|
|
|
|
// **** 0xe2: uart set parity
|
|
|
|
case 0xe2:
|
|
|
|
ur = get_ring_by_number(setup->b.wValue.w);
|
|
|
|
if (!ur) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch (setup->b.wIndex.w) {
|
|
|
|
case 0:
|
|
|
|
// disable parity, 8-bit
|
|
|
|
ur->uart->CR1 &= ~(USART_CR1_PCE | USART_CR1_M);
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
// even parity, 9-bit
|
|
|
|
ur->uart->CR1 &= ~USART_CR1_PS;
|
|
|
|
ur->uart->CR1 |= USART_CR1_PCE | USART_CR1_M;
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
// odd parity, 9-bit
|
|
|
|
ur->uart->CR1 |= USART_CR1_PS;
|
|
|
|
ur->uart->CR1 |= USART_CR1_PCE | USART_CR1_M;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xe4: uart set baud rate extended
|
|
|
|
case 0xe4:
|
|
|
|
ur = get_ring_by_number(setup->b.wValue.w);
|
|
|
|
if (!ur) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
uart_set_baud(ur->uart, (int)setup->b.wIndex.w*300);
|
|
|
|
break;
|
|
|
|
// **** 0xe5: set CAN loopback (for testing)
|
|
|
|
case 0xe5:
|
|
|
|
can_loopback = (setup->b.wValue.w > 0U);
|
|
|
|
can_init_all();
|
|
|
|
break;
|
|
|
|
// **** 0xe6: set USB power
|
|
|
|
case 0xe6:
|
|
|
|
current_board->set_usb_power_mode(setup->b.wValue.w);
|
|
|
|
break;
|
|
|
|
// **** 0xe7: set power save state
|
|
|
|
case 0xe7:
|
|
|
|
set_power_save_state(setup->b.wValue.w);
|
|
|
|
break;
|
|
|
|
// **** 0xf0: k-line/l-line wake-up pulse for KWP2000 fast initialization
|
|
|
|
case 0xf0:
|
|
|
|
if(current_board->has_lin) {
|
|
|
|
bool k = (setup->b.wValue.w == 0U) || (setup->b.wValue.w == 2U);
|
|
|
|
bool l = (setup->b.wValue.w == 1U) || (setup->b.wValue.w == 2U);
|
|
|
|
if (bitbang_wakeup(k, l)) {
|
|
|
|
resp_len = -1; // do not clear NAK yet (wait for bit banging to finish)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xf1: Clear CAN ring buffer.
|
|
|
|
case 0xf1:
|
|
|
|
if (setup->b.wValue.w == 0xFFFFU) {
|
|
|
|
puts("Clearing CAN Rx queue\n");
|
|
|
|
can_clear(&can_rx_q);
|
|
|
|
} else if (setup->b.wValue.w < BUS_CNT) {
|
|
|
|
puts("Clearing CAN Tx queue\n");
|
|
|
|
can_clear(can_queues[setup->b.wValue.w]);
|
|
|
|
} else {
|
|
|
|
puts("Clearing CAN CAN ring buffer failed: wrong bus number\n");
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xf2: Clear UART ring buffer.
|
|
|
|
case 0xf2:
|
|
|
|
{
|
|
|
|
uart_ring * rb = get_ring_by_number(setup->b.wValue.w);
|
|
|
|
if (rb != NULL) {
|
|
|
|
puts("Clearing UART queue.\n");
|
|
|
|
clear_uart_buff(rb);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// **** 0xf3: Heartbeat. Resets heartbeat counter.
|
|
|
|
case 0xf3:
|
|
|
|
{
|
|
|
|
heartbeat_counter = 0U;
|
|
|
|
heartbeat_lost = false;
|
|
|
|
heartbeat_disabled = false;
|
|
|
|
heartbeat_engaged = (setup->b.wValue.w == 1U);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// **** 0xf4: k-line/l-line 5 baud initialization
|
|
|
|
case 0xf4:
|
|
|
|
if(current_board->has_lin) {
|
|
|
|
bool k = (setup->b.wValue.w == 0U) || (setup->b.wValue.w == 2U);
|
|
|
|
bool l = (setup->b.wValue.w == 1U) || (setup->b.wValue.w == 2U);
|
|
|
|
uint8_t five_baud_addr = (setup->b.wIndex.w & 0xFFU);
|
|
|
|
if (bitbang_five_baud_addr(k, l, five_baud_addr)) {
|
|
|
|
resp_len = -1; // do not clear NAK yet (wait for bit banging to finish)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xf5: set clock source mode
|
|
|
|
case 0xf5:
|
|
|
|
current_board->set_clock_source_mode(setup->b.wValue.w);
|
|
|
|
break;
|
|
|
|
// **** 0xf6: set siren enabled
|
|
|
|
case 0xf6:
|
|
|
|
siren_enabled = (setup->b.wValue.w != 0U);
|
|
|
|
break;
|
|
|
|
// **** 0xf7: set green led enabled
|
|
|
|
case 0xf7:
|
|
|
|
green_led_enabled = (setup->b.wValue.w != 0U);
|
|
|
|
break;
|
|
|
|
#ifdef ALLOW_DEBUG
|
|
|
|
// **** 0xf8: disable heartbeat checks
|
|
|
|
case 0xf8:
|
|
|
|
heartbeat_disabled = true;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
// **** 0xde: set CAN FD data bitrate
|
|
|
|
case 0xf9:
|
|
|
|
if (setup->b.wValue.w < CAN_CNT) {
|
2022-02-21 09:13:22 +08:00
|
|
|
// TODO: add sanity check, ideally check if value is correct (from array of correct values)
|
2022-01-22 12:00:14 +08:00
|
|
|
bus_config[setup->b.wValue.w].can_data_speed = setup->b.wIndex.w;
|
2022-02-21 09:13:22 +08:00
|
|
|
bus_config[setup->b.wValue.w].canfd_enabled = (setup->b.wIndex.w >= bus_config[setup->b.wValue.w].can_speed);
|
|
|
|
bus_config[setup->b.wValue.w].brs_enabled = (setup->b.wIndex.w > bus_config[setup->b.wValue.w].can_speed);
|
2022-01-22 12:00:14 +08:00
|
|
|
bool ret = can_init(CAN_NUM_FROM_BUS_NUM(setup->b.wValue.w));
|
|
|
|
UNUSED(ret);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
// **** 0xfa: check if CAN FD and BRS are enabled
|
|
|
|
case 0xfa:
|
|
|
|
if (setup->b.wValue.w < CAN_CNT) {
|
|
|
|
resp[0] = bus_config[setup->b.wValue.w].canfd_enabled;
|
|
|
|
resp[1] = bus_config[setup->b.wValue.w].brs_enabled;
|
|
|
|
resp_len = 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
puts("NO HANDLER ");
|
|
|
|
puth(setup->b.bRequest);
|
|
|
|
puts("\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return resp_len;
|
|
|
|
}
|