VirtualBox

source: vbox/trunk/src/VBox/Main/src-server/solaris/USBProxyBackendSolaris.cpp@ 63542

Last change on this file since 63542 was 63542, checked in by vboxsync, 8 years ago

USBProxyBackendSolaris.cpp: warnings.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.9 KB
Line 
1/* $Id: USBProxyBackendSolaris.cpp 63542 2016-08-16 11:28:21Z vboxsync $ */
2/** @file
3 * VirtualBox USB Proxy Service, Solaris Specialization.
4 */
5
6/*
7 * Copyright (C) 2005-2016 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
18
19/*********************************************************************************************************************************
20* Header Files *
21*********************************************************************************************************************************/
22#include "USBProxyBackend.h"
23#include "Logging.h"
24
25#include <VBox/usb.h>
26#include <VBox/usblib.h>
27#include <VBox/err.h>
28#include <iprt/semaphore.h>
29#include <iprt/path.h>
30
31#include <sys/usb/usba.h>
32#include <syslog.h>
33
34
35/*********************************************************************************************************************************
36* Internal Functions *
37*********************************************************************************************************************************/
38static int solarisWalkDeviceNode(di_node_t Node, void *pvArg);
39static void solarisFreeUSBDevice(PUSBDEVICE pDevice);
40static USBDEVICESTATE solarisDetermineUSBDeviceState(PUSBDEVICE pDevice, di_node_t Node);
41
42
43/*********************************************************************************************************************************
44* Structures and Typedefs *
45*********************************************************************************************************************************/
46typedef struct USBDEVICELIST
47{
48 PUSBDEVICE pHead;
49 PUSBDEVICE pTail;
50} USBDEVICELIST;
51typedef USBDEVICELIST *PUSBDEVICELIST;
52
53
54/**
55 * Initialize data members.
56 */
57USBProxyBackendSolaris::USBProxyBackendSolaris()
58 : USBProxyBackend(), mNotifyEventSem(NIL_RTSEMEVENT), mUSBLibInitialized(false)
59{
60 LogFlowThisFunc(("\n"));
61}
62
63USBProxyBackendSolaris::~USBProxyBackendSolaris()
64{
65}
66
67/**
68 * Initializes the object (called right after construction).
69 *
70 * @returns VBox status code.
71 */
72int USBProxyBackendSolaris::init(USBProxyService *aUsbProxyService, const com::Utf8Str &strId, const com::Utf8Str &strAddress)
73{
74 USBProxyBackend::init(aUsbProxyService, strId, strAddress);
75
76 unconst(m_strBackend) = Utf8Str("host");
77
78 /*
79 * Create semaphore.
80 */
81 int rc = RTSemEventCreate(&mNotifyEventSem);
82 if (RT_FAILURE(rc))
83 return rc;
84
85 /*
86 * Initialize the USB library.
87 */
88 rc = USBLibInit();
89 if (RT_FAILURE(rc))
90 {
91 /* mNotifyEventSem will be destroyed in uninit */
92 return rc;
93 }
94
95 mUSBLibInitialized = true;
96
97 /*
98 * Start the poller thread.
99 */
100 start();
101 return VINF_SUCCESS;
102}
103
104
105/**
106 * Stop all service threads and free the device chain.
107 */
108void USBProxyBackendSolaris::uninit()
109{
110 LogFlowThisFunc(("destruct\n"));
111
112 /*
113 * Stop the service.
114 */
115 if (isActive())
116 stop();
117
118 /*
119 * Terminate the USB library
120 */
121 if (mUSBLibInitialized)
122 {
123 USBLibTerm();
124 mUSBLibInitialized = false;
125 }
126
127 if (mNotifyEventSem != NIL_RTSEMEVENT)
128 {
129 RTSemEventDestroy(mNotifyEventSem);
130 mNotifyEventSem = NIL_RTSEMEVENT;
131 }
132}
133
134
135void *USBProxyBackendSolaris::insertFilter(PCUSBFILTER aFilter)
136{
137 return USBLibAddFilter(aFilter);
138}
139
140
141void USBProxyBackendSolaris::removeFilter(void *pvID)
142{
143 USBLibRemoveFilter(pvID);
144}
145
146
147int USBProxyBackendSolaris::wait(RTMSINTERVAL aMillies)
148{
149 return RTSemEventWait(mNotifyEventSem, aMillies < 1000 ? 1000 : RT_MIN(aMillies, 5000));
150}
151
152
153int USBProxyBackendSolaris::interruptWait(void)
154{
155 return RTSemEventSignal(mNotifyEventSem);
156}
157
158
159PUSBDEVICE USBProxyBackendSolaris::getDevices(void)
160{
161 USBDEVICELIST DevList;
162 DevList.pHead = NULL;
163 DevList.pTail = NULL;
164 di_node_t RootNode = di_init("/", DINFOCPYALL);
165 if (RootNode != DI_NODE_NIL)
166 di_walk_node(RootNode, DI_WALK_CLDFIRST, &DevList, solarisWalkDeviceNode);
167
168 di_fini(RootNode);
169 return DevList.pHead;
170}
171
172
173static int solarisWalkDeviceNode(di_node_t Node, void *pvArg)
174{
175 PUSBDEVICELIST pList = (PUSBDEVICELIST)pvArg;
176 AssertPtrReturn(pList, DI_WALK_TERMINATE);
177
178 /*
179 * Check if it's a USB device in the first place.
180 */
181 bool fUSBDevice = false;
182 char *pszCompatNames = NULL;
183 int cCompatNames = di_compatible_names(Node, &pszCompatNames);
184 for (int i = 0; i < cCompatNames; i++, pszCompatNames += strlen(pszCompatNames) + 1)
185 if (!strncmp(pszCompatNames, RT_STR_TUPLE("usb")))
186 {
187 fUSBDevice = true;
188 break;
189 }
190
191 if (!fUSBDevice)
192 return DI_WALK_CONTINUE;
193
194 /*
195 * Check if it's a device node or interface.
196 */
197 int *pInt = NULL;
198 char *pStr = NULL;
199 int rc = DI_WALK_CONTINUE;
200 if (di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "interface", &pInt) < 0)
201 {
202 /* It's a device node. */
203 char *pszDevicePath = di_devfs_path(Node);
204 PUSBDEVICE pCur = (PUSBDEVICE)RTMemAllocZ(sizeof(*pCur));
205 if (!pCur)
206 {
207 LogRel(("USBService: failed to allocate %d bytes for PUSBDEVICE.\n", sizeof(*pCur)));
208 return DI_WALK_TERMINATE;
209 }
210
211 bool fValidDevice = false;
212 do
213 {
214 AssertBreak(pszDevicePath);
215
216 char *pszDriverName = di_driver_name(Node);
217
218 /*
219 * Skip hubs
220 */
221 if ( pszDriverName
222 && !strcmp(pszDriverName, "hubd"))
223 {
224 break;
225 }
226
227 /*
228 * Mandatory.
229 * snv_85 and above have usb-dev-descriptor node properties, but older one's do not.
230 * So if we cannot obtain the entire device descriptor, we try falling back to the
231 * individual properties (those must not fail, if it does we drop the device).
232 */
233 uchar_t *pDevData = NULL;
234 int cbProp = di_prop_lookup_bytes(DDI_DEV_T_ANY, Node, "usb-dev-descriptor", &pDevData);
235 if ( cbProp > 0
236 && pDevData)
237 {
238 usb_dev_descr_t *pDeviceDescriptor = (usb_dev_descr_t *)pDevData;
239 pCur->bDeviceClass = pDeviceDescriptor->bDeviceClass;
240 pCur->bDeviceSubClass = pDeviceDescriptor->bDeviceSubClass;
241 pCur->bDeviceProtocol = pDeviceDescriptor->bDeviceProtocol;
242 pCur->idVendor = pDeviceDescriptor->idVendor;
243 pCur->idProduct = pDeviceDescriptor->idProduct;
244 pCur->bcdDevice = pDeviceDescriptor->bcdDevice;
245 pCur->bcdUSB = pDeviceDescriptor->bcdUSB;
246 pCur->bNumConfigurations = pDeviceDescriptor->bNumConfigurations;
247 pCur->fPartialDescriptor = false;
248 }
249 else
250 {
251 AssertBreak(di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "usb-vendor-id", &pInt) > 0);
252 pCur->idVendor = (uint16_t)*pInt;
253
254 AssertBreak(di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "usb-product-id", &pInt) > 0);
255 pCur->idProduct = (uint16_t)*pInt;
256
257 AssertBreak(di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "usb-revision-id", &pInt) > 0);
258 pCur->bcdDevice = (uint16_t)*pInt;
259
260 AssertBreak(di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "usb-release", &pInt) > 0);
261 pCur->bcdUSB = (uint16_t)*pInt;
262
263 pCur->fPartialDescriptor = true;
264 }
265
266 char *pszPortAddr = di_bus_addr(Node);
267 if (pszPortAddr)
268 pCur->bPort = RTStrToUInt8(pszPortAddr); /* Bus & Port are mixed up (kernel driver/userland) */
269 else
270 pCur->bPort = 0;
271
272 char szBuf[PATH_MAX + 48];
273 RTStrPrintf(szBuf, sizeof(szBuf), "%#x:%#x:%d:%s", pCur->idVendor, pCur->idProduct, pCur->bcdDevice, pszDevicePath);
274 pCur->pszAddress = RTStrDup(szBuf);
275 AssertBreak(pCur->pszAddress);
276
277 pCur->pszDevicePath = RTStrDup(pszDevicePath);
278 AssertBreak(pCur->pszDevicePath);
279
280 pCur->pszBackend = RTStrDup("host");
281 AssertBreak(pCur->pszBackend);
282
283 /*
284 * Optional (some devices don't have all these)
285 */
286 char *pszCopy;
287 if (di_prop_lookup_strings(DDI_DEV_T_ANY, Node, "usb-product-name", &pStr) > 0)
288 {
289 pCur->pszProduct = pszCopy = RTStrDup(pStr);
290 USBLibPurgeEncoding(pszCopy);
291 }
292
293 if (di_prop_lookup_strings(DDI_DEV_T_ANY, Node, "usb-vendor-name", &pStr) > 0)
294 {
295 pCur->pszManufacturer = pszCopy = RTStrDup(pStr);
296 USBLibPurgeEncoding(pszCopy);
297 }
298
299 if (di_prop_lookup_strings(DDI_DEV_T_ANY, Node, "usb-serialno", &pStr) > 0)
300 {
301 pCur->pszSerialNumber = pszCopy = RTStrDup(pStr);
302 USBLibPurgeEncoding(pszCopy);
303 }
304
305 if (pCur->bcdUSB == 0x300)
306 pCur->enmSpeed = USBDEVICESPEED_SUPER;
307 else if (di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "low-speed", &pInt) >= 0)
308 pCur->enmSpeed = USBDEVICESPEED_LOW;
309 else if (di_prop_lookup_ints(DDI_DEV_T_ANY, Node, "high-speed", &pInt) >= 0)
310 pCur->enmSpeed = USBDEVICESPEED_HIGH;
311 else
312 pCur->enmSpeed = USBDEVICESPEED_FULL;
313
314 /* Determine state of the USB device. */
315 pCur->enmState = solarisDetermineUSBDeviceState(pCur, Node);
316
317 /*
318 * Valid device, add it to the list.
319 */
320 fValidDevice = true;
321 pCur->pPrev = pList->pTail;
322 if (pList->pTail)
323 pList->pTail = pList->pTail->pNext = pCur;
324 else
325 pList->pTail = pList->pHead = pCur;
326
327 rc = DI_WALK_CONTINUE;
328 } while (0);
329
330 di_devfs_path_free(pszDevicePath);
331 if (!fValidDevice)
332 solarisFreeUSBDevice(pCur);
333 }
334 return rc;
335}
336
337
338static USBDEVICESTATE solarisDetermineUSBDeviceState(PUSBDEVICE pDevice, di_node_t Node)
339{
340 char *pszDriverName = di_driver_name(Node);
341
342 /* Not possible unless a user explicitly unbinds the default driver. */
343 if (!pszDriverName)
344 return USBDEVICESTATE_UNUSED;
345
346 if (!strncmp(pszDriverName, RT_STR_TUPLE(VBOXUSB_DRIVER_NAME)))
347 return USBDEVICESTATE_HELD_BY_PROXY;
348
349 NOREF(pDevice);
350 return USBDEVICESTATE_USED_BY_HOST_CAPTURABLE;
351}
352
353
354int USBProxyBackendSolaris::captureDevice(HostUSBDevice *aDevice)
355{
356 /*
357 * Check preconditions.
358 */
359 AssertReturn(aDevice, VERR_GENERAL_FAILURE);
360 AssertReturn(!aDevice->isWriteLockOnCurrentThread(), VERR_GENERAL_FAILURE);
361
362 AutoReadLock devLock(aDevice COMMA_LOCKVAL_SRC_POS);
363 LogFlowThisFunc(("aDevice=%s\n", aDevice->i_getName().c_str()));
364
365 Assert(aDevice->i_getUnistate() == kHostUSBDeviceState_Capturing);
366 AssertReturn(aDevice->i_getUsbData(), VERR_INVALID_POINTER);
367
368 /*
369 * Create a one-shot capture filter for the device and reset the device.
370 */
371 USBFILTER Filter;
372 USBFilterInit(&Filter, USBFILTERTYPE_ONESHOT_CAPTURE);
373 initFilterFromDevice(&Filter, aDevice);
374
375 void *pvId = USBLibAddFilter(&Filter);
376 if (!pvId)
377 {
378 LogRel(("USBService: failed to add filter\n"));
379 return VERR_GENERAL_FAILURE;
380 }
381
382 PUSBDEVICE pDev = aDevice->i_getUsbData();
383 int rc = USBLibResetDevice(pDev->pszDevicePath, true);
384 if (RT_SUCCESS(rc))
385 aDevice->i_setBackendUserData(pvId);
386 else
387 {
388 USBLibRemoveFilter(pvId);
389 pvId = NULL;
390 }
391 LogFlowThisFunc(("returns %Rrc pvId=%p\n", rc, pvId));
392 return rc;
393}
394
395
396void USBProxyBackendSolaris::captureDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess)
397{
398 AssertReturnVoid(aDevice->isWriteLockOnCurrentThread());
399 /*
400 * Remove the one-shot filter if necessary.
401 */
402 LogFlowThisFunc(("aDevice=%s aSuccess=%RTbool mOneShotId=%p\n", aDevice->i_getName().c_str(), aSuccess, aDevice->i_getBackendUserData()));
403 if (!aSuccess && aDevice->i_getBackendUserData())
404 USBLibRemoveFilter(aDevice->i_getBackendUserData());
405 aDevice->i_setBackendUserData(NULL);
406 USBProxyBackend::captureDeviceCompleted(aDevice, aSuccess);
407}
408
409
410int USBProxyBackendSolaris::releaseDevice(HostUSBDevice *aDevice)
411{
412 /*
413 * Check preconditions.
414 */
415 AssertReturn(aDevice, VERR_GENERAL_FAILURE);
416 AssertReturn(!aDevice->isWriteLockOnCurrentThread(), VERR_GENERAL_FAILURE);
417
418 AutoReadLock devLock(aDevice COMMA_LOCKVAL_SRC_POS);
419 LogFlowThisFunc(("aDevice=%s\n", aDevice->i_getName().c_str()));
420
421 Assert(aDevice->i_getUnistate() == kHostUSBDeviceState_ReleasingToHost);
422 AssertReturn(aDevice->i_getUsbData(), VERR_INVALID_POINTER);
423
424 /*
425 * Create a one-shot ignore filter for the device and reset it.
426 */
427 USBFILTER Filter;
428 USBFilterInit(&Filter, USBFILTERTYPE_ONESHOT_IGNORE);
429 initFilterFromDevice(&Filter, aDevice);
430
431 void *pvId = USBLibAddFilter(&Filter);
432 if (!pvId)
433 {
434 LogRel(("USBService: Adding ignore filter failed!\n"));
435 return VERR_GENERAL_FAILURE;
436 }
437
438 PUSBDEVICE pDev = aDevice->i_getUsbData();
439 int rc = USBLibResetDevice(pDev->pszDevicePath, true /* Re-attach */);
440 if (RT_SUCCESS(rc))
441 aDevice->i_setBackendUserData(pvId);
442 else
443 {
444 USBLibRemoveFilter(pvId);
445 pvId = NULL;
446 }
447 LogFlowThisFunc(("returns %Rrc pvId=%p\n", rc, pvId));
448 return rc;
449}
450
451
452void USBProxyBackendSolaris::releaseDeviceCompleted(HostUSBDevice *aDevice, bool aSuccess)
453{
454 AssertReturnVoid(aDevice->isWriteLockOnCurrentThread());
455 /*
456 * Remove the one-shot filter if necessary.
457 */
458 LogFlowThisFunc(("aDevice=%s aSuccess=%RTbool mOneShotId=%p\n", aDevice->i_getName().c_str(), aSuccess, aDevice->i_getBackendUserData()));
459 if (!aSuccess && aDevice->i_getBackendUserData())
460 USBLibRemoveFilter(aDevice->i_getBackendUserData());
461 aDevice->i_setBackendUserData(NULL);
462 USBProxyBackend::releaseDeviceCompleted(aDevice, aSuccess);
463}
464
465
466/**
467 * Returns whether devices reported by this backend go through a de/re-attach
468 * and device re-enumeration cycle when they are captured or released.
469 */
470bool USBProxyBackendSolaris::i_isDevReEnumerationRequired()
471{
472 return true;
473}
474
475
476/**
477 * Wrapper called by walkDeviceNode.
478 *
479 * @param pDevice The USB device to free.
480 */
481void solarisFreeUSBDevice(PUSBDEVICE pDevice)
482{
483 USBProxyBackend::freeDevice(pDevice);
484}
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