VirtualBox

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

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

Devices: scm

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