USB device stack CDC ACM support
The CDC ACM class is used as backend for different subsystems in Zephyr. However, its configuration may not be easy for the inexperienced user. Below is a description of the different use cases and some pitfalls.
The interface for CDC ACM user is Universal Asynchronous Receiver-Transmitter (UART) driver API. But there are two important differences in behavior to a real UART controller:
Data transfer is only possible after the USB device stack has been initialized and started, until then any data is discarded
If device is connected to the host, it still needs an application on the host side which requests the data
The devicetree compatible property for CDC ACM UART is
zephyr,cdc-acm-uart
.
CDC ACM support is automatically selected when USB device support is enabled
and a compatible node in the devicetree sources is present. If necessary,
CDC ACM support can be explicitly disabled by CONFIG_USB_CDC_ACM
.
About four CDC ACM UART instances can be defined and used,
limited by the maximum number of supported endpoints on the controller.
CDC ACM UART node is supposed to be child of a USB device controller node.
Since the designation of the controller nodes varies from vendor to vendor,
and our samples and application should be as generic as possible,
the default USB device controller is usually assigned an zephyr_udc0
node label. Often, CDC ACM UART is described in a devicetree overlay file
and looks like this:
&zephyr_udc0 {
cdc_acm_uart0: cdc_acm_uart0 {
compatible = "zephyr,cdc-acm-uart";
label = "CDC_ACM_0";
};
};
Samples USB CDC ACM Sample Application and USB HID CDC ACM Application have similar overlay files. And since no special properties are present, it may seem overkill to use devicetree to describe CDC ACM UART. The motivation behind using devicetree is the easy interchangeability of a real UART controller and CDC ACM UART in applications.
Console over CDC ACM UART
With the CDC ACM UART node from above and zephyr,console
property of the
chosen node, we can describe that CDC ACM UART is to be used with the console.
A similar overlay file is used by Console over CDC ACM UART Sample.
/ {
chosen {
zephyr,console = &cdc_acm_uart0;
};
};
&zephyr_udc0 {
cdc_acm_uart0: cdc_acm_uart0 {
compatible = "zephyr,cdc-acm-uart";
label = "CDC_ACM_0";
};
};
Before the application uses the console, it is recommended to wait for the DTR signal:
const struct device *const dev = DEVICE_DT_GET(DT_CHOSEN(zephyr_console));
uint32_t dtr = 0;
if (usb_enable(NULL)) {
return;
}
while (!dtr) {
uart_line_ctrl_get(dev, UART_LINE_CTRL_DTR, &dtr);
k_sleep(K_MSEC(100));
}
printk("nuqneH\n");
CDC ACM UART as backend
As for the console sample, it is possible to configure CDC ACM UART as backend for other subsystems by setting Chosen nodes properties.
List of few Zephyr specific chosen properties which can be used to select CDC ACM UART as backend for a subsystem or application:
zephyr,bt-c2h-uart
used in Bluetooth, for example see Bluetooth: HCI UARTzephyr,ot-uart
used in OpenThread, for example see OpenThread Co-Processorzephyr,shell-uart
used by shell for serial backend, for example see samples/subsys/shell/shell_modulezephyr,uart-mcumgr
used by SMP Server Sample