119 lines
4.7 KiB
C
119 lines
4.7 KiB
C
|
/*
|
||
|
* Copyright (c) 2022, sakumisu
|
||
|
*
|
||
|
* SPDX-License-Identifier: Apache-2.0
|
||
|
*/
|
||
|
#include "usbd_core.h"
|
||
|
#include "usbd_cdc.h"
|
||
|
|
||
|
const char *stop_name[] = { "1", "1.5", "2" };
|
||
|
const char *parity_name[] = { "N", "O", "E", "M", "S" };
|
||
|
|
||
|
static int cdc_acm_class_interface_request_handler(uint8_t busid, struct usb_setup_packet *setup, uint8_t **data, uint32_t *len)
|
||
|
{
|
||
|
USB_LOG_DBG("CDC Class request: "
|
||
|
"bRequest 0x%02x\r\n",
|
||
|
setup->bRequest);
|
||
|
|
||
|
struct cdc_line_coding line_coding;
|
||
|
bool dtr, rts;
|
||
|
uint8_t intf_num = LO_BYTE(setup->wIndex);
|
||
|
|
||
|
switch (setup->bRequest) {
|
||
|
case CDC_REQUEST_SET_LINE_CODING:
|
||
|
|
||
|
/*******************************************************************************/
|
||
|
/* Line Coding Structure */
|
||
|
/*-----------------------------------------------------------------------------*/
|
||
|
/* Offset | Field | Size | Value | Description */
|
||
|
/* 0 | dwDTERate | 4 | Number |Data terminal rate, in bits per second*/
|
||
|
/* 4 | bCharFormat | 1 | Number | Stop bits */
|
||
|
/* 0 - 1 Stop bit */
|
||
|
/* 1 - 1.5 Stop bits */
|
||
|
/* 2 - 2 Stop bits */
|
||
|
/* 5 | bParityType | 1 | Number | Parity */
|
||
|
/* 0 - None */
|
||
|
/* 1 - Odd */
|
||
|
/* 2 - Even */
|
||
|
/* 3 - Mark */
|
||
|
/* 4 - Space */
|
||
|
/* 6 | bDataBits | 1 | Number Data bits (5, 6, 7, 8 or 16). */
|
||
|
/*******************************************************************************/
|
||
|
memcpy(&line_coding, *data, setup->wLength);
|
||
|
USB_LOG_DBG("Set intf:%d linecoding <%d %d %s %s>\r\n",
|
||
|
intf_num,
|
||
|
line_coding.dwDTERate,
|
||
|
line_coding.bDataBits,
|
||
|
parity_name[line_coding.bParityType],
|
||
|
stop_name[line_coding.bCharFormat]);
|
||
|
|
||
|
usbd_cdc_acm_set_line_coding(busid, intf_num, &line_coding);
|
||
|
break;
|
||
|
|
||
|
case CDC_REQUEST_SET_CONTROL_LINE_STATE:
|
||
|
dtr = (setup->wValue & 0x0001);
|
||
|
rts = (setup->wValue & 0x0002);
|
||
|
USB_LOG_DBG("Set intf:%d DTR 0x%x,RTS 0x%x\r\n",
|
||
|
intf_num,
|
||
|
dtr,
|
||
|
rts);
|
||
|
usbd_cdc_acm_set_dtr(busid, intf_num, dtr);
|
||
|
usbd_cdc_acm_set_rts(busid, intf_num, rts);
|
||
|
break;
|
||
|
|
||
|
case CDC_REQUEST_GET_LINE_CODING:
|
||
|
usbd_cdc_acm_get_line_coding(busid, intf_num, &line_coding);
|
||
|
memcpy(*data, &line_coding, 7);
|
||
|
*len = 7;
|
||
|
USB_LOG_DBG("Get intf:%d linecoding %d %d %d %d\r\n",
|
||
|
intf_num,
|
||
|
line_coding.dwDTERate,
|
||
|
line_coding.bCharFormat,
|
||
|
line_coding.bParityType,
|
||
|
line_coding.bDataBits);
|
||
|
break;
|
||
|
case CDC_REQUEST_SEND_BREAK:
|
||
|
usbd_cdc_acm_send_break(busid, intf_num);
|
||
|
break;
|
||
|
default:
|
||
|
USB_LOG_WRN("Unhandled CDC Class bRequest 0x%02x\r\n", setup->bRequest);
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
struct usbd_interface *usbd_cdc_acm_init_intf(uint8_t busid, struct usbd_interface *intf)
|
||
|
{
|
||
|
intf->class_interface_handler = cdc_acm_class_interface_request_handler;
|
||
|
intf->class_endpoint_handler = NULL;
|
||
|
intf->vendor_handler = NULL;
|
||
|
intf->notify_handler = NULL;
|
||
|
|
||
|
return intf;
|
||
|
}
|
||
|
|
||
|
__WEAK void usbd_cdc_acm_set_line_coding(uint8_t busid, uint8_t intf, struct cdc_line_coding *line_coding)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
__WEAK void usbd_cdc_acm_get_line_coding(uint8_t busid, uint8_t intf, struct cdc_line_coding *line_coding)
|
||
|
{
|
||
|
line_coding->dwDTERate = 2000000;
|
||
|
line_coding->bDataBits = 8;
|
||
|
line_coding->bParityType = 0;
|
||
|
line_coding->bCharFormat = 0;
|
||
|
}
|
||
|
|
||
|
__WEAK void usbd_cdc_acm_set_dtr(uint8_t busid, uint8_t intf, bool dtr)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
__WEAK void usbd_cdc_acm_set_rts(uint8_t busid, uint8_t intf, bool rts)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
__WEAK void usbd_cdc_acm_send_break(uint8_t busid, uint8_t intf)
|
||
|
{
|
||
|
}
|