VirtualBox

source: vbox/trunk/src/VBox/Devices/USB/VUSBUrbPool.cpp@ 60961

Last change on this file since 60961 was 59775, checked in by vboxsync, 9 years ago

VUSB: Remove unused read ahead buffering code and replace with new implementation which in the end will work in both directions (only out implemented so far)

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 9.0 KB
Line 
1/* $Id: VUSBUrbPool.cpp 59775 2016-02-22 13:58:44Z vboxsync $ */
2/** @file
3 * Virtual USB - URB pool.
4 */
5
6/*
7 * Copyright (C) 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#define LOG_GROUP LOG_GROUP_DRV_VUSB
23#include <VBox/log.h>
24#include <VBox/err.h>
25#include <iprt/mem.h>
26#include <iprt/critsect.h>
27
28#include "VUSBInternal.h"
29
30/*********************************************************************************************************************************
31* Defined Constants And Macros *
32*********************************************************************************************************************************/
33
34/** Maximum age for one URB. */
35#define VUSBURB_AGE_MAX 10
36
37/** Convert from an URB to the URB header. */
38#define VUSBURBPOOL_URB_2_URBHDR(a_pUrb) RT_FROM_MEMBER(a_pUrb, VUSBURBHDR, Urb);
39
40/*********************************************************************************************************************************
41* Structures and Typedefs *
42*********************************************************************************************************************************/
43
44/**
45 * URB header not visible to the caller allocating an URB
46 * and only for internal tracking.
47 */
48typedef struct VUSBURBHDR
49{
50 /** List node for keeping the URB in the free list. */
51 RTLISTNODE NdFree;
52 /** Size of the data allocated for the URB (Only the variable part including the
53 * HCI and TDs). */
54 size_t cbAllocated;
55 /** Age of the URB waiting on the list, if it is waiting for too long without being used
56 * again it will be freed. */
57 uint32_t cAge;
58#if HC_ARCH_BITS == 64
59 uint32_t u32Alignment0;
60#endif
61 /** The embedded URB. */
62 VUSBURB Urb;
63} VUSBURBHDR;
64/** Pointer to a URB header. */
65typedef VUSBURBHDR *PVUSBURBHDR;
66
67AssertCompileSizeAlignment(VUSBURBHDR, 8);
68
69/*********************************************************************************************************************************
70* Static Variables *
71*********************************************************************************************************************************/
72
73/*********************************************************************************************************************************
74* Internal Functions *
75*********************************************************************************************************************************/
76
77DECLHIDDEN(int) vusbUrbPoolInit(PVUSBURBPOOL pUrbPool)
78{
79 int rc = RTCritSectInit(&pUrbPool->CritSectPool);
80 if (RT_SUCCESS(rc))
81 {
82 pUrbPool->cUrbsInPool = 0;
83 for (unsigned i = 0; i < RT_ELEMENTS(pUrbPool->aLstFreeUrbs); i++)
84 RTListInit(&pUrbPool->aLstFreeUrbs[i]);
85 }
86
87 return rc;
88}
89
90
91DECLHIDDEN(void) vusbUrbPoolDestroy(PVUSBURBPOOL pUrbPool)
92{
93 RTCritSectEnter(&pUrbPool->CritSectPool);
94 for (unsigned i = 0; i < RT_ELEMENTS(pUrbPool->aLstFreeUrbs); i++)
95 {
96 PVUSBURBHDR pHdr, pHdrNext;
97 RTListForEachSafe(&pUrbPool->aLstFreeUrbs[i], pHdr, pHdrNext, VUSBURBHDR, NdFree)
98 {
99 RTListNodeRemove(&pHdr->NdFree);
100
101 pHdr->cbAllocated = 0;
102 pHdr->Urb.u32Magic = 0;
103 pHdr->Urb.enmState = VUSBURBSTATE_INVALID;
104 RTMemFree(pHdr);
105 }
106 }
107 RTCritSectLeave(&pUrbPool->CritSectPool);
108 RTCritSectDelete(&pUrbPool->CritSectPool);
109}
110
111
112DECLHIDDEN(PVUSBURB) vusbUrbPoolAlloc(PVUSBURBPOOL pUrbPool, VUSBXFERTYPE enmType,
113 VUSBDIRECTION enmDir, size_t cbData, size_t cbHci,
114 size_t cbHciTd, unsigned cTds)
115{
116 /*
117 * Reuse or allocate a new URB.
118 */
119 /** @todo The allocations should be done by the device, at least as an option, since the devices
120 * frequently wish to associate their own stuff with the in-flight URB or need special buffering
121 * (isochronous on Darwin for instance). */
122 /* Get the required amount of additional memory to allocate the whole state. */
123 size_t cbMem = cbData + sizeof(VUSBURBVUSBINT) + cbHci + cTds * cbHciTd;
124
125 AssertReturn(enmType < RT_ELEMENTS(pUrbPool->aLstFreeUrbs), NULL);
126
127 RTCritSectEnter(&pUrbPool->CritSectPool);
128 PVUSBURBHDR pHdr = NULL;
129 PVUSBURBHDR pIt, pItNext;
130 RTListForEachSafe(&pUrbPool->aLstFreeUrbs[enmType], pIt, pItNext, VUSBURBHDR, NdFree)
131 {
132 if (pIt->cbAllocated >= cbMem)
133 {
134 RTListNodeRemove(&pIt->NdFree);
135 Assert(pIt->Urb.u32Magic == VUSBURB_MAGIC);
136 Assert(pIt->Urb.enmState == VUSBURBSTATE_FREE);
137 /*
138 * If the allocation is far too big we increase the age counter too
139 * so we don't waste memory for a lot of small transfers
140 */
141 if (pIt->cbAllocated >= 2 * cbMem)
142 pIt->cAge++;
143 else
144 pIt->cAge = 0;
145 pHdr = pIt;
146 break;
147 }
148 else
149 {
150 /* Increase age and free if it reached a threshold. */
151 pIt->cAge++;
152 if (pIt->cAge == VUSBURB_AGE_MAX)
153 {
154 RTListNodeRemove(&pIt->NdFree);
155 ASMAtomicDecU32(&pUrbPool->cUrbsInPool);
156 RTMemFree(pIt);
157 }
158 }
159 }
160
161 if (!pHdr)
162 {
163 /* allocate a new one. */
164 size_t cbDataAllocated = cbMem <= _4K ? RT_ALIGN_32(cbMem, _1K)
165 : cbMem <= _32K ? RT_ALIGN_32(cbMem, _4K)
166 : RT_ALIGN_32(cbMem, 16*_1K);
167
168 pHdr = (PVUSBURBHDR)RTMemAllocZ(RT_OFFSETOF(VUSBURBHDR, Urb.abData[cbDataAllocated]));
169 if (RT_UNLIKELY(!pHdr))
170 {
171 RTCritSectLeave(&pUrbPool->CritSectPool);
172 AssertLogRelFailedReturn(NULL);
173 }
174
175 pHdr->cbAllocated = cbDataAllocated;
176 pHdr->cAge = 0;
177 ASMAtomicIncU32(&pUrbPool->cUrbsInPool);
178 }
179 RTCritSectLeave(&pUrbPool->CritSectPool);
180
181 Assert(pHdr->cbAllocated >= cbMem);
182
183 /*
184 * (Re)init the URB
185 */
186 uint32_t offAlloc = cbData;
187 PVUSBURB pUrb = &pHdr->Urb;
188 pUrb->u32Magic = VUSBURB_MAGIC;
189 pUrb->enmState = VUSBURBSTATE_ALLOCATED;
190 pUrb->fCompleting = false;
191 pUrb->pszDesc = NULL;
192 pUrb->pVUsb = (PVUSBURBVUSB)&pUrb->abData[offAlloc];
193 offAlloc += sizeof(VUSBURBVUSBINT);
194 pUrb->pVUsb->pUrb = pUrb;
195 pUrb->pVUsb->pvFreeCtx = NULL;
196 pUrb->pVUsb->pfnFree = NULL;
197 pUrb->pVUsb->pCtrlUrb = NULL;
198 pUrb->pVUsb->u64SubmitTS = 0;
199 pUrb->pVUsb->pvBuffered = NULL;
200 pUrb->Dev.pvPrivate = NULL;
201 pUrb->Dev.pNext = NULL;
202 pUrb->EndPt = ~0;
203 pUrb->enmType = enmType;
204 pUrb->enmDir = enmDir;
205 pUrb->fShortNotOk = false;
206 pUrb->enmStatus = VUSBSTATUS_INVALID;
207 pUrb->cbData = cbData;
208 pUrb->pHci = cbHci ? (PVUSBURBHCI)&pUrb->abData[offAlloc] : NULL;
209 offAlloc += cbHci;
210 pUrb->paTds = (cbHciTd && cTds) ? (PVUSBURBHCITD)&pUrb->abData[offAlloc] : NULL;
211
212 return pUrb;
213}
214
215
216DECLHIDDEN(void) vusbUrbPoolFree(PVUSBURBPOOL pUrbPool, PVUSBURB pUrb)
217{
218 PVUSBURBHDR pHdr = VUSBURBPOOL_URB_2_URBHDR(pUrb);
219
220 /* URBs which aged too much because they are too big are freed. */
221 if (pHdr->cAge == VUSBURB_AGE_MAX)
222 {
223 ASMAtomicDecU32(&pUrbPool->cUrbsInPool);
224 RTMemFree(pHdr);
225 }
226 else
227 {
228 /* Put it into the list of free URBs. */
229 VUSBXFERTYPE enmType = pUrb->enmType;
230 AssertReturnVoid(enmType < RT_ELEMENTS(pUrbPool->aLstFreeUrbs));
231 RTCritSectEnter(&pUrbPool->CritSectPool);
232 pUrb->enmState = VUSBURBSTATE_FREE;
233 RTListAppend(&pUrbPool->aLstFreeUrbs[enmType], &pHdr->NdFree);
234 RTCritSectLeave(&pUrbPool->CritSectPool);
235 }
236}
237
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