25#include <winpr/assert.h>
26#include <winpr/cast.h>
27#include <winpr/wtypes.h>
28#include <winpr/sysinfo.h>
29#include <winpr/collections.h>
33#include "libusb_udevice.h"
35#include "../common/urbdrc_types.h"
37#define BASIC_STATE_FUNC_DEFINED(_arg, _type) \
38 static _type udev_get_##_arg(IUDEVICE* idev) \
40 UDEVICE* pdev = (UDEVICE*)idev; \
43 static void udev_set_##_arg(IUDEVICE* idev, _type _t) \
45 UDEVICE* pdev = (UDEVICE*)idev; \
49#define BASIC_POINT_FUNC_DEFINED(_arg, _type) \
50 static _type udev_get_p_##_arg(IUDEVICE* idev) \
52 UDEVICE* pdev = (UDEVICE*)idev; \
55 static void udev_set_p_##_arg(IUDEVICE* idev, _type _t) \
57 UDEVICE* pdev = (UDEVICE*)idev; \
61#define BASIC_STATE_FUNC_REGISTER(_arg, _dev) \
62 _dev->iface.get_##_arg = udev_get_##_arg; \
63 (_dev)->iface.set_##_arg = udev_set_##_arg
65#if LIBUSB_API_VERSION >= 0x01000103
66#define HAVE_STREAM_ID_API 1
77 UINT32 OutputBufferSize;
79 t_isoch_transfer_cb cb;
81#if !defined(HAVE_STREAM_ID_API)
84} ASYNC_TRANSFER_USER_DATA;
86static void request_free(
void* value);
88static struct libusb_transfer* list_contains(wArrayList* list, UINT32 streamID)
93 count = ArrayList_Count(list);
94 for (
size_t x = 0; x < count; x++)
96 struct libusb_transfer* transfer = ArrayList_GetItem(list, x);
98#if defined(HAVE_STREAM_ID_API)
99 const UINT32 currentID = libusb_transfer_get_stream_id(transfer);
101 const ASYNC_TRANSFER_USER_DATA* user_data = (ASYNC_TRANSFER_USER_DATA*)transfer->user_data;
102 const UINT32 currentID = user_data->streamID;
104 if (currentID == streamID)
110static UINT32 stream_id_from_buffer(
struct libusb_transfer* transfer)
114#if defined(HAVE_STREAM_ID_API)
115 return libusb_transfer_get_stream_id(transfer);
117 ASYNC_TRANSFER_USER_DATA* user_data = (ASYNC_TRANSFER_USER_DATA*)transfer->user_data;
120 return user_data->streamID;
124static void set_stream_id_for_buffer(
struct libusb_transfer* transfer, UINT32 streamID)
126#if defined(HAVE_STREAM_ID_API)
127 libusb_transfer_set_stream_id(transfer, streamID);
129 ASYNC_TRANSFER_USER_DATA* user_data = (ASYNC_TRANSFER_USER_DATA*)transfer->user_data;
132 user_data->streamID = streamID;
136WINPR_ATTR_FORMAT_ARG(3, 8)
137static BOOL log_libusb_result_(wLog* log, DWORD lvl, WINPR_FORMAT_ARG const
char* fmt,
138 const
char* fkt, const
char* file,
size_t line,
int error, ...)
144 char buffer[8192] = WINPR_C_ARRAY_INIT;
145 va_list ap = WINPR_C_ARRAY_INIT;
147 (void)vsnprintf(buffer,
sizeof(buffer), fmt, ap);
150 WLog_Print(log, lvl,
"[%s:%" PRIuz
"]: %s: error %s[%d]", fkt, line, buffer,
151 libusb_error_name(error), error);
157#define log_libusb_result(log, lvl, fmt, error, ...) \
158 log_libusb_result_((log), (lvl), (fmt), __func__, __FILE__, __LINE__, error, ##__VA_ARGS__)
160const char* usb_interface_class_to_string(uint8_t c_class)
164 case LIBUSB_CLASS_PER_INTERFACE:
165 return "LIBUSB_CLASS_PER_INTERFACE";
166 case LIBUSB_CLASS_AUDIO:
167 return "LIBUSB_CLASS_AUDIO";
168 case LIBUSB_CLASS_COMM:
169 return "LIBUSB_CLASS_COMM";
170 case LIBUSB_CLASS_HID:
171 return "LIBUSB_CLASS_HID";
172 case LIBUSB_CLASS_PHYSICAL:
173 return "LIBUSB_CLASS_PHYSICAL";
174 case LIBUSB_CLASS_PRINTER:
175 return "LIBUSB_CLASS_PRINTER";
176 case LIBUSB_CLASS_IMAGE:
177 return "LIBUSB_CLASS_IMAGE";
178 case LIBUSB_CLASS_MASS_STORAGE:
179 return "LIBUSB_CLASS_MASS_STORAGE";
180 case LIBUSB_CLASS_HUB:
181 return "LIBUSB_CLASS_HUB";
182 case LIBUSB_CLASS_DATA:
183 return "LIBUSB_CLASS_DATA";
184 case LIBUSB_CLASS_SMART_CARD:
185 return "LIBUSB_CLASS_SMART_CARD";
186 case LIBUSB_CLASS_CONTENT_SECURITY:
187 return "LIBUSB_CLASS_CONTENT_SECURITY";
188 case LIBUSB_CLASS_VIDEO:
189 return "LIBUSB_CLASS_VIDEO";
190 case LIBUSB_CLASS_PERSONAL_HEALTHCARE:
191 return "LIBUSB_CLASS_PERSONAL_HEALTHCARE";
192 case LIBUSB_CLASS_DIAGNOSTIC_DEVICE:
193 return "LIBUSB_CLASS_DIAGNOSTIC_DEVICE";
194 case LIBUSB_CLASS_WIRELESS:
195 return "LIBUSB_CLASS_WIRELESS";
196 case LIBUSB_CLASS_APPLICATION:
197 return "LIBUSB_CLASS_APPLICATION";
198 case LIBUSB_CLASS_VENDOR_SPEC:
199 return "LIBUSB_CLASS_VENDOR_SPEC";
201 return "UNKNOWN_DEVICE_CLASS";
205static ASYNC_TRANSFER_USER_DATA* async_transfer_user_data_new(IUDEVICE* idev, UINT32 MessageId,
206 size_t offset,
size_t BufferSize,
207 const BYTE* data,
size_t packetSize,
208 BOOL NoAck, t_isoch_transfer_cb cb,
211 ASYNC_TRANSFER_USER_DATA* user_data =
nullptr;
214 if (BufferSize > UINT32_MAX)
217 user_data = calloc(1,
sizeof(ASYNC_TRANSFER_USER_DATA));
221 user_data->data = Stream_New(
nullptr, offset + BufferSize + packetSize);
223 if (!user_data->data)
229 Stream_Seek(user_data->data, offset);
231 memcpy(Stream_Pointer(user_data->data), data, BufferSize);
233 user_data->OutputBufferSize = (UINT32)BufferSize;
235 user_data->noack = NoAck;
237 user_data->callback = callback;
238 user_data->idev = idev;
239 user_data->MessageId = MessageId;
241 user_data->queue = pdev->request_queue;
246static void async_transfer_user_data_free(ASYNC_TRANSFER_USER_DATA* user_data)
250 Stream_Free(user_data->data, TRUE);
255static void LIBUSB_CALL func_iso_callback(
struct libusb_transfer* transfer)
257 ASYNC_TRANSFER_USER_DATA* user_data = (ASYNC_TRANSFER_USER_DATA*)transfer->user_data;
258 const UINT32 streamID = stream_id_from_buffer(transfer);
259 wArrayList* list = user_data->queue;
261 ArrayList_Lock(list);
262 switch (transfer->status)
264 case LIBUSB_TRANSFER_COMPLETED:
267 BYTE* dataStart = Stream_Pointer(user_data->data);
268 if (!Stream_SetPosition(user_data->data,
272 for (uint32_t i = 0; i < WINPR_ASSERTING_INT_CAST(uint32_t, transfer->num_iso_packets);
275 const UINT32 act_len = transfer->iso_packet_desc[i].actual_length;
276 Stream_Write_UINT32(user_data->data, index);
277 Stream_Write_UINT32(user_data->data, act_len);
278 Stream_Write_UINT32(user_data->data, transfer->iso_packet_desc[i].status);
280 if (transfer->iso_packet_desc[i].status != USBD_STATUS_SUCCESS)
281 user_data->ErrorCount++;
284 const unsigned char* packetBuffer =
285 libusb_get_iso_packet_buffer_simple(transfer, i);
286 BYTE* data = dataStart + index;
288 if (data != packetBuffer)
289 memmove(data, packetBuffer, act_len);
297 case LIBUSB_TRANSFER_CANCELLED:
300 case LIBUSB_TRANSFER_TIMED_OUT:
303 case LIBUSB_TRANSFER_ERROR:
305 const UINT32 InterfaceId =
306 ((STREAM_ID_PROXY << 30) | user_data->idev->get_ReqCompletion(user_data->idev));
308 if (list_contains(list, streamID))
310 if (!user_data->noack)
312 const UINT32 RequestID = streamID & INTERFACE_ID_MASK;
313 user_data->cb(user_data->idev, user_data->callback, user_data->data,
314 InterfaceId, user_data->noack, user_data->MessageId, RequestID,
315 WINPR_ASSERTING_INT_CAST(uint32_t, transfer->num_iso_packets),
316 transfer->status, user_data->StartFrame, user_data->ErrorCount,
317 user_data->OutputBufferSize);
318 user_data->data =
nullptr;
320 ArrayList_Remove(list, transfer);
327 ArrayList_Unlock(list);
330static const LIBUSB_ENDPOINT_DESCEIPTOR* func_get_ep_desc(LIBUSB_CONFIG_DESCRIPTOR* LibusbConfig,
332 UINT32 EndpointAddress)
335 const LIBUSB_INTERFACE*
interface = LibusbConfig->interface;
337 for (UINT32 inum = 0; inum < MsConfig->NumInterfaces; inum++)
339 BYTE alt = MsInterfaces[inum]->AlternateSetting;
340 const LIBUSB_ENDPOINT_DESCEIPTOR* endpoint = interface[inum].altsetting[alt].endpoint;
342 for (UINT32 pnum = 0; pnum < MsInterfaces[inum]->NumberOfPipes; pnum++)
344 if (endpoint[pnum].bEndpointAddress == EndpointAddress)
346 return &endpoint[pnum];
354static void LIBUSB_CALL func_bulk_transfer_cb(
struct libusb_transfer* transfer)
356 ASYNC_TRANSFER_USER_DATA* user_data =
nullptr;
357 uint32_t streamID = 0;
358 wArrayList* list =
nullptr;
360 user_data = (ASYNC_TRANSFER_USER_DATA*)transfer->user_data;
363 WLog_ERR(TAG,
"Invalid transfer->user_data!");
366 list = user_data->queue;
367 ArrayList_Lock(list);
368 streamID = stream_id_from_buffer(transfer);
370 if (list_contains(list, streamID))
372 const UINT32 InterfaceId =
373 ((STREAM_ID_PROXY << 30) | user_data->idev->get_ReqCompletion(user_data->idev));
374 const UINT32 RequestID = streamID & INTERFACE_ID_MASK;
376 user_data->cb(user_data->idev, user_data->callback, user_data->data, InterfaceId,
377 user_data->noack, user_data->MessageId, RequestID,
378 WINPR_ASSERTING_INT_CAST(uint32_t, transfer->num_iso_packets),
379 transfer->status, user_data->StartFrame, user_data->ErrorCount,
380 WINPR_ASSERTING_INT_CAST(uint32_t, transfer->actual_length));
381 user_data->data =
nullptr;
382 ArrayList_Remove(list, transfer);
384 ArrayList_Unlock(list);
390 if (!urbdrc || !status)
396 *status = USBD_STATUS_SUCCESS;
399 case LIBUSB_ERROR_IO:
400 *status = USBD_STATUS_STALL_PID;
403 case LIBUSB_ERROR_INVALID_PARAM:
404 *status = USBD_STATUS_INVALID_PARAMETER;
407 case LIBUSB_ERROR_ACCESS:
408 *status = USBD_STATUS_NOT_ACCESSED;
411 case LIBUSB_ERROR_NO_DEVICE:
412 *status = USBD_STATUS_DEVICE_GONE;
416 if (!(pdev->status & URBDRC_DEVICE_NOT_FOUND))
417 pdev->status |= URBDRC_DEVICE_NOT_FOUND;
422 case LIBUSB_ERROR_NOT_FOUND:
423 *status = USBD_STATUS_STALL_PID;
426 case LIBUSB_ERROR_BUSY:
427 *status = USBD_STATUS_STALL_PID;
430 case LIBUSB_ERROR_TIMEOUT:
431 *status = USBD_STATUS_TIMEOUT;
434 case LIBUSB_ERROR_OVERFLOW:
435 *status = USBD_STATUS_STALL_PID;
438 case LIBUSB_ERROR_PIPE:
439 *status = USBD_STATUS_STALL_PID;
442 case LIBUSB_ERROR_INTERRUPTED:
443 *status = USBD_STATUS_STALL_PID;
446 case LIBUSB_ERROR_NO_MEM:
447 *status = USBD_STATUS_NO_MEMORY;
450 case LIBUSB_ERROR_NOT_SUPPORTED:
451 *status = USBD_STATUS_NOT_SUPPORTED;
454 case LIBUSB_ERROR_OTHER:
455 *status = USBD_STATUS_STALL_PID;
459 *status = USBD_STATUS_SUCCESS;
466static int func_config_release_all_interface(
URBDRC_PLUGIN* urbdrc,
467 LIBUSB_DEVICE_HANDLE* libusb_handle,
468 UINT32 NumInterfaces)
470 if (NumInterfaces > INT32_MAX)
472 for (INT32 i = 0; i < (INT32)NumInterfaces; i++)
474 int ret = libusb_release_interface(libusb_handle, i);
476 if (log_libusb_result(urbdrc->log, WLOG_WARN,
"libusb_release_interface", ret))
483static int func_claim_all_interface(
URBDRC_PLUGIN* urbdrc, LIBUSB_DEVICE_HANDLE* libusb_handle,
488 for (
int i = 0; i < NumInterfaces; i++)
490 ret = libusb_claim_interface(libusb_handle, i);
492 if (log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_claim_interface", ret))
499static LIBUSB_DEVICE* udev_get_libusb_dev(libusb_context* context, uint8_t bus_number,
502 LIBUSB_DEVICE** libusb_list =
nullptr;
503 LIBUSB_DEVICE* device =
nullptr;
504 const ssize_t total_device = libusb_get_device_list(context, &libusb_list);
506 for (ssize_t i = 0; i < total_device; i++)
508 LIBUSB_DEVICE* dev = libusb_list[i];
509 if ((bus_number == libusb_get_bus_number(dev)) &&
510 (dev_number == libusb_get_device_address(dev)))
513 libusb_unref_device(dev);
516 libusb_free_device_list(libusb_list, 0);
520static LIBUSB_DEVICE_DESCRIPTOR* udev_new_descript(
URBDRC_PLUGIN* urbdrc, LIBUSB_DEVICE* libusb_dev)
523 LIBUSB_DEVICE_DESCRIPTOR* descriptor =
524 (LIBUSB_DEVICE_DESCRIPTOR*)calloc(1,
sizeof(LIBUSB_DEVICE_DESCRIPTOR));
527 ret = libusb_get_device_descriptor(libusb_dev, descriptor);
529 if (log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_get_device_descriptor", ret))
538static int libusb_udev_select_interface(IUDEVICE* idev, BYTE InterfaceNumber, BYTE AlternateSetting)
542 if (!pdev || !pdev->urbdrc)
548 libusb_set_interface_alt_setting(pdev->libusb_handle, InterfaceNumber, AlternateSetting);
550 log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_set_interface_alt_setting", error);
559 UINT32 MsOutSize = 0;
561 if (!pdev || !pdev->LibusbConfig || !pdev->urbdrc || !MsConfig)
565 LIBUSB_CONFIG_DESCRIPTOR* LibusbConfig = pdev->LibusbConfig;
567 if (LibusbConfig->bNumInterfaces != MsConfig->NumInterfaces)
569 WLog_Print(urbdrc->log, WLOG_ERROR,
570 "Select Configuration: Libusb NumberInterfaces(%" PRIu8
") is different "
571 "with MsConfig NumberInterfaces(%" PRIu32
")",
572 LibusbConfig->bNumInterfaces, MsConfig->NumInterfaces);
579 for (UINT32 inum = 0; inum < MsConfig->NumInterfaces; inum++)
582 if (MsInterface->InterfaceNumber >= MsConfig->NumInterfaces)
584 WLog_Print(urbdrc->log, WLOG_ERROR,
585 "MSUSB_CONFIG_DESCRIPTOR::NumInterfaces (%" PRIu32
586 " <= MSUSB_INTERFACE_DESCRIPTOR::InterfaceNumber( %" PRIu8
")",
587 MsConfig->NumInterfaces, MsInterface->InterfaceNumber);
591 const LIBUSB_INTERFACE* LibusbInterface =
592 &LibusbConfig->interface[MsInterface->InterfaceNumber];
593 if (MsInterface->AlternateSetting >= LibusbInterface->num_altsetting)
595 WLog_Print(urbdrc->log, WLOG_ERROR,
596 "LIBUSB_INTERFACE::num_altsetting (%" PRId32
597 " <= MSUSB_INTERFACE_DESCRIPTOR::AlternateSetting( %" PRIu8
")",
598 LibusbInterface->num_altsetting, MsInterface->AlternateSetting);
603 for (UINT32 inum = 0; inum < MsConfig->NumInterfaces; inum++)
608 const LIBUSB_INTERFACE* LibusbInterface =
609 &LibusbConfig->interface[MsInterface->InterfaceNumber];
610 const LIBUSB_INTERFACE_DESCRIPTOR* LibusbAltsetting =
611 &LibusbInterface->altsetting[MsInterface->AlternateSetting];
612 const BYTE LibusbNumEndpoint = LibusbAltsetting->bNumEndpoints;
616 for (UINT32 pnum = 0; pnum < LibusbNumEndpoint; pnum++)
621 if (pnum < MsInterface->NumberOfPipes && MsInterface->MsPipes)
624 t_MsPipe->MaximumPacketSize = MsPipe->MaximumPacketSize;
625 t_MsPipe->MaximumTransferSize = MsPipe->MaximumTransferSize;
626 t_MsPipe->PipeFlags = MsPipe->PipeFlags;
630 t_MsPipe->MaximumPacketSize = 0;
631 t_MsPipe->MaximumTransferSize = 0xffffffff;
632 t_MsPipe->PipeFlags = 0;
635 t_MsPipe->PipeHandle = 0;
636 t_MsPipe->bEndpointAddress = 0;
637 t_MsPipe->bInterval = 0;
638 t_MsPipe->PipeType = 0;
639 t_MsPipe->InitCompleted = 0;
640 t_MsPipes[pnum] = t_MsPipe;
643 msusb_mspipes_replace(MsInterface, t_MsPipes, LibusbNumEndpoint);
654 MsConfig->ConfigurationHandle = (uint32_t)MsConfig->bConfigurationValue |
655 ((uint32_t)pdev->bus_number << 24) |
656 (((uint32_t)pdev->dev_number << 16) & 0xFF0000);
657 MsInterfaces = MsConfig->MsInterfaces;
659 for (UINT32 inum = 0; inum < MsConfig->NumInterfaces; inum++)
664 const LIBUSB_INTERFACE* LibusbInterface =
665 &LibusbConfig->interface[MsInterface->InterfaceNumber];
666 const LIBUSB_INTERFACE_DESCRIPTOR* LibusbAltsetting =
667 &LibusbInterface->altsetting[MsInterface->AlternateSetting];
674 MsInterface->InterfaceHandle =
675 WINPR_ASSERTING_INT_CAST(UINT32, (LibusbAltsetting->bInterfaceNumber |
676 (LibusbAltsetting->bAlternateSetting << 8) |
677 (pdev->dev_number << 16) | (pdev->bus_number << 24)));
678 const size_t len = 16 + (MsInterface->NumberOfPipes * 20);
679 MsInterface->Length = WINPR_ASSERTING_INT_CAST(UINT16, len);
680 MsInterface->bInterfaceClass = LibusbAltsetting->bInterfaceClass;
681 MsInterface->bInterfaceSubClass = LibusbAltsetting->bInterfaceSubClass;
682 MsInterface->bInterfaceProtocol = LibusbAltsetting->bInterfaceProtocol;
683 MsInterface->InitCompleted = 1;
685 const BYTE LibusbNumEndpoint = LibusbAltsetting->bNumEndpoints;
687 for (UINT32 pnum = 0; pnum < LibusbNumEndpoint; pnum++)
693 const LIBUSB_ENDPOINT_DESCEIPTOR* LibusbEndpoint = &LibusbAltsetting->endpoint[pnum];
700 MsPipe->PipeHandle = LibusbEndpoint->bEndpointAddress |
701 (((uint32_t)pdev->dev_number << 16) & 0xFF0000) |
702 (((uint32_t)pdev->bus_number << 24) & 0xFF000000);
704 unsigned max = LibusbEndpoint->wMaxPacketSize & 0x07ff;
705 BYTE attr = LibusbEndpoint->bmAttributes;
707 if ((attr & 0x3) == 1 || (attr & 0x3) == 3)
709 max *= (1 + ((LibusbEndpoint->wMaxPacketSize >> 11) & 3));
712 MsPipe->MaximumPacketSize = WINPR_ASSERTING_INT_CAST(uint16_t, max);
713 MsPipe->bEndpointAddress = LibusbEndpoint->bEndpointAddress;
714 MsPipe->bInterval = LibusbEndpoint->bInterval;
715 MsPipe->PipeType = attr & 0x3;
716 MsPipe->InitCompleted = 1;
720 MsConfig->MsOutSize = WINPR_ASSERTING_INT_CAST(
int, MsOutSize);
721 MsConfig->InitCompleted = 1;
724 if (MsConfig != pdev->MsConfig)
726 msusb_msconfig_free(pdev->MsConfig);
727 pdev->MsConfig = MsConfig;
733static int libusb_udev_select_configuration(IUDEVICE* idev, UINT32 bConfigurationValue)
737 LIBUSB_DEVICE_HANDLE* libusb_handle =
nullptr;
738 LIBUSB_DEVICE* libusb_dev =
nullptr;
740 LIBUSB_CONFIG_DESCRIPTOR** LibusbConfig =
nullptr;
743 if (!pdev || !pdev->MsConfig || !pdev->LibusbConfig || !pdev->urbdrc)
746 urbdrc = pdev->urbdrc;
747 MsConfig = pdev->MsConfig;
748 libusb_handle = pdev->libusb_handle;
749 libusb_dev = pdev->libusb_dev;
750 LibusbConfig = &pdev->LibusbConfig;
752 if (MsConfig->InitCompleted)
754 func_config_release_all_interface(pdev->urbdrc, libusb_handle,
755 (*LibusbConfig)->bNumInterfaces);
759 if (bConfigurationValue == 0)
760 ret = libusb_set_configuration(libusb_handle, -1);
762 ret = libusb_set_configuration(libusb_handle,
763 WINPR_ASSERTING_INT_CAST(
int, bConfigurationValue));
765 if (log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_set_configuration", ret))
767 func_claim_all_interface(urbdrc, libusb_handle, (*LibusbConfig)->bNumInterfaces);
772 ret = libusb_get_active_config_descriptor(libusb_dev, LibusbConfig);
774 if (log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_set_configuration", ret))
776 func_claim_all_interface(urbdrc, libusb_handle, (*LibusbConfig)->bNumInterfaces);
781 func_claim_all_interface(urbdrc, libusb_handle, (*LibusbConfig)->bNumInterfaces);
785static int libusb_udev_control_pipe_request(IUDEVICE* idev, WINPR_ATTR_UNUSED UINT32 RequestId,
786 UINT32 EndpointAddress, UINT32* UsbdStatus,
int command)
791 WINPR_ASSERT(EndpointAddress <= UINT8_MAX);
799 idev->cancel_all_transfer_request(idev);
805 error = libusb_control_transfer(pdev->libusb_handle,
806 (uint8_t)LIBUSB_ENDPOINT_OUT |
807 (uint8_t)LIBUSB_RECIPIENT_ENDPOINT,
808 LIBUSB_REQUEST_SET_FEATURE, ENDPOINT_HALT,
809 (uint16_t)EndpointAddress,
nullptr, 0, 1000);
813 idev->cancel_all_transfer_request(idev);
814 error = libusb_clear_halt(pdev->libusb_handle, (uint8_t)EndpointAddress);
827static UINT32 libusb_udev_control_query_device_text(IUDEVICE* idev, UINT32 TextType,
828 UINT16 LocaleId, UINT8* BufferSize,
832 LIBUSB_DEVICE_DESCRIPTOR* devDescriptor =
nullptr;
833 const char strDesc[] =
"Generic Usb String";
834 char deviceLocation[25] = WINPR_C_ARRAY_INIT;
836 BYTE device_address = 0;
839 WCHAR* text = (WCHAR*)Buffer;
842 const UINT8 inSize = *BufferSize;
845 if (!pdev || !pdev->devDescriptor || !pdev->urbdrc)
846 return ERROR_INVALID_DATA;
848 urbdrc = pdev->urbdrc;
849 devDescriptor = pdev->devDescriptor;
853 case DeviceTextDescription:
855 BYTE data[0x100] = WINPR_C_ARRAY_INIT;
856 ret = libusb_get_string_descriptor(pdev->libusb_handle, devDescriptor->iProduct,
857 LocaleId, data, 0xFF);
866 if ((ret <= 0) || (ret <= 4) || (slen <= 4) || (locale != LIBUSB_DT_STRING) ||
869 const char* msg =
"SHORT_DESCRIPTOR";
871 msg = libusb_error_name(ret);
872 WLog_Print(urbdrc->log, WLOG_DEBUG,
873 "libusb_get_string_descriptor: "
874 "%s [%d], iProduct: %" PRIu8
"!",
875 msg, ret, devDescriptor->iProduct);
877 size_t len = MIN(
sizeof(strDesc), inSize);
878 for (
size_t i = 0; i < len; i++)
879 text[i] = (WCHAR)strDesc[i];
881 *BufferSize = (BYTE)(len *
sizeof(WCHAR));
885 size_t maxlen = inSize;
887 if (inSize >
sizeof(WCHAR))
889 maxlen -=
sizeof(WCHAR);
897 _wcsnlen((WCHAR*)&data[2], (
sizeof(data) /
sizeof(WCHAR)) - 1);
898 len = MIN((BYTE)ret - 2, slen);
899 len = MIN(len, rchar *
sizeof(WCHAR));
900 len = MIN(len, maxlen);
902 memcpy(Buffer, &data[2], len);
906 if (Buffer[len] !=
'\0')
908 Buffer[len++] =
'\0';
909 Buffer[len++] =
'\0';
912 *BufferSize = (BYTE)len;
917 case DeviceTextLocationInformation:
919 bus_number = libusb_get_bus_number(pdev->libusb_dev);
920 device_address = libusb_get_device_address(pdev->libusb_dev);
921 (void)sprintf_s(deviceLocation,
sizeof(deviceLocation),
922 "Port_#%04" PRIu8
".Hub_#%04" PRIu8
"", device_address, bus_number);
924 size_t len = strnlen(deviceLocation,
925 MIN(
sizeof(deviceLocation), (inSize > 0) ? inSize - 1U : 0));
926 for (
size_t i = 0; i < len; i++)
927 text[i] = (WCHAR)deviceLocation[i];
929 *BufferSize = (UINT8)(len *
sizeof(WCHAR));
934 WLog_Print(urbdrc->log, WLOG_DEBUG,
"Query Text: unknown TextType %" PRIu32
"",
936 return ERROR_INVALID_DATA;
942static int libusb_udev_os_feature_descriptor_request(IUDEVICE* idev,
943 WINPR_ATTR_UNUSED UINT32 RequestId,
944 BYTE Recipient, BYTE InterfaceNumber,
945 BYTE Ms_PageIndex, UINT16 Ms_featureDescIndex,
946 UINT32* UsbdStatus, UINT32* BufferSize,
947 BYTE* Buffer, UINT32 Timeout)
950 BYTE ms_string_desc[0x13] = WINPR_C_ARRAY_INIT;
954 WINPR_ASSERT(UsbdStatus);
955 WINPR_ASSERT(BufferSize);
956 WINPR_ASSERT(*BufferSize <= UINT16_MAX);
961 error = libusb_control_transfer(pdev->libusb_handle, LIBUSB_ENDPOINT_IN | Recipient,
962 LIBUSB_REQUEST_GET_DESCRIPTOR, 0x03ee, 0, ms_string_desc, 0x12,
965 log_libusb_result(pdev->urbdrc->log, WLOG_DEBUG,
"libusb_control_transfer", error);
969 const BYTE bMS_Vendorcode = ms_string_desc[16];
971 error = libusb_control_transfer(
973 (uint8_t)LIBUSB_ENDPOINT_IN | (uint8_t)LIBUSB_REQUEST_TYPE_VENDOR | Recipient,
974 bMS_Vendorcode, (UINT16)((InterfaceNumber << 8) | Ms_PageIndex), Ms_featureDescIndex,
975 Buffer, (UINT16)*BufferSize, Timeout);
976 log_libusb_result(pdev->urbdrc->log, WLOG_DEBUG,
"libusb_control_transfer", error);
979 *BufferSize = (UINT32)error;
983 *UsbdStatus = USBD_STATUS_STALL_PID;
985 *UsbdStatus = USBD_STATUS_SUCCESS;
987 return ERROR_SUCCESS;
990static int libusb_udev_query_device_descriptor(IUDEVICE* idev,
int offset)
997 return pdev->devDescriptor->bLength;
999 case B_DESCRIPTOR_TYPE:
1000 return pdev->devDescriptor->bDescriptorType;
1003 return pdev->devDescriptor->bcdUSB;
1005 case B_DEVICE_CLASS:
1006 return pdev->devDescriptor->bDeviceClass;
1008 case B_DEVICE_SUBCLASS:
1009 return pdev->devDescriptor->bDeviceSubClass;
1011 case B_DEVICE_PROTOCOL:
1012 return pdev->devDescriptor->bDeviceProtocol;
1014 case B_MAX_PACKET_SIZE0:
1015 return pdev->devDescriptor->bMaxPacketSize0;
1018 return pdev->devDescriptor->idVendor;
1021 return pdev->devDescriptor->idProduct;
1024 return pdev->devDescriptor->bcdDevice;
1026 case I_MANUFACTURER:
1027 return pdev->devDescriptor->iManufacturer;
1030 return pdev->devDescriptor->iProduct;
1032 case I_SERIAL_NUMBER:
1033 return pdev->devDescriptor->iSerialNumber;
1035 case B_NUM_CONFIGURATIONS:
1036 return pdev->devDescriptor->bNumConfigurations;
1043static BOOL libusb_udev_detach_kernel_driver(IUDEVICE* idev)
1049 if (!pdev || !pdev->LibusbConfig || !pdev->libusb_handle || !pdev->urbdrc)
1055 urbdrc = pdev->urbdrc;
1057 if ((pdev->status & URBDRC_DEVICE_DETACH_KERNEL) == 0)
1059 for (
int i = 0; i < pdev->LibusbConfig->bNumInterfaces; i++)
1061 err = libusb_kernel_driver_active(pdev->libusb_handle, i);
1062 log_libusb_result(urbdrc->log, WLOG_DEBUG,
"libusb_kernel_driver_active", err);
1066 err = libusb_detach_kernel_driver(pdev->libusb_handle, i);
1067 log_libusb_result(urbdrc->log, WLOG_DEBUG,
"libusb_detach_kernel_driver", err);
1071 pdev->status |= URBDRC_DEVICE_DETACH_KERNEL;
1078static BOOL libusb_udev_attach_kernel_driver(IUDEVICE* idev)
1083 if (!pdev || !pdev->LibusbConfig || !pdev->libusb_handle || !pdev->urbdrc)
1086 for (
int i = 0; i < pdev->LibusbConfig->bNumInterfaces && err != LIBUSB_ERROR_NO_DEVICE; i++)
1088 err = libusb_release_interface(pdev->libusb_handle, i);
1090 log_libusb_result(pdev->urbdrc->log, WLOG_DEBUG,
"libusb_release_interface", err);
1093 if (err != LIBUSB_ERROR_NO_DEVICE)
1095 err = libusb_attach_kernel_driver(pdev->libusb_handle, i);
1096 log_libusb_result(pdev->urbdrc->log, WLOG_DEBUG,
"libusb_attach_kernel_driver if=%d",
1105static int libusb_udev_is_composite_device(IUDEVICE* idev)
1108 return pdev->isCompositeDevice;
1111static int libusb_udev_is_exist(IUDEVICE* idev)
1114 return (pdev->status & URBDRC_DEVICE_NOT_FOUND) ? 0 : 1;
1117static int libusb_udev_is_channel_closed(IUDEVICE* idev)
1120 IUDEVMAN* udevman =
nullptr;
1121 if (!pdev || !pdev->urbdrc)
1124 udevman = pdev->urbdrc->udevman;
1127 if (udevman->status & URBDRC_DEVICE_CHANNEL_CLOSED)
1131 if (pdev->status & URBDRC_DEVICE_CHANNEL_CLOSED)
1137static int libusb_udev_is_already_send(IUDEVICE* idev)
1140 return (pdev->status & URBDRC_DEVICE_ALREADY_SEND) ? 1 : 0;
1145static void libusb_udev_mark_channel_closed(IUDEVICE* idev)
1148 if (pdev && ((pdev->status & URBDRC_DEVICE_CHANNEL_CLOSED) == 0))
1151 const uint8_t busNr = idev->get_bus_number(idev);
1152 const uint8_t devNr = idev->get_dev_number(idev);
1154 pdev->status |= URBDRC_DEVICE_CHANNEL_CLOSED;
1155 pdev->iface.cancel_all_transfer_request(&pdev->iface);
1156 if (!urbdrc->udevman->unregister_udevice(urbdrc->udevman, busNr, devNr))
1158 WLog_Print(pdev->urbdrc->log, WLOG_WARN,
"unregister_udevice failed for %d, %d", busNr,
1166static void libusb_udev_channel_closed(IUDEVICE* idev)
1169 if (pdev && ((pdev->status & URBDRC_DEVICE_CHANNEL_CLOSED) == 0))
1172 const uint8_t busNr = idev->get_bus_number(idev);
1173 const uint8_t devNr = idev->get_dev_number(idev);
1174 IWTSVirtualChannel* channel =
nullptr;
1176 if (pdev->channelManager)
1177 channel = IFCALLRESULT(
nullptr, pdev->channelManager->FindChannelById,
1178 pdev->channelManager, pdev->channelID);
1180 pdev->status |= URBDRC_DEVICE_CHANNEL_CLOSED;
1184 const UINT rc = channel->Write(channel, 0,
nullptr,
nullptr);
1185 if (rc != CHANNEL_RC_OK)
1186 WLog_Print(urbdrc->log, WLOG_WARN,
"channel->Write failed with %" PRIu32, rc);
1189 if (!urbdrc->udevman->unregister_udevice(urbdrc->udevman, busNr, devNr))
1190 WLog_Print(urbdrc->log, WLOG_WARN,
"unregister_udevice failed for %d, %d", busNr,
1195static void libusb_udev_set_already_send(IUDEVICE* idev)
1198 pdev->status |= URBDRC_DEVICE_ALREADY_SEND;
1201static char* libusb_udev_get_path(IUDEVICE* idev)
1207static int libusb_udev_query_device_port_status(IUDEVICE* idev, UINT32* UsbdStatus,
1208 UINT32* BufferSize, BYTE* Buffer)
1215 if (!pdev || !pdev->urbdrc)
1218 urbdrc = pdev->urbdrc;
1220 if (pdev->hub_handle !=
nullptr)
1222 ret = idev->control_transfer(
1224 (uint8_t)LIBUSB_ENDPOINT_IN | (uint8_t)LIBUSB_REQUEST_TYPE_CLASS |
1225 (uint8_t)LIBUSB_RECIPIENT_OTHER,
1226 LIBUSB_REQUEST_GET_STATUS, 0, pdev->port_number, UsbdStatus, BufferSize, Buffer, 1000);
1228 if (log_libusb_result(urbdrc->log, WLOG_DEBUG,
"libusb_control_transfer", ret))
1232 WLog_Print(urbdrc->log, WLOG_DEBUG,
1233 "PORT STATUS:0x%02" PRIx8
"%02" PRIx8
"%02" PRIx8
"%02" PRIx8
"", Buffer[3],
1234 Buffer[2], Buffer[1], Buffer[0]);
1243 UINT32 MessageId, UINT32 RequestId, UINT32 EndpointAddress,
1244 WINPR_ATTR_UNUSED UINT32 TransferFlags, UINT32 StartFrame,
1245 UINT32 ErrorCount, BOOL NoAck,
1246 WINPR_ATTR_UNUSED
const BYTE* packetDescriptorData,
1247 UINT32 NumberOfPackets, UINT32 BufferSize,
const BYTE* Buffer,
1248 t_isoch_transfer_cb cb, UINT32 Timeout)
1251 UINT32 iso_packet_size = 0;
1253 ASYNC_TRANSFER_USER_DATA* user_data =
nullptr;
1254 struct libusb_transfer* iso_transfer =
nullptr;
1256 size_t outSize = (12ULL * NumberOfPackets);
1257 uint32_t streamID = 0x40000000 | RequestId;
1259 if (!pdev || !pdev->urbdrc)
1262 urbdrc = pdev->urbdrc;
1263 user_data = async_transfer_user_data_new(idev, MessageId, 48, BufferSize, Buffer,
1264 outSize + 1024, NoAck, cb, callback);
1269 user_data->ErrorCount = ErrorCount;
1270 user_data->StartFrame = StartFrame;
1273 Stream_Seek(user_data->data, (12ULL * NumberOfPackets));
1275 if (NumberOfPackets > 0)
1277 iso_packet_size = BufferSize / NumberOfPackets;
1278 iso_transfer = libusb_alloc_transfer((
int)NumberOfPackets);
1281 if (iso_transfer ==
nullptr)
1283 WLog_Print(urbdrc->log, WLOG_ERROR,
1284 "Error: libusb_alloc_transfer [NumberOfPackets=%" PRIu32
", BufferSize=%" PRIu32
1286 NumberOfPackets, BufferSize);
1287 async_transfer_user_data_free(user_data);
1292 libusb_fill_iso_transfer(
1293 iso_transfer, pdev->libusb_handle, WINPR_ASSERTING_INT_CAST(uint8_t, EndpointAddress),
1294 Stream_Pointer(user_data->data), WINPR_ASSERTING_INT_CAST(
int, BufferSize),
1295 WINPR_ASSERTING_INT_CAST(
int, NumberOfPackets), func_iso_callback, user_data, Timeout);
1296 set_stream_id_for_buffer(iso_transfer, streamID);
1297 libusb_set_iso_packet_lengths(iso_transfer, iso_packet_size);
1299 if (!ArrayList_Append(pdev->request_queue, iso_transfer))
1301 WLog_Print(urbdrc->log, WLOG_WARN,
1302 "Failed to queue iso transfer, streamID %08" PRIx32
" already in use!",
1304 request_free(iso_transfer);
1307 rc = libusb_submit_transfer(iso_transfer);
1308 if (log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_submit_transfer", rc))
1313static BOOL libusb_udev_control_transfer(IUDEVICE* idev, WINPR_ATTR_UNUSED UINT32 RequestId,
1314 WINPR_ATTR_UNUSED UINT32 EndpointAddress,
1315 WINPR_ATTR_UNUSED UINT32 TransferFlags, BYTE bmRequestType,
1316 BYTE Request, UINT16 Value, UINT16 Index,
1317 UINT32* UrbdStatus, UINT32* BufferSize, BYTE* Buffer,
1323 WINPR_ASSERT(BufferSize);
1324 WINPR_ASSERT(*BufferSize <= UINT16_MAX);
1326 if (!pdev || !pdev->urbdrc)
1329 status = libusb_control_transfer(pdev->libusb_handle, bmRequestType, Request, Value, Index,
1330 Buffer, (UINT16)*BufferSize, Timeout);
1333 *BufferSize = (UINT32)status;
1335 log_libusb_result(pdev->urbdrc->log, WLOG_ERROR,
"libusb_control_transfer", status);
1337 if (!func_set_usbd_status(pdev->urbdrc, pdev, UrbdStatus, status))
1343static int libusb_udev_bulk_or_interrupt_transfer(IUDEVICE* idev,
1345 UINT32 MessageId, UINT32 RequestId,
1346 UINT32 EndpointAddress, UINT32 TransferFlags,
1347 BOOL NoAck, UINT32 BufferSize,
const BYTE* data,
1348 t_isoch_transfer_cb cb, UINT32 Timeout)
1351 UINT32 transfer_type = 0;
1353 const LIBUSB_ENDPOINT_DESCEIPTOR* ep_desc =
nullptr;
1354 struct libusb_transfer* transfer =
nullptr;
1356 ASYNC_TRANSFER_USER_DATA* user_data =
nullptr;
1357 uint32_t streamID = 0x80000000 | RequestId;
1359 if (!pdev || !pdev->LibusbConfig || !pdev->urbdrc)
1362 urbdrc = pdev->urbdrc;
1364 async_transfer_user_data_new(idev, MessageId, 36, BufferSize, data, 0, NoAck, cb, callback);
1370 transfer = libusb_alloc_transfer(0);
1373 async_transfer_user_data_free(user_data);
1376 transfer->user_data = user_data;
1378 ep_desc = func_get_ep_desc(pdev->LibusbConfig, pdev->MsConfig, EndpointAddress);
1382 WLog_Print(urbdrc->log, WLOG_ERROR,
"func_get_ep_desc: endpoint 0x%" PRIx32
" not found",
1384 request_free(transfer);
1388 transfer_type = (ep_desc->bmAttributes) & 0x3;
1389 WLog_Print(urbdrc->log, WLOG_DEBUG,
1390 "urb_bulk_or_interrupt_transfer: ep:0x%" PRIx32
" "
1391 "transfer_type %" PRIu32
" flag:%" PRIu32
" OutputBufferSize:0x%" PRIx32
"",
1392 EndpointAddress, transfer_type, TransferFlags, BufferSize);
1394 switch (transfer_type)
1398 libusb_fill_bulk_transfer(
1399 transfer, pdev->libusb_handle, WINPR_ASSERTING_INT_CAST(uint8_t, EndpointAddress),
1400 Stream_Pointer(user_data->data), WINPR_ASSERTING_INT_CAST(
int, BufferSize),
1401 func_bulk_transfer_cb, user_data, Timeout);
1404 case INTERRUPT_TRANSFER:
1406 libusb_fill_interrupt_transfer(
1407 transfer, pdev->libusb_handle, WINPR_ASSERTING_INT_CAST(uint8_t, EndpointAddress),
1408 Stream_Pointer(user_data->data), WINPR_ASSERTING_INT_CAST(
int, BufferSize),
1409 func_bulk_transfer_cb, user_data, Timeout);
1413 WLog_Print(urbdrc->log, WLOG_DEBUG,
1414 "urb_bulk_or_interrupt_transfer:"
1415 " other transfer type 0x%" PRIX32
"",
1417 request_free(transfer);
1421 set_stream_id_for_buffer(transfer, streamID);
1423 if (!ArrayList_Append(pdev->request_queue, transfer))
1425 WLog_Print(urbdrc->log, WLOG_WARN,
1426 "Failed to queue transfer, streamID %08" PRIx32
" already in use!", streamID);
1427 request_free(transfer);
1430 rc = libusb_submit_transfer(transfer);
1431 if (log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_submit_transfer", rc))
1436static int func_cancel_xact_request(
URBDRC_PLUGIN* urbdrc,
struct libusb_transfer* transfer)
1440 if (!urbdrc || !transfer)
1443 status = libusb_cancel_transfer(transfer);
1445 if (log_libusb_result(urbdrc->log, WLOG_WARN,
"libusb_cancel_transfer", status))
1447 if (status == LIBUSB_ERROR_NOT_FOUND)
1456static void libusb_udev_cancel_all_transfer_request(IUDEVICE* idev)
1461 if (!pdev || !pdev->request_queue || !pdev->urbdrc)
1464 ArrayList_Lock(pdev->request_queue);
1465 count = ArrayList_Count(pdev->request_queue);
1467 for (
size_t x = 0; x < count; x++)
1469 struct libusb_transfer* transfer = ArrayList_GetItem(pdev->request_queue, x);
1470 func_cancel_xact_request(pdev->urbdrc, transfer);
1473 ArrayList_Unlock(pdev->request_queue);
1476static int libusb_udev_cancel_transfer_request(IUDEVICE* idev, UINT32 RequestId)
1480 struct libusb_transfer* transfer =
nullptr;
1481 uint32_t cancelID1 = 0x40000000 | RequestId;
1482 uint32_t cancelID2 = 0x80000000 | RequestId;
1484 if (!idev || !pdev->urbdrc || !pdev->request_queue)
1487 ArrayList_Lock(pdev->request_queue);
1488 transfer = list_contains(pdev->request_queue, cancelID1);
1490 transfer = list_contains(pdev->request_queue, cancelID2);
1496 rc = func_cancel_xact_request(urbdrc, transfer);
1498 ArrayList_Unlock(pdev->request_queue);
1502BASIC_STATE_FUNC_DEFINED(channelManager, IWTSVirtualChannelManager*)
1503BASIC_STATE_FUNC_DEFINED(channelID, UINT32)
1504BASIC_STATE_FUNC_DEFINED(ReqCompletion, UINT32)
1505BASIC_STATE_FUNC_DEFINED(bus_number, BYTE)
1506BASIC_STATE_FUNC_DEFINED(dev_number, BYTE)
1507BASIC_STATE_FUNC_DEFINED(port_number, UINT8)
1510BASIC_POINT_FUNC_DEFINED(udev,
void*)
1511BASIC_POINT_FUNC_DEFINED(prev,
void*)
1512BASIC_POINT_FUNC_DEFINED(next,
void*)
1514static UINT32 udev_get_UsbDevice(IUDEVICE* idev)
1521 return pdev->UsbDevice;
1524static void udev_set_UsbDevice(IUDEVICE* idev, UINT32 val)
1531 pdev->UsbDevice = val;
1534static void udev_free(IUDEVICE* idev)
1540 if (!idev || !udev->urbdrc)
1543 urbdrc = udev->urbdrc;
1545 libusb_udev_cancel_all_transfer_request(&udev->iface);
1546 if (udev->libusb_handle)
1548 rc = libusb_reset_device(udev->libusb_handle);
1550 log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_reset_device", rc);
1559 if (!udev->iface.attach_kernel_driver(idev))
1560 WLog_Print(udev->urbdrc->log, WLOG_WARN,
"attach_kernel_driver failed for device");
1561 ArrayList_Free(udev->request_queue);
1563 msusb_msconfig_free(udev->MsConfig);
1564 libusb_unref_device(udev->libusb_dev);
1565 libusb_close(udev->libusb_handle);
1566 libusb_close(udev->hub_handle);
1567 free(udev->devDescriptor);
1571static void udev_load_interface(
UDEVICE* pdev)
1577 BASIC_STATE_FUNC_REGISTER(channelManager, pdev);
1578 BASIC_STATE_FUNC_REGISTER(channelID, pdev);
1579 BASIC_STATE_FUNC_REGISTER(UsbDevice, pdev);
1580 BASIC_STATE_FUNC_REGISTER(ReqCompletion, pdev);
1581 BASIC_STATE_FUNC_REGISTER(bus_number, pdev);
1582 BASIC_STATE_FUNC_REGISTER(dev_number, pdev);
1583 BASIC_STATE_FUNC_REGISTER(port_number, pdev);
1584 BASIC_STATE_FUNC_REGISTER(MsConfig, pdev);
1585 BASIC_STATE_FUNC_REGISTER(p_udev, pdev);
1586 BASIC_STATE_FUNC_REGISTER(p_prev, pdev);
1587 BASIC_STATE_FUNC_REGISTER(p_next, pdev);
1588 pdev->iface.isCompositeDevice = libusb_udev_is_composite_device;
1589 pdev->iface.isExist = libusb_udev_is_exist;
1590 pdev->iface.isAlreadySend = libusb_udev_is_already_send;
1591 pdev->iface.isChannelClosed = libusb_udev_is_channel_closed;
1592 pdev->iface.setAlreadySend = libusb_udev_set_already_send;
1593 pdev->iface.setChannelClosed = libusb_udev_channel_closed;
1594 pdev->iface.markChannelClosed = libusb_udev_mark_channel_closed;
1595 pdev->iface.getPath = libusb_udev_get_path;
1597 pdev->iface.isoch_transfer = libusb_udev_isoch_transfer;
1598 pdev->iface.control_transfer = libusb_udev_control_transfer;
1599 pdev->iface.bulk_or_interrupt_transfer = libusb_udev_bulk_or_interrupt_transfer;
1600 pdev->iface.select_interface = libusb_udev_select_interface;
1601 pdev->iface.select_configuration = libusb_udev_select_configuration;
1602 pdev->iface.complete_msconfig_setup = libusb_udev_complete_msconfig_setup;
1603 pdev->iface.control_pipe_request = libusb_udev_control_pipe_request;
1604 pdev->iface.control_query_device_text = libusb_udev_control_query_device_text;
1605 pdev->iface.os_feature_descriptor_request = libusb_udev_os_feature_descriptor_request;
1606 pdev->iface.cancel_all_transfer_request = libusb_udev_cancel_all_transfer_request;
1607 pdev->iface.cancel_transfer_request = libusb_udev_cancel_transfer_request;
1608 pdev->iface.query_device_descriptor = libusb_udev_query_device_descriptor;
1609 pdev->iface.detach_kernel_driver = libusb_udev_detach_kernel_driver;
1610 pdev->iface.attach_kernel_driver = libusb_udev_attach_kernel_driver;
1611 pdev->iface.query_device_port_status = libusb_udev_query_device_port_status;
1612 pdev->iface.free = udev_free;
1616 UINT16 bus_number, UINT16 dev_number)
1619 uint8_t port_numbers[16] = WINPR_C_ARRAY_INIT;
1620 LIBUSB_DEVICE** libusb_list =
nullptr;
1621 const ssize_t total_device = libusb_get_device_list(ctx, &libusb_list);
1623 WINPR_ASSERT(urbdrc);
1626 for (ssize_t i = 0; i < total_device; i++)
1628 LIBUSB_DEVICE* dev = libusb_list[i];
1630 if ((bus_number != libusb_get_bus_number(dev)) ||
1631 (dev_number != libusb_get_device_address(dev)))
1632 libusb_unref_device(dev);
1635 error = libusb_open(dev, &pdev->libusb_handle);
1637 if (log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_open", error))
1639 libusb_unref_device(dev);
1644 error = libusb_get_port_numbers(dev, port_numbers,
sizeof(port_numbers));
1648 log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_get_port_numbers", error);
1649 libusb_unref_device(dev);
1653 pdev->port_number = port_numbers[(error - 1)];
1655 WLog_Print(urbdrc->log, WLOG_DEBUG,
" Port: %" PRIu8, pdev->port_number);
1657 (void)_snprintf(pdev->path,
sizeof(pdev->path),
"%" PRIu16
"-%d", bus_number,
1660 WLog_Print(urbdrc->log, WLOG_DEBUG,
" DevPath: %s", pdev->path);
1663 libusb_free_device_list(libusb_list, 0);
1671 UINT16 bus_number, WINPR_ATTR_UNUSED UINT16 dev_number)
1674 LIBUSB_DEVICE** libusb_list =
nullptr;
1675 LIBUSB_DEVICE_HANDLE* handle =
nullptr;
1676 const ssize_t total_device = libusb_get_device_list(ctx, &libusb_list);
1678 WINPR_ASSERT(urbdrc);
1681 for (ssize_t i = 0; i < total_device; i++)
1683 LIBUSB_DEVICE* dev = libusb_list[i];
1685 if ((bus_number != libusb_get_bus_number(dev)) ||
1686 (1 != libusb_get_device_address(dev)))
1687 libusb_unref_device(dev);
1690 WLog_Print(urbdrc->log, WLOG_DEBUG,
" Open hub: %" PRIu16
"", bus_number);
1691 error = libusb_open(dev, &handle);
1693 if (!log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_open", error))
1694 pdev->hub_handle = handle;
1696 libusb_unref_device(dev);
1700 libusb_free_device_list(libusb_list, 0);
1708static void request_free(
void* value)
1710 ASYNC_TRANSFER_USER_DATA* user_data =
nullptr;
1711 struct libusb_transfer* transfer = (
struct libusb_transfer*)value;
1715 user_data = (ASYNC_TRANSFER_USER_DATA*)transfer->user_data;
1716 async_transfer_user_data_free(user_data);
1717 transfer->user_data =
nullptr;
1718 libusb_free_transfer(transfer);
1721static IUDEVICE* udev_init(
URBDRC_PLUGIN* urbdrc, libusb_context* context, LIBUSB_DEVICE* device,
1722 BYTE bus_number, BYTE dev_number)
1725 int status = LIBUSB_ERROR_OTHER;
1726 LIBUSB_DEVICE_DESCRIPTOR* devDescriptor =
nullptr;
1727 LIBUSB_CONFIG_DESCRIPTOR* config_temp =
nullptr;
1728 LIBUSB_INTERFACE_DESCRIPTOR interface_temp;
1730 WINPR_ASSERT(urbdrc);
1737 pdev->urbdrc = urbdrc;
1738 udev_load_interface(pdev);
1741 pdev->libusb_dev = device;
1743 pdev->libusb_dev = udev_get_libusb_dev(context, bus_number, dev_number);
1745 if (pdev->libusb_dev ==
nullptr)
1748 if (urbdrc->listener_callback)
1749 udev_set_channelManager(&pdev->iface, urbdrc->listener_callback->channel_mgr);
1752 status = udev_get_device_handle(urbdrc, context, pdev, bus_number, dev_number);
1753 if (status != LIBUSB_SUCCESS)
1755 struct libusb_device_descriptor desc;
1756 const uint8_t port = libusb_get_port_number(pdev->libusb_dev);
1757 libusb_get_device_descriptor(pdev->libusb_dev, &desc);
1759 log_libusb_result(urbdrc->log, WLOG_ERROR,
1760 "libusb_open [b=0x%02X,p=0x%02X,a=0x%02X,VID=0x%04X,PID=0x%04X]", status,
1761 bus_number, port, dev_number, desc.idVendor, desc.idProduct);
1766 status = udev_get_hub_handle(urbdrc, context, pdev, bus_number, dev_number);
1769 pdev->hub_handle =
nullptr;
1771 pdev->devDescriptor = udev_new_descript(urbdrc, pdev->libusb_dev);
1773 if (!pdev->devDescriptor)
1776 status = libusb_get_active_config_descriptor(pdev->libusb_dev, &pdev->LibusbConfig);
1778 if (status == LIBUSB_ERROR_NOT_FOUND)
1779 status = libusb_get_config_descriptor(pdev->libusb_dev, 0, &pdev->LibusbConfig);
1783 log_libusb_result(urbdrc->log, WLOG_ERROR,
"libusb_get_config_descriptor", status);
1787 config_temp = pdev->LibusbConfig;
1789 interface_temp = config_temp->interface[0].altsetting[0];
1790 WLog_Print(urbdrc->log, WLOG_DEBUG,
1791 "Registered Device: Vid: 0x%04" PRIX16
" Pid: 0x%04" PRIX16
""
1792 " InterfaceClass = %s",
1793 pdev->devDescriptor->idVendor, pdev->devDescriptor->idProduct,
1794 usb_interface_class_to_string(interface_temp.bInterfaceClass));
1796 devDescriptor = pdev->devDescriptor;
1798 if ((devDescriptor->bNumConfigurations == 1) && (config_temp->bNumInterfaces > 1) &&
1799 (devDescriptor->bDeviceClass == LIBUSB_CLASS_PER_INTERFACE))
1801 pdev->isCompositeDevice = 1;
1803 else if ((devDescriptor->bDeviceClass == 0xef) &&
1804 (devDescriptor->bDeviceSubClass == LIBUSB_CLASS_COMM) &&
1805 (devDescriptor->bDeviceProtocol == 0x01))
1807 pdev->isCompositeDevice = 1;
1810 pdev->isCompositeDevice = 0;
1813 devDescriptor->bDeviceClass = interface_temp.bInterfaceClass;
1814 devDescriptor->bDeviceSubClass = interface_temp.bInterfaceSubClass;
1815 devDescriptor->bDeviceProtocol = interface_temp.bInterfaceProtocol;
1817 pdev->bus_number = bus_number;
1818 pdev->dev_number = dev_number;
1819 pdev->request_queue = ArrayList_New(TRUE);
1821 if (!pdev->request_queue)
1824 ArrayList_Object(pdev->request_queue)->
fnObjectFree = request_free;
1827 pdev->MsConfig = msusb_msconfig_new();
1829 if (!pdev->MsConfig)
1833 return &pdev->iface;
1835 pdev->iface.free(&pdev->iface);
1839size_t udev_new_by_id(
URBDRC_PLUGIN* urbdrc, libusb_context* ctx, UINT16 idVendor, UINT16 idProduct,
1840 IUDEVICE*** devArray)
1842 WINPR_ASSERT(urbdrc);
1843 WINPR_ASSERT(devArray);
1846 LIBUSB_DEVICE** libusb_list =
nullptr;
1848 *devArray =
nullptr;
1850 WLog_Print(urbdrc->log, WLOG_INFO,
"VID: 0x%04" PRIX16
", PID: 0x%04" PRIX16
"", idVendor,
1852 const ssize_t total_device = libusb_get_device_list(ctx, &libusb_list);
1853 if (total_device < 0)
1855 WLog_Print(urbdrc->log, WLOG_ERROR,
"libusb_get_device_list -> [%" PRIdz
"]", total_device);
1858 if (total_device == 0)
1860 WLog_Print(urbdrc->log, WLOG_WARN,
"libusb_get_device_list -> [%" PRIdz
"]", total_device);
1869 for (ssize_t i = 0; i < total_device; i++)
1871 LIBUSB_DEVICE* dev = libusb_list[i];
1872 LIBUSB_DEVICE_DESCRIPTOR* descriptor = udev_new_descript(urbdrc, dev);
1874 if ((descriptor->idVendor == idVendor) && (descriptor->idProduct == idProduct))
1876 const uint8_t nr = libusb_get_bus_number(dev);
1877 const uint8_t addr = libusb_get_device_address(dev);
1878 array[num] = (
PUDEVICE)udev_init(urbdrc, ctx, dev, nr, addr);
1880 if (array[num] !=
nullptr)
1884 WLog_Print(urbdrc->log, WLOG_WARN,
1885 "udev_init(nr=%" PRIu8
", addr=%" PRIu8
") failed", nr, addr);
1889 libusb_unref_device(dev);
1895 libusb_free_device_list(libusb_list, 0);
1896 *devArray = (IUDEVICE**)array;
1900IUDEVICE* udev_new_by_addr(
URBDRC_PLUGIN* urbdrc, libusb_context* context, BYTE bus_number,
1903 WLog_Print(urbdrc->log, WLOG_DEBUG,
"bus:%d dev:%d", bus_number, dev_number);
1904 return udev_init(urbdrc, context,
nullptr, bus_number, dev_number);
OBJECT_FREE_FN fnObjectFree