VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/VBoxUSB/win/cmn/VBoxUsbTool.cpp@ 69500

Last change on this file since 69500 was 69500, checked in by vboxsync, 7 years ago

*: scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 13.8 KB
Line 
1/* $Id: VBoxUsbTool.cpp 69500 2017-10-28 15:14:05Z vboxsync $ */
2/** @file
3 * Windows USB R0 Tooling.
4 */
5
6/*
7 * Copyright (C) 2011-2017 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27#define INITGUID
28#include "VBoxUsbTool.h"
29#include <usbbusif.h>
30
31#include <iprt/assert.h>
32#include <iprt/string.h>
33#include <VBox/log.h>
34#include <VBox/usblib.h>
35
36#include "../../../win/VBoxDbgLog.h"
37
38#define VBOXUSBTOOL_MEMTAG 'TUBV'
39
40static PVOID vboxUsbToolMemAlloc(SIZE_T cbBytes)
41{
42 PVOID pvMem = ExAllocatePoolWithTag(NonPagedPool, cbBytes, VBOXUSBTOOL_MEMTAG);
43 Assert(pvMem);
44 return pvMem;
45}
46
47static PVOID vboxUsbToolMemAllocZ(SIZE_T cbBytes)
48{
49 PVOID pvMem = vboxUsbToolMemAlloc(cbBytes);
50 if (pvMem)
51 {
52 RtlZeroMemory(pvMem, cbBytes);
53 }
54 return pvMem;
55}
56
57static VOID vboxUsbToolMemFree(PVOID pvMem)
58{
59 ExFreePoolWithTag(pvMem, VBOXUSBTOOL_MEMTAG);
60}
61
62VBOXUSBTOOL_DECL(PURB) VBoxUsbToolUrbAlloc(USHORT u16Function, USHORT cbSize)
63{
64 PURB pUrb = (PURB)vboxUsbToolMemAlloc(cbSize);
65 Assert(pUrb);
66 if (!pUrb)
67 return NULL;
68
69 pUrb->UrbHeader.Length = cbSize;
70 pUrb->UrbHeader.Function = u16Function;
71 return pUrb;
72}
73
74VBOXUSBTOOL_DECL(PURB) VBoxUsbToolUrbAllocZ(USHORT u16Function, USHORT cbSize)
75{
76 PURB pUrb = (PURB)vboxUsbToolMemAllocZ(cbSize);
77 Assert(pUrb);
78 if (!pUrb)
79 return NULL;
80
81 pUrb->UrbHeader.Length = cbSize;
82 pUrb->UrbHeader.Function = u16Function;
83 return pUrb;
84}
85
86VBOXUSBTOOL_DECL(PURB) VBoxUsbToolUrbReinit(PURB pUrb, USHORT cbSize, USHORT u16Function)
87{
88 Assert(pUrb->UrbHeader.Length == cbSize);
89 if (pUrb->UrbHeader.Length < cbSize)
90 return NULL;
91 pUrb->UrbHeader.Length = cbSize;
92 pUrb->UrbHeader.Function = u16Function;
93 return pUrb;
94}
95
96VBOXUSBTOOL_DECL(VOID) VBoxUsbToolUrbFree(PURB pUrb)
97{
98 vboxUsbToolMemFree(pUrb);
99}
100
101VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolUrbPost(PDEVICE_OBJECT pDevObj, PURB pUrb, ULONG dwTimeoutMs)
102{
103 if (dwTimeoutMs == RT_INDEFINITE_WAIT)
104 return VBoxUsbToolIoInternalCtlSendSync(pDevObj, IOCTL_INTERNAL_USB_SUBMIT_URB, pUrb, NULL);
105 return VBoxUsbToolIoInternalCtlSendSyncWithTimeout(pDevObj, IOCTL_INTERNAL_USB_SUBMIT_URB, pUrb, NULL, dwTimeoutMs);
106}
107
108VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolGetDescriptor(PDEVICE_OBJECT pDevObj, void *pvBuffer, int cbBuffer, int Type, int iIndex, int LangId, ULONG dwTimeoutMs)
109{
110 NTSTATUS Status;
111 USHORT cbUrb = sizeof (struct _URB_CONTROL_DESCRIPTOR_REQUEST);
112 PURB pUrb = VBoxUsbToolUrbAllocZ(URB_FUNCTION_GET_DESCRIPTOR_FROM_DEVICE, cbUrb);
113 if (!pUrb)
114 {
115 WARN(("allocating URB failed"));
116 return STATUS_INSUFFICIENT_RESOURCES;
117 }
118
119 PUSB_COMMON_DESCRIPTOR pCmn = (PUSB_COMMON_DESCRIPTOR)pvBuffer;
120 pCmn->bLength = cbBuffer;
121 pCmn->bDescriptorType = Type;
122
123 pUrb->UrbHeader.Function = URB_FUNCTION_GET_DESCRIPTOR_FROM_DEVICE;
124 pUrb->UrbHeader.Length = sizeof(struct _URB_CONTROL_DESCRIPTOR_REQUEST);
125 pUrb->UrbControlDescriptorRequest.TransferBufferLength = cbBuffer;
126 pUrb->UrbControlDescriptorRequest.TransferBuffer = pvBuffer;
127 pUrb->UrbControlDescriptorRequest.Index = (UCHAR)iIndex;
128 pUrb->UrbControlDescriptorRequest.DescriptorType = (UCHAR)Type;
129 pUrb->UrbControlDescriptorRequest.LanguageId = (USHORT)LangId;
130
131 Status = VBoxUsbToolUrbPost(pDevObj, pUrb, dwTimeoutMs);
132 ASSERT_WARN(Status == STATUS_SUCCESS, ("VBoxUsbToolUrbPost failed Status (0x%x)", Status));
133
134 VBoxUsbToolUrbFree(pUrb);
135
136 return Status;
137}
138
139VBOXUSBTOOL_DECL(VOID) VBoxUsbToolStringDescriptorToUnicodeString(PUSB_STRING_DESCRIPTOR pDr, PUNICODE_STRING pUnicode)
140{
141 /* for some reason the string dr sometimes contains a non-null terminated string
142 * although we zeroed up the complete descriptor buffer
143 * this is why RtlInitUnicodeString won't work
144 * we need to init the scting length based on dr length */
145 pUnicode->Buffer = pDr->bString;
146 pUnicode->Length = pUnicode->MaximumLength = pDr->bLength - RT_OFFSETOF(USB_STRING_DESCRIPTOR, bString);
147}
148
149VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolGetStringDescriptor(PDEVICE_OBJECT pDevObj, char *pszResult, ULONG cbResult,
150 int iIndex, int LangId, ULONG dwTimeoutMs)
151{
152 char aBuf[MAXIMUM_USB_STRING_LENGTH];
153 AssertCompile(sizeof (aBuf) <= UINT8_MAX);
154 UCHAR cbBuf = (UCHAR)sizeof (aBuf);
155 PUSB_STRING_DESCRIPTOR pDr = (PUSB_STRING_DESCRIPTOR)&aBuf;
156
157 Assert(pszResult);
158 *pszResult = 0;
159
160 memset(pDr, 0, cbBuf);
161 pDr->bLength = cbBuf;
162 pDr->bDescriptorType = USB_STRING_DESCRIPTOR_TYPE;
163
164 NTSTATUS Status = VBoxUsbToolGetDescriptor(pDevObj, pDr, cbBuf, USB_STRING_DESCRIPTOR_TYPE, iIndex, LangId, dwTimeoutMs);
165 if (NT_SUCCESS(Status))
166 {
167 if (pDr->bLength >= sizeof (USB_STRING_DESCRIPTOR))
168 {
169 int rc = RTUtf16ToUtf8Ex(pDr->bString, (pDr->bLength - RT_OFFSETOF(USB_STRING_DESCRIPTOR, bString)) / sizeof(RTUTF16),
170 &pszResult, cbResult, NULL /*pcch*/);
171 if (RT_SUCCESS(rc))
172 {
173 USBLibPurgeEncoding(pszResult);
174 Status = STATUS_SUCCESS;
175 }
176 else
177 Status = STATUS_UNSUCCESSFUL;
178 }
179 else
180 Status = STATUS_INVALID_PARAMETER;
181 }
182 return Status;
183}
184
185VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolGetLangID(PDEVICE_OBJECT pDevObj, int *pLangId, ULONG dwTimeoutMs)
186{
187 char aBuf[MAXIMUM_USB_STRING_LENGTH];
188 AssertCompile(sizeof (aBuf) <= UINT8_MAX);
189 UCHAR cbBuf = (UCHAR)sizeof (aBuf);
190 PUSB_STRING_DESCRIPTOR pDr = (PUSB_STRING_DESCRIPTOR)&aBuf;
191
192 Assert(pLangId);
193 *pLangId = 0;
194
195 memset(pDr, 0, cbBuf);
196 pDr->bLength = cbBuf;
197 pDr->bDescriptorType = USB_STRING_DESCRIPTOR_TYPE;
198
199 NTSTATUS Status = VBoxUsbToolGetDescriptor(pDevObj, pDr, cbBuf, USB_STRING_DESCRIPTOR_TYPE, 0, 0, dwTimeoutMs);
200 if (NT_SUCCESS(Status))
201 {
202 /* Just grab the first lang ID if available. In 99% cases, it will be US English (0x0409).*/
203 if (pDr->bLength >= sizeof (USB_STRING_DESCRIPTOR))
204 {
205 AssertCompile(sizeof (pDr->bString[0]) == sizeof (uint16_t));
206 *pLangId = pDr->bString[0];
207 Status = STATUS_SUCCESS;
208 }
209 else
210 {
211 Status = STATUS_INVALID_PARAMETER;
212 }
213 }
214 return Status;
215}
216
217VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolGetDeviceSpeed(PDEVICE_OBJECT pDevObj, BOOLEAN *pbIsHigh)
218{
219 Assert(pbIsHigh);
220 *pbIsHigh = FALSE;
221
222 PIRP pIrp = IoAllocateIrp(pDevObj->StackSize, FALSE);
223 Assert(pIrp);
224 if (!pIrp)
225 {
226 return STATUS_INSUFFICIENT_RESOURCES;
227 }
228
229 USB_BUS_INTERFACE_USBDI_V1 BusIf;
230 PIO_STACK_LOCATION pSl = IoGetNextIrpStackLocation(pIrp);
231 pSl->MajorFunction = IRP_MJ_PNP;
232 pSl->MinorFunction = IRP_MN_QUERY_INTERFACE;
233 pSl->Parameters.QueryInterface.InterfaceType = &USB_BUS_INTERFACE_USBDI_GUID;
234 pSl->Parameters.QueryInterface.Size = sizeof (BusIf);
235 pSl->Parameters.QueryInterface.Version = USB_BUSIF_USBDI_VERSION_1;
236 pSl->Parameters.QueryInterface.Interface = (PINTERFACE)&BusIf;
237 pSl->Parameters.QueryInterface.InterfaceSpecificData = NULL;
238
239 pIrp->IoStatus.Status = STATUS_NOT_SUPPORTED;
240
241 NTSTATUS Status = VBoxDrvToolIoPostSync(pDevObj, pIrp);
242 Assert(NT_SUCCESS(Status) || Status == STATUS_NOT_SUPPORTED);
243 if (NT_SUCCESS(Status))
244 {
245 *pbIsHigh = BusIf.IsDeviceHighSpeed(BusIf.BusContext);
246 BusIf.InterfaceDereference(BusIf.BusContext);
247 }
248 IoFreeIrp(pIrp);
249
250 return Status;
251}
252
253VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolPipeClear(PDEVICE_OBJECT pDevObj, HANDLE hPipe, bool fReset)
254{
255 if (!hPipe)
256 {
257 Log(("Resetting the control pipe??\n"));
258 return STATUS_SUCCESS;
259 }
260 USHORT u16Function = fReset ? URB_FUNCTION_RESET_PIPE : URB_FUNCTION_ABORT_PIPE;
261 PURB pUrb = VBoxUsbToolUrbAlloc(u16Function, sizeof (struct _URB_PIPE_REQUEST));
262 if (!pUrb)
263 {
264 AssertMsgFailed((__FUNCTION__": VBoxUsbToolUrbAlloc failed!\n"));
265 return STATUS_INSUFFICIENT_RESOURCES;
266 }
267 pUrb->UrbPipeRequest.PipeHandle = hPipe;
268 pUrb->UrbPipeRequest.Reserved = 0;
269
270 NTSTATUS Status = VBoxUsbToolUrbPost(pDevObj, pUrb, RT_INDEFINITE_WAIT);
271 if (!NT_SUCCESS(Status) || !USBD_SUCCESS(pUrb->UrbHeader.Status))
272 {
273 AssertMsgFailed((__FUNCTION__": vboxUsbToolRequest failed with %x (%x)\n", Status, pUrb->UrbHeader.Status));
274 }
275
276 VBoxUsbToolUrbFree(pUrb);
277
278 return Status;
279}
280
281VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolCurrentFrame(PDEVICE_OBJECT pDevObj, PIRP pIrp, PULONG piFrame)
282{
283 struct _URB_GET_CURRENT_FRAME_NUMBER Urb;
284 Urb.Hdr.Function = URB_FUNCTION_GET_CURRENT_FRAME_NUMBER;
285 Urb.Hdr.Length = sizeof(Urb);
286 Urb.FrameNumber = (ULONG)-1;
287
288 Assert(piFrame);
289 *piFrame = (ULONG)-1;
290
291 PIO_STACK_LOCATION pSl = IoGetNextIrpStackLocation(pIrp);
292 pSl->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
293 pSl->Parameters.DeviceIoControl.IoControlCode = IOCTL_INTERNAL_USB_SUBMIT_URB;
294 pSl->Parameters.Others.Argument1 = (PVOID)&Urb;
295 pSl->Parameters.Others.Argument2 = NULL;
296
297 NTSTATUS Status = VBoxUsbToolUrbPost(pDevObj, (PURB)&Urb, RT_INDEFINITE_WAIT);
298 Assert(NT_SUCCESS(Status));
299 if (NT_SUCCESS(Status))
300 {
301 *piFrame = Urb.FrameNumber;
302 }
303
304 return Status;
305}
306
307VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolDevUnconfigure(PDEVICE_OBJECT pDevObj)
308{
309 USHORT cbUrb = sizeof (struct _URB_SELECT_CONFIGURATION);
310 PURB pUrb = VBoxUsbToolUrbAlloc(URB_FUNCTION_SELECT_CONFIGURATION, cbUrb);
311 Assert(pUrb);
312 if (!pUrb)
313 return STATUS_INSUFFICIENT_RESOURCES;
314
315 UsbBuildSelectConfigurationRequest(pUrb, (USHORT)cbUrb, NULL);
316
317 NTSTATUS Status = VBoxUsbToolUrbPost(pDevObj, pUrb, RT_INDEFINITE_WAIT);
318 Assert(NT_SUCCESS(Status));
319
320 VBoxUsbToolUrbFree(pUrb);
321
322 return Status;
323}
324
325VBOXUSBTOOL_DECL(PIRP) VBoxUsbToolIoBuildAsyncInternalCtl(PDEVICE_OBJECT pDevObj, ULONG uCtl, void *pvArg1, void *pvArg2)
326{
327 PIRP pIrp = IoAllocateIrp(pDevObj->StackSize, FALSE);
328 Assert(pIrp);
329 if (!pIrp)
330 {
331 return NULL;
332 }
333
334 pIrp->IoStatus.Status = STATUS_SUCCESS;
335 pIrp->IoStatus.Information = NULL;
336
337 PIO_STACK_LOCATION pSl = IoGetNextIrpStackLocation(pIrp);
338 pSl->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
339 pSl->MinorFunction = 0;
340 pSl->Parameters.DeviceIoControl.IoControlCode = uCtl;
341 pSl->Parameters.Others.Argument1 = pvArg1;
342 pSl->Parameters.Others.Argument2 = pvArg2;
343 return pIrp;
344}
345
346VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolIoInternalCtlSendSyncWithTimeout(PDEVICE_OBJECT pDevObj, ULONG uCtl,
347 void *pvArg1, void *pvArg2, ULONG dwTimeoutMs)
348{
349 /* since we're going to cancel the irp on timeout, we should allocate our own IRP rather than using the threaded one
350 * */
351 PIRP pIrp = VBoxUsbToolIoBuildAsyncInternalCtl(pDevObj, uCtl, pvArg1, pvArg2);
352 if (!pIrp)
353 {
354 WARN(("VBoxUsbToolIoBuildAsyncInternalCtl failed"));
355 return STATUS_INSUFFICIENT_RESOURCES;
356 }
357
358 NTSTATUS Status = VBoxDrvToolIoPostSyncWithTimeout(pDevObj, pIrp, dwTimeoutMs);
359
360 IoFreeIrp(pIrp);
361
362 return Status;
363}
364
365VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolIoInternalCtlSendAsync(PDEVICE_OBJECT pDevObj, ULONG uCtl, void *pvArg1, void *pvArg2,
366 PKEVENT pEvent, PIO_STATUS_BLOCK pIoStatus)
367{
368 NTSTATUS Status;
369 PIRP pIrp;
370 PIO_STACK_LOCATION pSl;
371 Assert(KeGetCurrentIrql() == PASSIVE_LEVEL);
372
373 pIrp = IoBuildDeviceIoControlRequest(uCtl, pDevObj, NULL, 0, NULL, 0, TRUE, pEvent, pIoStatus);
374 if (!pIrp)
375 {
376 WARN(("IoBuildDeviceIoControlRequest failed!!\n"));
377 pIoStatus->Status = STATUS_INSUFFICIENT_RESOURCES;
378 pIoStatus->Information = 0;
379 return STATUS_INSUFFICIENT_RESOURCES;
380 }
381
382 /* Get the next stack location as that is used for the new irp */
383 pSl = IoGetNextIrpStackLocation(pIrp);
384 pSl->Parameters.Others.Argument1 = pvArg1;
385 pSl->Parameters.Others.Argument2 = pvArg2;
386
387 Status = IoCallDriver(pDevObj, pIrp);
388
389 return Status;
390}
391
392VBOXUSBTOOL_DECL(NTSTATUS) VBoxUsbToolIoInternalCtlSendSync(PDEVICE_OBJECT pDevObj, ULONG uCtl, void *pvArg1, void *pvArg2)
393{
394 IO_STATUS_BLOCK IoStatus = {0};
395 KEVENT Event;
396 NTSTATUS Status;
397
398 KeInitializeEvent(&Event, NotificationEvent, FALSE);
399
400 LOG(("Sending sync Ctl pDevObj(0x%p), uCtl(0x%x), pvArg1(0x%p), pvArg2(0x%p)", pDevObj, uCtl, pvArg1, pvArg2));
401
402 Status = VBoxUsbToolIoInternalCtlSendAsync(pDevObj, uCtl, pvArg1, pvArg2, &Event, &IoStatus);
403
404 if (Status == STATUS_PENDING)
405 {
406 LOG(("VBoxUsbToolIoInternalCtlSendAsync returned pending for pDevObj(0x%x)", pDevObj));
407 KeWaitForSingleObject(&Event, Executive, KernelMode, FALSE, NULL);
408 Status = IoStatus.Status;
409 LOG(("Pending VBoxUsbToolIoInternalCtlSendAsync completed with Status (0x%x) for pDevObj(0x%x)", Status, pDevObj));
410 }
411 else
412 {
413 LOG(("VBoxUsbToolIoInternalCtlSendAsync completed with Status (0x%x) for pDevObj(0x%x)", Status, pDevObj));
414 }
415
416 return Status;
417}
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette