2#if defined(USE_ESP32_VARIANT_ESP32P4) || defined(USE_ESP32_VARIANT_ESP32S2) || defined(USE_ESP32_VARIANT_ESP32S3)
15#pragma GCC diagnostic ignored "-Wparentheses"
17using namespace bytebuffer;
19#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
20static void print_ep_desc(
const usb_ep_desc_t *ep_desc) {
21 const char *ep_type_str;
22 int type = ep_desc->bmAttributes & USB_BM_ATTRIBUTES_XFERTYPE_MASK;
25 case USB_BM_ATTRIBUTES_XFER_CONTROL:
28 case USB_BM_ATTRIBUTES_XFER_ISOC:
31 case USB_BM_ATTRIBUTES_XFER_BULK:
34 case USB_BM_ATTRIBUTES_XFER_INT:
43 "\t\t*** Endpoint descriptor ***\n"
45 "\t\tbDescriptorType %d\n"
46 "\t\tbEndpointAddress 0x%x\tEP %d %s\n"
47 "\t\tbmAttributes 0x%x\t%s\n"
48 "\t\twMaxPacketSize %d\n"
50 ep_desc->bLength, ep_desc->bDescriptorType, ep_desc->bEndpointAddress, USB_EP_DESC_GET_EP_NUM(ep_desc),
51 USB_EP_DESC_GET_EP_DIR(ep_desc) ?
"IN" :
"OUT", ep_desc->bmAttributes, ep_type_str, ep_desc->wMaxPacketSize,
55static void usbh_print_intf_desc(
const usb_intf_desc_t *intf_desc) {
57 "\t*** Interface descriptor ***\n"
59 "\tbDescriptorType %d\n"
60 "\tbInterfaceNumber %d\n"
61 "\tbAlternateSetting %d\n"
62 "\tbNumEndpoints %d\n"
63 "\tbInterfaceClass 0x%x\n"
65 intf_desc->bLength, intf_desc->bDescriptorType, intf_desc->bInterfaceNumber, intf_desc->bAlternateSetting,
66 intf_desc->bNumEndpoints, intf_desc->bInterfaceProtocol, intf_desc->iInterface);
69static void usbh_print_cfg_desc(
const usb_config_desc_t *cfg_desc) {
71 "*** Configuration descriptor ***\n"
73 " bDescriptorType %d\n"
75 " bNumInterfaces %d\n"
76 " bConfigurationValue %d\n"
77 " iConfiguration %d\n"
78 " bmAttributes 0x%x\n"
80 cfg_desc->bLength, cfg_desc->bDescriptorType, cfg_desc->wTotalLength, cfg_desc->bNumInterfaces,
81 cfg_desc->bConfigurationValue, cfg_desc->iConfiguration, cfg_desc->bmAttributes, cfg_desc->bMaxPower * 2);
84static void usb_client_print_device_descriptor(
const usb_device_desc_t *devc_desc) {
85 if (devc_desc == NULL) {
90 "*** Device descriptor ***\n"
92 " bDescriptorType %d\n"
94 " bDeviceClass 0x%x\n"
95 " bDeviceSubClass 0x%x\n"
96 " bDeviceProtocol 0x%x\n"
97 " bMaxPacketSize0 %d\n"
100 " bcdDevice %d.%d0\n"
101 " iManufacturer %d\n"
103 " iSerialNumber %d\n"
104 " bNumConfigurations %d",
105 devc_desc->bLength, devc_desc->bDescriptorType, ((devc_desc->bcdUSB >> 8) & 0xF),
106 ((devc_desc->bcdUSB >> 4) & 0xF), devc_desc->bDeviceClass, devc_desc->bDeviceSubClass,
107 devc_desc->bDeviceProtocol, devc_desc->bMaxPacketSize0, devc_desc->idVendor, devc_desc->idProduct,
108 ((devc_desc->bcdDevice >> 8) & 0xF), ((devc_desc->bcdDevice >> 4) & 0xF), devc_desc->iManufacturer,
109 devc_desc->iProduct, devc_desc->iSerialNumber, devc_desc->bNumConfigurations);
112static void usb_client_print_config_descriptor(
const usb_config_desc_t *cfg_desc,
113 print_class_descriptor_cb class_specific_cb) {
114 if (cfg_desc ==
nullptr) {
119 uint16_t w_total_length = cfg_desc->wTotalLength;
120 const usb_standard_desc_t *next_desc = (
const usb_standard_desc_t *) cfg_desc;
123 switch (next_desc->bDescriptorType) {
124 case USB_W_VALUE_DT_CONFIG:
125 usbh_print_cfg_desc((
const usb_config_desc_t *) next_desc);
127 case USB_W_VALUE_DT_INTERFACE:
128 usbh_print_intf_desc((
const usb_intf_desc_t *) next_desc);
130 case USB_W_VALUE_DT_ENDPOINT:
131 print_ep_desc((
const usb_ep_desc_t *) next_desc);
134 if (class_specific_cb) {
135 class_specific_cb(next_desc);
140 next_desc = usb_parse_next_descriptor(next_desc, w_total_length, &offset);
142 }
while (next_desc != NULL);
147static constexpr size_t DESC_STRING_BUF_SIZE = 128;
149static const char *get_descriptor_string(
const usb_str_desc_t *desc, std::span<char, DESC_STRING_BUF_SIZE> buffer) {
150 if (desc ==
nullptr || desc->bLength < 2)
151 return "(unspecified)";
152 int char_count = (desc->bLength - 2) / 2;
153 char *p = buffer.data();
154 char *
end = p + buffer.size() - 1;
155 for (
int i = 0; i != char_count && p <
end; i++) {
156 auto c = desc->wData[i];
158 *p++ =
static_cast<char>(c);
161 return buffer.data();
165static void client_event_cb(
const usb_host_client_event_msg_t *event_msg,
void *ptr) {
166 auto *client =
static_cast<USBClient *
>(ptr);
169 UsbEvent *
event = client->event_pool.allocate();
170 if (event ==
nullptr) {
172 client->event_queue.increment_dropped_count();
177 switch (event_msg->event) {
178 case USB_HOST_CLIENT_EVENT_NEW_DEV: {
179 ESP_LOGD(TAG,
"New device %d", event_msg->new_dev.address);
181 event->data.device_new.address = event_msg->new_dev.address;
184 case USB_HOST_CLIENT_EVENT_DEV_GONE: {
185 ESP_LOGD(TAG,
"Device gone");
187 event->data.device_gone.handle = event_msg->dev_gone.dev_hdl;
191 ESP_LOGD(TAG,
"Unknown event %d", event_msg->event);
192 client->event_pool.release(event);
198 client->event_queue.push(event);
201 client->enable_loop_soon_any_context();
204#if defined(USE_SOCKET_SELECT_SUPPORT) && defined(USE_WAKE_LOOP_THREADSAFE)
209 usb_host_client_config_t config{.is_synchronous =
false,
210 .max_num_event_msg = 5,
211 .async = {.client_event_callback = client_event_cb, .callback_arg =
this}};
212 auto err = usb_host_client_register(&config, &this->
handle_);
214 ESP_LOGE(TAG,
"client register failed: %s", esp_err_to_name(err));
222 usb_host_transfer_alloc(64, 0, &request.transfer);
223 request.client =
this;
234 ESP_LOGE(TAG,
"Failed to create USB task");
240 auto *client =
static_cast<USBClient *
>(arg);
245 usb_host_client_handle_events(this->
handle_, portMAX_DELAY);
250 bool had_work =
false;
254 while ((event = this->
event_queue.pop()) !=
nullptr) {
256 switch (event->
type) {
269 uint16_t dropped = this->
event_queue.get_and_reset_dropped_count();
271 ESP_LOGW(TAG,
"Dropped %u USB events due to queue overflow", dropped);
293 ESP_LOGW(TAG,
"Device open failed: %s", esp_err_to_name(err));
297 ESP_LOGD(TAG,
"Get descriptor device %d", this->
device_addr_);
298 const usb_device_desc_t *desc;
299 err = usb_host_get_device_descriptor(this->
device_handle_, &desc);
301 ESP_LOGW(TAG,
"Device get_desc failed: %s", esp_err_to_name(err));
305 ESP_LOGD(TAG,
"Device descriptor: vid %X pid %X", desc->idVendor, desc->idProduct);
306 if (desc->idVendor != this->vid_ || desc->idProduct != this->pid_) {
307 if (this->
vid_ != 0 || this->
pid_ != 0) {
308 ESP_LOGD(TAG,
"Not our device, closing");
313 usb_device_info_t dev_info;
316 ESP_LOGW(TAG,
"Device info failed: %s", esp_err_to_name(err));
321 char buf_manuf[DESC_STRING_BUF_SIZE];
322 char buf_product[DESC_STRING_BUF_SIZE];
323 char buf_serial[DESC_STRING_BUF_SIZE];
324 ESP_LOGD(TAG,
"Device connected: Manuf: %s; Prod: %s; Serial: %s",
325 get_descriptor_string(dev_info.str_desc_manufacturer, buf_manuf),
326 get_descriptor_string(dev_info.str_desc_product, buf_product),
327 get_descriptor_string(dev_info.str_desc_serial_num, buf_serial));
329#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
330 const usb_device_desc_t *device_desc;
331 err = usb_host_get_device_descriptor(this->
device_handle_, &device_desc);
333 usb_client_print_device_descriptor(device_desc);
334 const usb_config_desc_t *config_desc;
335 err = usb_host_get_active_config_descriptor(this->
device_handle_, &config_desc);
337 usb_client_print_config_descriptor(config_desc,
nullptr);
355static void control_callback(
const usb_transfer_t *xfer) {
358 trq->status.success = xfer->status == USB_TRANSFER_STATUS_COMPLETED;
359 trq->status.endpoint = xfer->bEndpointAddress;
360 trq->status.data = xfer->data_buffer;
361 trq->status.data_len = xfer->actual_num_bytes;
364 if (trq->callback !=
nullptr) {
365 trq->callback(trq->status);
370 trq->client->release_trq(trq);
386 if (mask == ALL_REQUESTS_IN_USE) {
387 ESP_LOGE(TAG,
"All %zu transfer slots in use", MAX_REQUESTS);
396 if (this->
trq_in_use_.compare_exchange_weak(mask, desired, std::memory_order::acquire)) {
397 auto i = __builtin_ctz(lsb);
413 ESP_LOGE(TAG,
"Device close failed: %s", esp_err_to_name(err));
423 const transfer_cb_t &callback,
const std::vector<uint8_t> &data) {
427 auto length = data.size();
428 if (
length > trq->transfer->data_buffer_size - SETUP_PACKET_SIZE) {
429 ESP_LOGE(TAG,
"Control transfer data size too large: %u > %u",
length,
430 trq->transfer->data_buffer_size - SETUP_PACKET_SIZE);
435 control_packet.put_uint8(
type);
436 control_packet.put_uint8(request);
437 control_packet.put_uint16(value);
438 control_packet.put_uint16(index);
439 control_packet.put_uint16(
length);
440 memcpy(trq->transfer->data_buffer, control_packet.get_data().data(), SETUP_PACKET_SIZE);
442 memcpy(trq->transfer->data_buffer + SETUP_PACKET_SIZE, data.data(),
length);
444 trq->callback = callback;
445 trq->transfer->bEndpointAddress =
type & USB_DIR_MASK;
446 trq->transfer->num_bytes =
static_cast<int>(
length + SETUP_PACKET_SIZE);
447 trq->transfer->callback =
reinterpret_cast<usb_transfer_cb_t
>(control_callback);
448 auto err = usb_host_transfer_submit_control(this->
handle_, trq->transfer);
450 ESP_LOGE(TAG,
"Failed to submit control transfer, err=%s", esp_err_to_name(err));
458static void transfer_callback(usb_transfer_t *xfer) {
461 trq->status.success = xfer->status == USB_TRANSFER_STATUS_COMPLETED;
462 trq->status.endpoint = xfer->bEndpointAddress;
463 trq->status.data = xfer->data_buffer;
464 trq->status.data_len = xfer->actual_num_bytes;
468 if (trq->callback !=
nullptr) {
469 trq->callback(trq->status);
476 trq->client->release_trq(trq);
492 if (trq ==
nullptr) {
493 ESP_LOGE(TAG,
"Too many requests queued");
496 if (
length > trq->transfer->data_buffer_size) {
497 ESP_LOGE(TAG,
"transfer_in: data length %u exceeds buffer size %u",
length, trq->transfer->data_buffer_size);
501 trq->callback = callback;
502 trq->transfer->callback = transfer_callback;
503 trq->transfer->bEndpointAddress = ep_address | USB_DIR_IN;
504 trq->transfer->num_bytes =
length;
505 auto err = usb_host_transfer_submit(trq->transfer);
507 ESP_LOGE(TAG,
"Failed to submit transfer, address=%x, length=%d, err=%x", ep_address,
length, err);
533 if (trq ==
nullptr) {
534 ESP_LOGE(TAG,
"Too many requests queued");
537 if (
length > trq->transfer->data_buffer_size) {
538 ESP_LOGE(TAG,
"transfer_out: data length %u exceeds buffer size %u",
length, trq->transfer->data_buffer_size);
542 trq->callback = callback;
543 trq->transfer->callback = transfer_callback;
544 trq->transfer->bEndpointAddress = ep_address | USB_DIR_OUT;
545 trq->transfer->num_bytes =
length;
546 memcpy(trq->transfer->data_buffer, data,
length);
547 auto err = usb_host_transfer_submit(trq->transfer);
549 ESP_LOGE(TAG,
"Failed to submit transfer, address=%x, length=%d, err=%x", ep_address,
length, err);
574 if (index >= MAX_REQUESTS) {
575 ESP_LOGE(TAG,
"Invalid TransferRequest pointer");
582 this->
trq_in_use_.fetch_and(mask, std::memory_order_release);
void wake_loop_threadsafe()
Wake the main event loop from another FreeRTOS task.
void mark_failed()
Mark this component as failed.
void disable_loop()
Disable this component's loop.
A class modelled on the Java ByteBuffer class.
TransferRequest * get_trq_()
usb_host_client_handle_t handle_
void on_opened(uint8_t addr)
TransferRequest requests_[MAX_REQUESTS]
void handle_open_state_()
bool transfer_out(uint8_t ep_address, const transfer_cb_t &callback, const uint8_t *data, uint16_t length)
Performs an output transfer operation.
void dump_config() override
void release_trq(TransferRequest *trq)
static void usb_task_fn(void *arg)
virtual void on_connected()
virtual void disconnect()
EventPool< UsbEvent, USB_EVENT_QUEUE_SIZE - 1 > event_pool
TaskHandle_t usb_task_handle_
bool process_usb_events_()
virtual void on_disconnected()
bool control_transfer(uint8_t type, uint8_t request, uint16_t value, uint16_t index, const transfer_cb_t &callback, const std::vector< uint8_t > &data={})
bool transfer_in(uint8_t ep_address, const transfer_cb_t &callback, uint16_t length)
Performs a transfer input operation.
void on_removed(usb_device_handle_t handle)
usb_device_handle_t device_handle_
void usb_task_loop() const
std::atomic< trq_bitmask_t > trq_in_use_
LockFreeQueue< UsbEvent, USB_EVENT_QUEUE_SIZE > event_queue
std::conditional<(MAX_REQUESTS<=16), uint16_t, uint32_t >::type trq_bitmask_t
std::function< void(const TransferStatus &)> transfer_cb_t
Application App
Global storage of Application pointer - only one Application can exist.
usb_transfer_t * transfer
struct esphome::usb_host::UsbEvent::@163::@164 device_new
struct esphome::usb_host::UsbEvent::@163::@165 device_gone
usb_device_handle_t handle
union esphome::usb_host::UsbEvent::@163 data