VirtualBox

source: vbox/trunk/src/VBox/Devices/USB/VUSBReadAhead.cpp@ 27901

Last change on this file since 27901 was 27901, checked in by vboxsync, 15 years ago

OSE header fixes

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.5 KB
Line 
1/* $Id: VUSBReadAhead.cpp 27901 2010-03-31 14:07:26Z vboxsync $ */
2/** @file
3 * Virtual USB - Read-ahead buffering for periodic endpoints.
4 */
5
6/*
7 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_DRV_VUSB
27#include <VBox/pdm.h>
28#include <VBox/vmapi.h>
29#include <VBox/err.h>
30#include <VBox/log.h>
31#include <iprt/alloc.h>
32#include <iprt/time.h>
33#include <iprt/thread.h>
34#include <iprt/semaphore.h>
35#include <iprt/string.h>
36#include <iprt/assert.h>
37#include <iprt/asm.h>
38#include "VUSBInternal.h"
39
40
41/*******************************************************************************
42* Structures and Typedefs *
43*******************************************************************************/
44
45/**
46 * Argument package of vusbDevReadAheadThread().
47 */
48typedef struct vusb_read_ahead_args
49{
50 /** Pointer to the device which the thread is for. */
51 PVUSBDEV pDev;
52 /** Pointer to the pipe which the thread is servicing. */
53 PVUSBPIPE pPipe;
54 /** A flag indicating a high-speed (vs. low/full-speed) endpoint. */
55 bool fHighSpeed;
56 /** A flag telling the thread to terminate. */
57 bool fTerminate;
58} VUSBREADAHEADARGS, *PVUSBREADAHEADARGS;
59
60
61/*******************************************************************************
62* Implementation *
63*******************************************************************************/
64
65static PVUSBURB vusbDevNewIsocUrb(PVUSBDEV pDev, unsigned uEndPt, unsigned uInterval, unsigned uPktSize)
66{
67 PVUSBURB pUrb;
68 unsigned cPackets = 0;
69 uint32_t cbTotal = 0;
70 unsigned uNextIndex = 0;
71
72 Assert(pDev);
73 Assert(uEndPt);
74 Assert(uInterval);
75 Assert(uPktSize);
76
77 /* Calculate the amount of data needed, taking the endpoint's bInterval into account */
78 for (unsigned i = 0; i < 8; ++i)
79 {
80 if (i == uNextIndex)
81 {
82 cbTotal += uPktSize;
83 cPackets++;
84 uNextIndex += uInterval;
85 }
86 }
87 Assert(cbTotal <= 24576);
88
89 // @todo: What do we do if cPackets is 0?
90
91 /*
92 * Allocate and initialize the URB.
93 */
94 Assert(pDev->u8Address != VUSB_INVALID_ADDRESS);
95
96 PVUSBROOTHUB pRh = vusbDevGetRh(pDev);
97 if (!pRh)
98 /* can happen during disconnect */
99 return NULL;
100
101 pUrb = vusbRhNewUrb(pRh, pDev->u8Address, cbTotal, 1);
102 if (!pUrb)
103 /* not much we can do here... */
104 return NULL;
105
106 pUrb->enmType = VUSBXFERTYPE_ISOC;
107 pUrb->EndPt = uEndPt;
108 pUrb->enmDir = VUSBDIRECTION_IN;
109 pUrb->fShortNotOk = false;
110 pUrb->enmStatus = VUSBSTATUS_OK;
111 pUrb->Hci.EdAddr = 0;
112 pUrb->Hci.fUnlinked = false;
113// @todo: fill in the rest? The Hci member is not relevant
114#ifdef LOG_ENABLED
115 static unsigned s_iSerial = 0;
116 s_iSerial = (s_iSerial + 1) % 10000;
117 RTStrAPrintf(&pUrb->pszDesc, "URB %p prab<%04d", pUrb, s_iSerial); // prab = Periodic Read-Ahead Buffer
118#endif
119
120 /* Set up the individual packets, again with bInterval in mind */
121 pUrb->cIsocPkts = 8;
122 unsigned off = 0;
123 uNextIndex = 0;
124 for (unsigned i = 0; i < 8; i++)
125 {
126 pUrb->aIsocPkts[i].enmStatus = VUSBSTATUS_NOT_ACCESSED;
127 pUrb->aIsocPkts[i].off = off;
128 if (i == uNextIndex) // skip unused packets
129 {
130 pUrb->aIsocPkts[i].cb = uPktSize;
131 off += uPktSize;
132 uNextIndex += uInterval;
133 }
134 else
135 pUrb->aIsocPkts[i].cb = 0;
136 }
137 Assert(off == cbTotal);
138 return pUrb;
139}
140
141/**
142 * Thread function for performing read-ahead buffering of periodic input.
143 *
144 * This thread keeps a buffer (queue) filled with data read from a periodic
145 * input endpoint.
146 *
147 * The problem: In the EHCI emulation, there is a very short period between the
148 * time when the guest can schedule a request and the time when it expects the results.
149 * This leads to many dropped URBs because by the time we get the data from the host,
150 * the guest already gave up and moved on.
151 *
152 * The solution: For periodic transfers, we know the worst-case bandwidth. We can
153 * read ahead and buffer a few milliseconds worth of data. That way data is available
154 * by the time the guest asks for it and we can deliver it immediately.
155 *
156 * @returns success indicator.
157 * @param Thread This thread.
158 * @param pvUser Pointer to a VUSBREADAHEADARGS structure.
159 */
160static DECLCALLBACK(int) vusbDevReadAheadThread(RTTHREAD Thread, void *pvUser)
161{
162 PVUSBPIPE pPipe;
163 PVUSBREADAHEADARGS pArgs = (PVUSBREADAHEADARGS)pvUser;
164 PCVUSBDESCENDPOINT pDesc;
165 PVUSBURB pUrb;
166 int rc = VINF_SUCCESS;
167 unsigned max_pkt_size, mult, interval;
168
169 LogFlow(("vusb: periodic read-ahead buffer thread started\n"));
170 Assert(pArgs);
171 Assert(pArgs->pPipe && pArgs->pDev);
172
173 pPipe = pArgs->pPipe;
174 pDesc = &pPipe->in->Core;
175 Assert(pDesc);
176
177 /* The previous read-ahead thread could be still running (vusbReadAheadStop sets only
178 * fTerminate to true and returns immediately). Therefore we have to wait until the
179 * previous thread is done and all submitted URBs are completed. */
180 while (pPipe->cSubmitted > 0)
181 {
182 Log2(("vusbDevReadAheadThread: still %u packets submitted, waiting before starting...\n", pPipe->cSubmitted));
183 RTThreadSleep(1);
184 }
185 pPipe->pvReadAheadArgs = pArgs;
186 pPipe->cBuffered = 0;
187
188 /* Figure out the maximum bandwidth we might need */
189 if (pArgs->fHighSpeed)
190 {
191 /* High-speed endpoint */
192 Assert((pDesc->wMaxPacketSize & 0x1fff) == pDesc->wMaxPacketSize);
193 Assert(pDesc->bInterval <= 16);
194 interval = pDesc->bInterval ? 1 << (pDesc->bInterval - 1) : 1;
195 max_pkt_size = pDesc->wMaxPacketSize & 0x7ff;
196 mult = ((pDesc->wMaxPacketSize & 0x1800) >> 11) + 1;
197 }
198 else
199 {
200 /* Full- or low-speed endpoint */
201 Assert((pDesc->wMaxPacketSize & 0x7ff) == pDesc->wMaxPacketSize);
202 interval = pDesc->bInterval;
203 max_pkt_size = pDesc->wMaxPacketSize;
204 mult = 1;
205 }
206 Log(("vusb: interval=%u, max pkt size=%u, multiplier=%u\n", interval, max_pkt_size, mult));
207
208 /*
209 * Submit new URBs in a loop unless the buffer is too full (paused VM etc.). Note that we only
210 * queue the URBs here, they are reaped on a different thread.
211 */
212 while (pArgs->fTerminate == false)
213 {
214 while (pPipe->cSubmitted < 120 && pPipe->cBuffered < 120)
215 {
216 pUrb = vusbDevNewIsocUrb(pArgs->pDev, pDesc->bEndpointAddress & 0xF, interval, max_pkt_size * mult);
217 if (!pUrb) {
218 /* Happens if device was unplugged. */
219 Log(("vusb: read-ahead thread failed to allocate new URB; exiting\n"));
220 vusbReadAheadStop(pvUser);
221 break;
222 }
223
224 Assert(pUrb->enmState == VUSBURBSTATE_ALLOCATED);
225
226 // @todo: at the moment we abuse the Hci.pNext member (which is otherwise entirely unused!)
227 pUrb->Hci.pNext = (PVUSBURB)pvUser;
228
229 pUrb->enmState = VUSBURBSTATE_IN_FLIGHT;
230 rc = vusbUrbQueueAsyncRh(pUrb);
231 if (RT_FAILURE(rc))
232 {
233 /* Happens if device was unplugged. */
234 Log(("vusb: read-ahead thread failed to queue URB with %Rrc; exiting\n", rc));
235 vusbReadAheadStop(pvUser);
236 break;
237 }
238 ++pPipe->cSubmitted;
239 }
240 RTThreadSleep(1);
241 }
242 LogFlow(("vusb: periodic read-ahead buffer thread exiting\n"));
243 pPipe->pvReadAheadArgs = NULL;
244
245 /* wait until there are no more submitted packets */
246 while (pPipe->cSubmitted > 0)
247 {
248 Log2(("vusbDevReadAheadThread: still %u packets submitted, waiting before terminating...\n", pPipe->cSubmitted));
249 RTThreadSleep(1);
250 }
251
252 RTMemTmpFree(pArgs);
253
254 return rc;
255}
256
257/**
258 * Completes a read-ahead URB. This function does *not* free the URB but puts
259 * it on a queue instead. The URB is only freed when the guest asks for the data
260 * (by reading on the buffered pipe) or when the pipe/device is torn down.
261 */
262void vusbUrbCompletionReadAhead(PVUSBURB pUrb)
263{
264 Assert(pUrb);
265 Assert(pUrb->Hci.pNext);
266 PVUSBREADAHEADARGS pArgs = (PVUSBREADAHEADARGS)pUrb->Hci.pNext;
267 PVUSBPIPE pPipe = pArgs->pPipe;
268 Assert(pPipe);
269
270 pUrb->Hci.pNext = NULL; // @todo: use a more suitable field
271 if (pPipe->pBuffUrbHead == NULL)
272 {
273 // The queue is empty, this is easy
274 Assert(!pPipe->pBuffUrbTail);
275 pPipe->pBuffUrbTail = pPipe->pBuffUrbHead = pUrb;
276 }
277 else
278 {
279 // Some URBs are queued already
280 Assert(pPipe->pBuffUrbTail);
281 Assert(!pPipe->pBuffUrbTail->Hci.pNext);
282 pPipe->pBuffUrbTail = pPipe->pBuffUrbTail->Hci.pNext = pUrb;
283 }
284 --pPipe->cSubmitted;
285 ++pPipe->cBuffered;
286}
287
288/**
289 * Process a submit of an input URB on a pipe with read-ahead buffering. Instead
290 * of passing the URB to the proxy, we use previously read data stored in the
291 * read-ahead buffer, immediately complete the input URB and free the buffered URB.
292 *
293 * @param pUrb The URB submitted by HC
294 * @param pPipe The pipe with read-ahead buffering
295 *
296 * @return int Status code
297 */
298int vusbUrbSubmitBufferedRead(PVUSBURB pUrb, PVUSBPIPE pPipe)
299{
300 PVUSBURB pBufferedUrb;
301 Assert(pUrb && pPipe);
302
303 pBufferedUrb = pPipe->pBuffUrbHead;
304 if (pBufferedUrb)
305 {
306 unsigned cbTotal;
307
308 // There's a URB available in the read-ahead buffer; use it
309 pPipe->pBuffUrbHead = pBufferedUrb->Hci.pNext;
310 if (pPipe->pBuffUrbHead == NULL)
311 pPipe->pBuffUrbTail = NULL;
312
313 --pPipe->cBuffered;
314
315 // Make sure the buffered URB is what we expect
316 Assert(pUrb->enmType == pBufferedUrb->enmType);
317 Assert(pUrb->EndPt == pBufferedUrb->EndPt);
318 Assert(pUrb->enmDir == pBufferedUrb->enmDir);
319
320 pUrb->enmState = VUSBURBSTATE_REAPED;
321 pUrb->enmStatus = pBufferedUrb->enmStatus;
322 cbTotal = 0;
323 // Copy status and data received from the device
324 for (unsigned i = 0; i < pUrb->cIsocPkts; ++i)
325 {
326 unsigned off, len;
327
328 off = pBufferedUrb->aIsocPkts[i].off;
329 len = pBufferedUrb->aIsocPkts[i].cb;
330 pUrb->aIsocPkts[i].cb = len;
331 pUrb->aIsocPkts[i].off = off;
332 pUrb->aIsocPkts[i].enmStatus = pBufferedUrb->aIsocPkts[i].enmStatus;
333 cbTotal += len;
334 Assert(pUrb->VUsb.cbDataAllocated >= cbTotal);
335 memcpy(&pUrb->abData[off], &pBufferedUrb->abData[off], len);
336 }
337 // Give back the data to the HC right away and then free the buffered URB
338 vusbUrbCompletionRh(pUrb);
339 // This assertion is wrong as the URB could be re-allocated in the meantime by the EMT (race)
340 // Assert(pUrb->enmState == VUSBURBSTATE_FREE);
341 Assert(pBufferedUrb->enmState == VUSBURBSTATE_REAPED);
342 LogFlow(("%s: vusbUrbSubmitBufferedRead: Freeing buffered URB\n", pBufferedUrb->pszDesc));
343 pBufferedUrb->VUsb.pfnFree(pBufferedUrb);
344 // This assertion is wrong as the URB could be re-allocated in the meantime by the EMT (race)
345 // Assert(pBufferedUrb->enmState == VUSBURBSTATE_FREE);
346 }
347 else
348 {
349 // No URB on hand. Either we exhausted the buffer (shouldn't happen!) or the guest simply
350 // asked for data too soon. Pretend that the device didn't deliver any data.
351 pUrb->enmState = VUSBURBSTATE_REAPED;
352 pUrb->enmStatus = VUSBSTATUS_DATA_UNDERRUN;
353 for (unsigned i = 0; i < pUrb->cIsocPkts; ++i)
354 {
355 pUrb->aIsocPkts[i].cb = 0;
356 pUrb->aIsocPkts[i].enmStatus = VUSBSTATUS_NOT_ACCESSED;
357 }
358 vusbUrbCompletionRh(pUrb);
359 // This assertion is wrong as the URB could be re-allocated in the meantime by the EMT (race)
360 // Assert(pUrb->enmState == VUSBURBSTATE_FREE);
361 LogFlow(("%s: vusbUrbSubmitBufferedRead: No buffered URB available!\n", pBufferedUrb->pszDesc));
362 }
363 return VINF_SUCCESS;
364}
365
366/* Read-ahead start/stop functions, used primarily to keep the PVUSBREADAHEADARGS struct private to this module. */
367
368void vusbReadAheadStart(PVUSBDEV pDev, PVUSBPIPE pPipe)
369{
370 int rc;
371 PVUSBREADAHEADARGS pArgs = (PVUSBREADAHEADARGS)RTMemTmpAlloc(sizeof(*pArgs));
372
373 if (pArgs)
374 {
375 pArgs->pDev = pDev;
376 pArgs->pPipe = pPipe;
377 pArgs->fTerminate = false;
378 pArgs->fHighSpeed = ((vusbDevGetRh(pDev)->fHcVersions & VUSB_STDVER_20) != 0);
379 if (pArgs->fHighSpeed)
380 rc = RTThreadCreate(&pPipe->ReadAheadThread, vusbDevReadAheadThread, pArgs, 0, RTTHREADTYPE_IO, RTTHREADFLAGS_WAITABLE, "USBISOC");
381 else
382 rc = VERR_VUSB_DEVICE_NOT_ATTACHED; // No buffering for low/full-speed devices at the moment, needs testing.
383 if (RT_SUCCESS(rc))
384 {
385 Log(("vusb: created isochronous read-ahead thread\n"));
386 }
387 else
388 {
389 Log(("vusb: isochronous read-ahead thread creation failed, rc=%d\n", rc));
390 pPipe->ReadAheadThread = NIL_RTTHREAD;
391 RTMemTmpFree(pArgs);
392 }
393 }
394 /* If thread creation failed for any reason, simply fall back to standard processing. */
395}
396
397void vusbReadAheadStop(void *pvReadAheadArgs)
398{
399 PVUSBREADAHEADARGS pArgs = (PVUSBREADAHEADARGS)pvReadAheadArgs;
400 Log(("vusb: terminating read-ahead thread for endpoint\n"));
401 pArgs->fTerminate = true;
402}
403
404/*
405 * Local Variables:
406 * mode: c
407 * c-file-style: "bsd"
408 * c-basic-offset: 4
409 * tab-width: 4
410 * indent-tabs-mode: s
411 * End:
412 */
413
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