VirtualBox

source: vbox/trunk/src/VBox/VMM/VMMR3/PDMAsyncCompletionFile.cpp@ 42418

Last change on this file since 42418 was 41861, checked in by vboxsync, 13 years ago

PDMAsyncCompletionFile.cpp: Removed RT_STRICT as it breaks the strict build (added 'temporarily' about a year back, r71670).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 43.7 KB
Line 
1/* $Id: PDMAsyncCompletionFile.cpp 41861 2012-06-21 15:06:20Z vboxsync $ */
2/** @file
3 * PDM Async I/O - Transport data asynchronous in R3 using EMT.
4 */
5
6/*
7 * Copyright (C) 2006-2011 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_PDM_ASYNC_COMPLETION
23#include "PDMInternal.h"
24#include <VBox/vmm/pdm.h>
25#include <VBox/vmm/mm.h>
26#include <VBox/vmm/vm.h>
27#include <VBox/err.h>
28#include <VBox/log.h>
29#include <VBox/dbg.h>
30#include <VBox/vmm/uvm.h>
31
32#include <iprt/asm.h>
33#include <iprt/assert.h>
34#include <iprt/critsect.h>
35#include <iprt/env.h>
36#include <iprt/file.h>
37#include <iprt/mem.h>
38#include <iprt/semaphore.h>
39#include <iprt/string.h>
40#include <iprt/thread.h>
41#include <iprt/path.h>
42
43#include "PDMAsyncCompletionFileInternal.h"
44
45/*******************************************************************************
46* Internal Functions *
47*******************************************************************************/
48#ifdef VBOX_WITH_DEBUGGER
49static DECLCALLBACK(int) pdmacEpFileErrorInject(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR pArgs, unsigned cArgs);
50# ifdef PDM_ASYNC_COMPLETION_FILE_WITH_DELAY
51static DECLCALLBACK(int) pdmacEpFileDelayInject(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR pArgs, unsigned cArgs);
52# endif
53#endif
54
55/*******************************************************************************
56* Global Variables *
57*******************************************************************************/
58#ifdef VBOX_WITH_DEBUGGER
59static const DBGCVARDESC g_aInjectErrorArgs[] =
60{
61 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
62 { 1, 1, DBGCVAR_CAT_STRING, 0, "direction", "write/read." },
63 { 1, 1, DBGCVAR_CAT_STRING, 0, "filename", "Filename." },
64 { 1, 1, DBGCVAR_CAT_NUMBER, 0, "errcode", "VBox status code." },
65};
66
67# ifdef PDM_ASYNC_COMPLETION_FILE_WITH_DELAY
68static const DBGCVARDESC g_aInjectDelayArgs[] =
69{
70 /* cTimesMin, cTimesMax, enmCategory, fFlags, pszName, pszDescription */
71 { 1, 1, DBGCVAR_CAT_STRING, 0, "direction", "write|read|flush|any." },
72 { 1, 1, DBGCVAR_CAT_STRING, 0, "filename", "Filename." },
73 { 1, 1, DBGCVAR_CAT_NUMBER, 0, "delay", "Delay in milliseconds." },
74 { 1, 1, DBGCVAR_CAT_NUMBER, 0, "reqs", "Number of requests to delay." },
75};
76# endif
77
78/** Command descriptors. */
79static const DBGCCMD g_aCmds[] =
80{
81 /* pszCmd, cArgsMin, cArgsMax, paArgDesc, cArgDescs, fFlags, pfnHandler pszSyntax, ....pszDescription */
82 { "injecterror", 3, 3, &g_aInjectErrorArgs[0], 3, 0, pdmacEpFileErrorInject, "", "Inject error into I/O subsystem." }
83# ifdef PDM_ASYNC_COMPLETION_FILE_WITH_DELAY
84 ,{ "injectdelay", 4, 4, &g_aInjectDelayArgs[0], 4, 0, pdmacEpFileDelayInject, "", "Inject a delay of a request." }
85# endif
86};
87#endif
88
89
90/**
91 * Frees a task.
92 *
93 * @returns nothing.
94 * @param pEndpoint Pointer to the endpoint the segment was for.
95 * @param pTask The task to free.
96 */
97void pdmacFileTaskFree(PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint, PPDMACTASKFILE pTask)
98{
99 PPDMASYNCCOMPLETIONEPCLASSFILE pEpClass = (PPDMASYNCCOMPLETIONEPCLASSFILE)pEndpoint->Core.pEpClass;
100
101 LogFlowFunc((": pEndpoint=%p pTask=%p\n", pEndpoint, pTask));
102
103 /* Try the per endpoint cache first. */
104 if (pEndpoint->cTasksCached < pEpClass->cTasksCacheMax)
105 {
106 /* Add it to the list. */
107 pEndpoint->pTasksFreeTail->pNext = pTask;
108 pEndpoint->pTasksFreeTail = pTask;
109 ASMAtomicIncU32(&pEndpoint->cTasksCached);
110 }
111 else
112 {
113 Log(("Freeing task %p because all caches are full\n", pTask));
114 MMR3HeapFree(pTask);
115 }
116}
117
118/**
119 * Allocates a task segment
120 *
121 * @returns Pointer to the new task segment or NULL
122 * @param pEndpoint Pointer to the endpoint
123 */
124PPDMACTASKFILE pdmacFileTaskAlloc(PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint)
125{
126 PPDMACTASKFILE pTask = NULL;
127
128 /* Try the small per endpoint cache first. */
129 if (pEndpoint->pTasksFreeHead == pEndpoint->pTasksFreeTail)
130 {
131 /* Try the bigger endpoint class cache. */
132 PPDMASYNCCOMPLETIONEPCLASSFILE pEndpointClass = (PPDMASYNCCOMPLETIONEPCLASSFILE)pEndpoint->Core.pEpClass;
133
134 /*
135 * Allocate completely new.
136 * If this fails we return NULL.
137 */
138 int rc = MMR3HeapAllocZEx(pEndpointClass->Core.pVM, MM_TAG_PDM_ASYNC_COMPLETION,
139 sizeof(PDMACTASKFILE),
140 (void **)&pTask);
141 if (RT_FAILURE(rc))
142 pTask = NULL;
143
144 LogFlow(("Allocated task %p\n", pTask));
145 }
146 else
147 {
148 /* Grab a free task from the head. */
149 AssertMsg(pEndpoint->cTasksCached > 0, ("No tasks cached but list contains more than one element\n"));
150
151 pTask = pEndpoint->pTasksFreeHead;
152 pEndpoint->pTasksFreeHead = pTask->pNext;
153 ASMAtomicDecU32(&pEndpoint->cTasksCached);
154 }
155
156 pTask->pNext = NULL;
157
158 return pTask;
159}
160
161PPDMACTASKFILE pdmacFileEpGetNewTasks(PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint)
162{
163 /*
164 * Get pending tasks.
165 */
166 PPDMACTASKFILE pTasks = ASMAtomicXchgPtrT(&pEndpoint->pTasksNewHead, NULL, PPDMACTASKFILE);
167
168 /* Reverse the list to process in FIFO order. */
169 if (pTasks)
170 {
171 PPDMACTASKFILE pTask = pTasks;
172
173 pTasks = NULL;
174
175 while (pTask)
176 {
177 PPDMACTASKFILE pCur = pTask;
178 pTask = pTask->pNext;
179 pCur->pNext = pTasks;
180 pTasks = pCur;
181 }
182 }
183
184 return pTasks;
185}
186
187static void pdmacFileAioMgrWakeup(PPDMACEPFILEMGR pAioMgr)
188{
189 bool fWokenUp = ASMAtomicXchgBool(&pAioMgr->fWokenUp, true);
190 if (!fWokenUp)
191 {
192 bool fWaitingEventSem = ASMAtomicReadBool(&pAioMgr->fWaitingEventSem);
193 if (fWaitingEventSem)
194 {
195 int rc = RTSemEventSignal(pAioMgr->EventSem);
196 AssertRC(rc);
197 }
198 }
199}
200
201static int pdmacFileAioMgrWaitForBlockingEvent(PPDMACEPFILEMGR pAioMgr, PDMACEPFILEAIOMGRBLOCKINGEVENT enmEvent)
202{
203 ASMAtomicWriteU32((volatile uint32_t *)&pAioMgr->enmBlockingEvent, enmEvent);
204 Assert(!pAioMgr->fBlockingEventPending);
205 ASMAtomicXchgBool(&pAioMgr->fBlockingEventPending, true);
206
207 /* Wakeup the async I/O manager */
208 pdmacFileAioMgrWakeup(pAioMgr);
209
210 /* Wait for completion. */
211 int rc = RTSemEventWait(pAioMgr->EventSemBlock, RT_INDEFINITE_WAIT);
212 AssertRC(rc);
213
214 ASMAtomicXchgBool(&pAioMgr->fBlockingEventPending, false);
215 ASMAtomicWriteU32((volatile uint32_t *)&pAioMgr->enmBlockingEvent, PDMACEPFILEAIOMGRBLOCKINGEVENT_INVALID);
216
217 return rc;
218}
219
220int pdmacFileAioMgrAddEndpoint(PPDMACEPFILEMGR pAioMgr, PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint)
221{
222 LogFlowFunc(("pAioMgr=%#p pEndpoint=%#p{%s}\n", pAioMgr, pEndpoint, pEndpoint->Core.pszUri));
223
224 /* Update the assigned I/O manager. */
225 ASMAtomicWritePtr(&pEndpoint->pAioMgr, pAioMgr);
226
227 int rc = RTCritSectEnter(&pAioMgr->CritSectBlockingEvent);
228 AssertRCReturn(rc, rc);
229
230 ASMAtomicWritePtr(&pAioMgr->BlockingEventData.AddEndpoint.pEndpoint, pEndpoint);
231 rc = pdmacFileAioMgrWaitForBlockingEvent(pAioMgr, PDMACEPFILEAIOMGRBLOCKINGEVENT_ADD_ENDPOINT);
232 ASMAtomicWriteNullPtr(&pAioMgr->BlockingEventData.AddEndpoint.pEndpoint);
233
234 RTCritSectLeave(&pAioMgr->CritSectBlockingEvent);
235
236 return rc;
237}
238
239#ifdef SOME_UNUSED_FUNCTION
240static int pdmacFileAioMgrRemoveEndpoint(PPDMACEPFILEMGR pAioMgr, PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint)
241{
242 int rc = RTCritSectEnter(&pAioMgr->CritSectBlockingEvent);
243 AssertRCReturn(rc, rc);
244
245 ASMAtomicWritePtr(&pAioMgr->BlockingEventData.RemoveEndpoint.pEndpoint, pEndpoint);
246 rc = pdmacFileAioMgrWaitForBlockingEvent(pAioMgr, PDMACEPFILEAIOMGRBLOCKINGEVENT_REMOVE_ENDPOINT);
247 ASMAtomicWriteNullPtr(&pAioMgr->BlockingEventData.RemoveEndpoint.pEndpoint);
248
249 RTCritSectLeave(&pAioMgr->CritSectBlockingEvent);
250
251 return rc;
252}
253#endif
254
255static int pdmacFileAioMgrCloseEndpoint(PPDMACEPFILEMGR pAioMgr, PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint)
256{
257 int rc = RTCritSectEnter(&pAioMgr->CritSectBlockingEvent);
258 AssertRCReturn(rc, rc);
259
260 ASMAtomicWritePtr(&pAioMgr->BlockingEventData.CloseEndpoint.pEndpoint, pEndpoint);
261 rc = pdmacFileAioMgrWaitForBlockingEvent(pAioMgr, PDMACEPFILEAIOMGRBLOCKINGEVENT_CLOSE_ENDPOINT);
262 ASMAtomicWriteNullPtr(&pAioMgr->BlockingEventData.CloseEndpoint.pEndpoint);
263
264 RTCritSectLeave(&pAioMgr->CritSectBlockingEvent);
265
266 return rc;
267}
268
269static int pdmacFileAioMgrShutdown(PPDMACEPFILEMGR pAioMgr)
270{
271 int rc = RTCritSectEnter(&pAioMgr->CritSectBlockingEvent);
272 AssertRCReturn(rc, rc);
273
274 rc = pdmacFileAioMgrWaitForBlockingEvent(pAioMgr, PDMACEPFILEAIOMGRBLOCKINGEVENT_SHUTDOWN);
275
276 RTCritSectLeave(&pAioMgr->CritSectBlockingEvent);
277
278 return rc;
279}
280
281int pdmacFileEpAddTask(PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint, PPDMACTASKFILE pTask)
282{
283 PPDMACTASKFILE pNext;
284 do
285 {
286 pNext = pEndpoint->pTasksNewHead;
287 pTask->pNext = pNext;
288 } while (!ASMAtomicCmpXchgPtr(&pEndpoint->pTasksNewHead, pTask, pNext));
289
290 pdmacFileAioMgrWakeup(ASMAtomicReadPtrT(&pEndpoint->pAioMgr, PPDMACEPFILEMGR));
291
292 return VINF_SUCCESS;
293}
294
295void pdmacFileEpTaskCompleted(PPDMACTASKFILE pTask, void *pvUser, int rc)
296{
297 PPDMASYNCCOMPLETIONTASKFILE pTaskFile = (PPDMASYNCCOMPLETIONTASKFILE)pvUser;
298
299 LogFlowFunc(("pTask=%#p pvUser=%#p rc=%Rrc\n", pTask, pvUser, rc));
300
301 if (pTask->enmTransferType == PDMACTASKFILETRANSFER_FLUSH)
302 pdmR3AsyncCompletionCompleteTask(&pTaskFile->Core, rc, true);
303 else
304 {
305 Assert((uint32_t)pTask->DataSeg.cbSeg == pTask->DataSeg.cbSeg && (int32_t)pTask->DataSeg.cbSeg >= 0);
306 uint32_t uOld = ASMAtomicSubS32(&pTaskFile->cbTransferLeft, (int32_t)pTask->DataSeg.cbSeg);
307
308 /* The first error will be returned. */
309 if (RT_FAILURE(rc))
310 ASMAtomicCmpXchgS32(&pTaskFile->rc, rc, VINF_SUCCESS);
311#ifdef VBOX_WITH_DEBUGGER
312 else
313 {
314 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pTaskFile->Core.pEndpoint;
315
316 /* Overwrite with injected error code. */
317 if (pTask->enmTransferType == PDMACTASKFILETRANSFER_READ)
318 rc = ASMAtomicXchgS32(&pEpFile->rcReqRead, VINF_SUCCESS);
319 else
320 rc = ASMAtomicXchgS32(&pEpFile->rcReqWrite, VINF_SUCCESS);
321
322 if (RT_FAILURE(rc))
323 ASMAtomicCmpXchgS32(&pTaskFile->rc, rc, VINF_SUCCESS);
324 }
325#endif
326
327 if (!(uOld - pTask->DataSeg.cbSeg)
328 && !ASMAtomicXchgBool(&pTaskFile->fCompleted, true))
329 {
330#ifdef PDM_ASYNC_COMPLETION_FILE_WITH_DELAY
331 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pTaskFile->Core.pEndpoint;
332
333 /* Check if we should delay completion of the request. */
334 if ( ASMAtomicReadU32(&pEpFile->msDelay) > 0
335 && ASMAtomicReadU32(&pEpFile->cReqsDelay) > 0)
336 {
337 ASMAtomicDecU32(&pEpFile->cReqsDelay);
338
339 /* Arm the delay. */
340 pTaskFile->tsDelayEnd = RTTimeProgramMilliTS() + pEpFile->msDelay;
341
342 /* Append to the list. */
343 PPDMASYNCCOMPLETIONTASKFILE pHead = NULL;
344 do
345 {
346 pHead = ASMAtomicReadPtrT(&pEpFile->pDelayedHead, PPDMASYNCCOMPLETIONTASKFILE);
347 pTaskFile->pDelayedNext = pHead;
348 } while (!ASMAtomicCmpXchgPtr(&pEpFile->pDelayedHead, pTaskFile, pHead));
349
350 LogRel(("AIOMgr: Delaying request %#p for %u ms\n", pTaskFile, pEpFile->msDelay));
351 return;
352 }
353#endif
354 pdmR3AsyncCompletionCompleteTask(&pTaskFile->Core, pTaskFile->rc, true);
355
356#ifdef PDM_ASYNC_COMPLETION_FILE_WITH_DELAY
357 /* Check for an expired delay. */
358 if (pEpFile->pDelayedHead != NULL)
359 {
360 uint64_t tsCur = RTTimeProgramMilliTS();
361 pTaskFile = ASMAtomicXchgPtrT(&pEpFile->pDelayedHead, NULL, PPDMASYNCCOMPLETIONTASKFILE);
362
363 while (pTaskFile)
364 {
365 PPDMASYNCCOMPLETIONTASKFILE pTmp = pTaskFile;
366 pTaskFile = pTaskFile->pDelayedNext;
367
368 if (tsCur >= pTmp->tsDelayEnd)
369 {
370 LogRel(("AIOMgr: Delayed request %#p completed\n", pTmp));
371 pdmR3AsyncCompletionCompleteTask(&pTmp->Core, pTmp->rc, true);
372 }
373 else
374 {
375 /* Prepend to the delayed list again. */
376 PPDMASYNCCOMPLETIONTASKFILE pHead = NULL;
377 do
378 {
379 pHead = ASMAtomicReadPtrT(&pEpFile->pDelayedHead, PPDMASYNCCOMPLETIONTASKFILE);
380 pTmp->pDelayedNext = pHead;
381 } while (!ASMAtomicCmpXchgPtr(&pEpFile->pDelayedHead, pTmp, pHead));
382 }
383 }
384 }
385#endif
386 }
387 }
388}
389
390DECLINLINE(void) pdmacFileEpTaskInit(PPDMASYNCCOMPLETIONTASK pTask, size_t cbTransfer)
391{
392 PPDMASYNCCOMPLETIONTASKFILE pTaskFile = (PPDMASYNCCOMPLETIONTASKFILE)pTask;
393
394 Assert((uint32_t)cbTransfer == cbTransfer && (int32_t)cbTransfer >= 0);
395 ASMAtomicWriteS32(&pTaskFile->cbTransferLeft, (int32_t)cbTransfer);
396 ASMAtomicWriteBool(&pTaskFile->fCompleted, false);
397 ASMAtomicWriteS32(&pTaskFile->rc, VINF_SUCCESS);
398}
399
400int pdmacFileEpTaskInitiate(PPDMASYNCCOMPLETIONTASK pTask,
401 PPDMASYNCCOMPLETIONENDPOINT pEndpoint, RTFOFF off,
402 PCRTSGSEG paSegments, size_t cSegments,
403 size_t cbTransfer, PDMACTASKFILETRANSFER enmTransfer)
404{
405 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
406 PPDMASYNCCOMPLETIONTASKFILE pTaskFile = (PPDMASYNCCOMPLETIONTASKFILE)pTask;
407
408 Assert( (enmTransfer == PDMACTASKFILETRANSFER_READ)
409 || (enmTransfer == PDMACTASKFILETRANSFER_WRITE));
410
411 for (size_t i = 0; i < cSegments; i++)
412 {
413 PPDMACTASKFILE pIoTask = pdmacFileTaskAlloc(pEpFile);
414 AssertPtr(pIoTask);
415
416 pIoTask->pEndpoint = pEpFile;
417 pIoTask->enmTransferType = enmTransfer;
418 pIoTask->Off = off;
419 pIoTask->DataSeg.cbSeg = paSegments[i].cbSeg;
420 pIoTask->DataSeg.pvSeg = paSegments[i].pvSeg;
421 pIoTask->pvUser = pTaskFile;
422 pIoTask->pfnCompleted = pdmacFileEpTaskCompleted;
423
424 /* Send it off to the I/O manager. */
425 pdmacFileEpAddTask(pEpFile, pIoTask);
426 off += paSegments[i].cbSeg;
427 cbTransfer -= paSegments[i].cbSeg;
428 }
429
430 AssertMsg(!cbTransfer, ("Incomplete transfer %u bytes left\n", cbTransfer));
431
432 return VINF_AIO_TASK_PENDING;
433}
434
435/**
436 * Creates a new async I/O manager.
437 *
438 * @returns VBox status code.
439 * @param pEpClass Pointer to the endpoint class data.
440 * @param ppAioMgr Where to store the pointer to the new async I/O manager on success.
441 * @param enmMgrType Wanted manager type - can be overwritten by the global override.
442 */
443int pdmacFileAioMgrCreate(PPDMASYNCCOMPLETIONEPCLASSFILE pEpClass, PPPDMACEPFILEMGR ppAioMgr,
444 PDMACEPFILEMGRTYPE enmMgrType)
445{
446 LogFlowFunc((": Entered\n"));
447
448 PPDMACEPFILEMGR pAioMgrNew;
449 int rc = MMR3HeapAllocZEx(pEpClass->Core.pVM, MM_TAG_PDM_ASYNC_COMPLETION, sizeof(PDMACEPFILEMGR), (void **)&pAioMgrNew);
450 if (RT_SUCCESS(rc))
451 {
452 if (enmMgrType < pEpClass->enmMgrTypeOverride)
453 pAioMgrNew->enmMgrType = enmMgrType;
454 else
455 pAioMgrNew->enmMgrType = pEpClass->enmMgrTypeOverride;
456
457 pAioMgrNew->msBwLimitExpired = RT_INDEFINITE_WAIT;
458
459 rc = RTSemEventCreate(&pAioMgrNew->EventSem);
460 if (RT_SUCCESS(rc))
461 {
462 rc = RTSemEventCreate(&pAioMgrNew->EventSemBlock);
463 if (RT_SUCCESS(rc))
464 {
465 rc = RTCritSectInit(&pAioMgrNew->CritSectBlockingEvent);
466 if (RT_SUCCESS(rc))
467 {
468 /* Init the rest of the manager. */
469 if (pAioMgrNew->enmMgrType != PDMACEPFILEMGRTYPE_SIMPLE)
470 rc = pdmacFileAioMgrNormalInit(pAioMgrNew);
471
472 if (RT_SUCCESS(rc))
473 {
474 pAioMgrNew->enmState = PDMACEPFILEMGRSTATE_RUNNING;
475
476 rc = RTThreadCreateF(&pAioMgrNew->Thread,
477 pAioMgrNew->enmMgrType == PDMACEPFILEMGRTYPE_SIMPLE
478 ? pdmacFileAioMgrFailsafe
479 : pdmacFileAioMgrNormal,
480 pAioMgrNew,
481 0,
482 RTTHREADTYPE_IO,
483 0,
484 "AioMgr%d-%s", pEpClass->cAioMgrs,
485 pAioMgrNew->enmMgrType == PDMACEPFILEMGRTYPE_SIMPLE
486 ? "F"
487 : "N");
488 if (RT_SUCCESS(rc))
489 {
490 /* Link it into the list. */
491 RTCritSectEnter(&pEpClass->CritSect);
492 pAioMgrNew->pNext = pEpClass->pAioMgrHead;
493 if (pEpClass->pAioMgrHead)
494 pEpClass->pAioMgrHead->pPrev = pAioMgrNew;
495 pEpClass->pAioMgrHead = pAioMgrNew;
496 pEpClass->cAioMgrs++;
497 RTCritSectLeave(&pEpClass->CritSect);
498
499 *ppAioMgr = pAioMgrNew;
500
501 Log(("PDMAC: Successfully created new file AIO Mgr {%s}\n", RTThreadGetName(pAioMgrNew->Thread)));
502 return VINF_SUCCESS;
503 }
504 pdmacFileAioMgrNormalDestroy(pAioMgrNew);
505 }
506 RTCritSectDelete(&pAioMgrNew->CritSectBlockingEvent);
507 }
508 RTSemEventDestroy(pAioMgrNew->EventSem);
509 }
510 RTSemEventDestroy(pAioMgrNew->EventSemBlock);
511 }
512 MMR3HeapFree(pAioMgrNew);
513 }
514
515 LogFlowFunc((": Leave rc=%Rrc\n", rc));
516
517 return rc;
518}
519
520/**
521 * Destroys a async I/O manager.
522 *
523 * @returns nothing.
524 * @param pAioMgr The async I/O manager to destroy.
525 */
526static void pdmacFileAioMgrDestroy(PPDMASYNCCOMPLETIONEPCLASSFILE pEpClassFile, PPDMACEPFILEMGR pAioMgr)
527{
528 int rc = pdmacFileAioMgrShutdown(pAioMgr);
529 AssertRC(rc);
530
531 /* Unlink from the list. */
532 rc = RTCritSectEnter(&pEpClassFile->CritSect);
533 AssertRC(rc);
534
535 PPDMACEPFILEMGR pPrev = pAioMgr->pPrev;
536 PPDMACEPFILEMGR pNext = pAioMgr->pNext;
537
538 if (pPrev)
539 pPrev->pNext = pNext;
540 else
541 pEpClassFile->pAioMgrHead = pNext;
542
543 if (pNext)
544 pNext->pPrev = pPrev;
545
546 pEpClassFile->cAioMgrs--;
547 rc = RTCritSectLeave(&pEpClassFile->CritSect);
548 AssertRC(rc);
549
550 /* Free the resources. */
551 RTCritSectDelete(&pAioMgr->CritSectBlockingEvent);
552 RTSemEventDestroy(pAioMgr->EventSem);
553 if (pAioMgr->enmMgrType != PDMACEPFILEMGRTYPE_SIMPLE)
554 pdmacFileAioMgrNormalDestroy(pAioMgr);
555
556 MMR3HeapFree(pAioMgr);
557}
558
559static int pdmacFileMgrTypeFromName(const char *pszVal, PPDMACEPFILEMGRTYPE penmMgrType)
560{
561 int rc = VINF_SUCCESS;
562
563 if (!RTStrCmp(pszVal, "Simple"))
564 *penmMgrType = PDMACEPFILEMGRTYPE_SIMPLE;
565 else if (!RTStrCmp(pszVal, "Async"))
566 *penmMgrType = PDMACEPFILEMGRTYPE_ASYNC;
567 else
568 rc = VERR_CFGM_CONFIG_UNKNOWN_VALUE;
569
570 return rc;
571}
572
573static const char *pdmacFileMgrTypeToName(PDMACEPFILEMGRTYPE enmMgrType)
574{
575 if (enmMgrType == PDMACEPFILEMGRTYPE_SIMPLE)
576 return "Simple";
577 if (enmMgrType == PDMACEPFILEMGRTYPE_ASYNC)
578 return "Async";
579
580 return NULL;
581}
582
583static int pdmacFileBackendTypeFromName(const char *pszVal, PPDMACFILEEPBACKEND penmBackendType)
584{
585 int rc = VINF_SUCCESS;
586
587 if (!RTStrCmp(pszVal, "Buffered"))
588 *penmBackendType = PDMACFILEEPBACKEND_BUFFERED;
589 else if (!RTStrCmp(pszVal, "NonBuffered"))
590 *penmBackendType = PDMACFILEEPBACKEND_NON_BUFFERED;
591 else
592 rc = VERR_CFGM_CONFIG_UNKNOWN_VALUE;
593
594 return rc;
595}
596
597static const char *pdmacFileBackendTypeToName(PDMACFILEEPBACKEND enmBackendType)
598{
599 if (enmBackendType == PDMACFILEEPBACKEND_BUFFERED)
600 return "Buffered";
601 if (enmBackendType == PDMACFILEEPBACKEND_NON_BUFFERED)
602 return "NonBuffered";
603
604 return NULL;
605}
606
607/**
608 * Get the size of the given file.
609 * Works for block devices too.
610 *
611 * @returns VBox status code.
612 * @param hFile The file handle.
613 * @param pcbSize Where to store the size of the file on success.
614 */
615static int pdmacFileEpNativeGetSize(RTFILE hFile, uint64_t *pcbSize)
616{
617 uint64_t cbFile;
618 int rc = RTFileGetSize(hFile, &cbFile);
619 if (RT_SUCCESS(rc))
620 *pcbSize = cbFile;
621
622 return rc;
623}
624
625#ifdef VBOX_WITH_DEBUGGER
626
627/**
628 * Error inject callback.
629 */
630static DECLCALLBACK(int) pdmacEpFileErrorInject(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR pArgs, unsigned cArgs)
631{
632 /*
633 * Validate input.
634 */
635 DBGC_CMDHLP_REQ_VM_RET(pCmdHlp, pCmd, pVM);
636 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, -1, cArgs == 3);
637 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, 0, pArgs[0].enmType == DBGCVAR_TYPE_STRING);
638 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, 1, pArgs[1].enmType == DBGCVAR_TYPE_STRING);
639 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, 2, pArgs[2].enmType == DBGCVAR_TYPE_NUMBER);
640
641 PPDMASYNCCOMPLETIONEPCLASSFILE pEpClassFile;
642 pEpClassFile = (PPDMASYNCCOMPLETIONEPCLASSFILE)pVM->pUVM->pdm.s.apAsyncCompletionEndpointClass[PDMASYNCCOMPLETIONEPCLASSTYPE_FILE];
643
644 /* Syntax is "read|write <filename> <status code>" */
645 bool fWrite;
646 if (!RTStrCmp(pArgs[0].u.pszString, "read"))
647 fWrite = false;
648 else if (!RTStrCmp(pArgs[0].u.pszString, "write"))
649 fWrite = true;
650 else
651 return DBGCCmdHlpFail(pCmdHlp, pCmd, "invalid transfer direction '%s'", pArgs[0].u.pszString);
652
653 int32_t rcToInject = (int32_t)pArgs[2].u.u64Number;
654 if ((uint64_t)rcToInject != pArgs[2].u.u64Number)
655 return DBGCCmdHlpFail(pCmdHlp, pCmd, "The status code '%lld' is out of range", pArgs[0].u.u64Number);
656
657
658 /*
659 * Search for the matching endpoint.
660 */
661 RTCritSectEnter(&pEpClassFile->Core.CritSect);
662
663 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEpClassFile->Core.pEndpointsHead;
664 while (pEpFile)
665 {
666 if (!RTStrCmp(pArgs[1].u.pszString, RTPathFilename(pEpFile->Core.pszUri)))
667 break;
668 pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEpFile->Core.pNext;
669 }
670
671 if (pEpFile)
672 {
673 /*
674 * Do the job.
675 */
676 if (fWrite)
677 ASMAtomicXchgS32(&pEpFile->rcReqWrite, rcToInject);
678 else
679 ASMAtomicXchgS32(&pEpFile->rcReqRead, rcToInject);
680
681 DBGCCmdHlpPrintf(pCmdHlp, "Injected %Rrc into '%s' for %s\n",
682 (int)rcToInject, pArgs[1].u.pszString, pArgs[0].u.pszString);
683 }
684
685 RTCritSectLeave(&pEpClassFile->Core.CritSect);
686
687 if (!pEpFile)
688 return DBGCCmdHlpFail(pCmdHlp, pCmd, "No file with name '%s' found", pArgs[1].u.pszString);
689 return VINF_SUCCESS;
690}
691
692# ifdef PDM_ASYNC_COMPLETION_FILE_WITH_DELAY
693/**
694 * Delay inject callback.
695 */
696static DECLCALLBACK(int) pdmacEpFileDelayInject(PCDBGCCMD pCmd, PDBGCCMDHLP pCmdHlp, PVM pVM, PCDBGCVAR pArgs, unsigned cArgs)
697{
698 /*
699 * Validate input.
700 */
701 DBGC_CMDHLP_REQ_VM_RET(pCmdHlp, pCmd, pVM);
702 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, -1, cArgs >= 3);
703 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, 0, pArgs[0].enmType == DBGCVAR_TYPE_STRING);
704 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, 1, pArgs[1].enmType == DBGCVAR_TYPE_STRING);
705 DBGC_CMDHLP_ASSERT_PARSER_RET(pCmdHlp, pCmd, 2, pArgs[2].enmType == DBGCVAR_TYPE_NUMBER);
706
707 PPDMASYNCCOMPLETIONEPCLASSFILE pEpClassFile;
708 pEpClassFile = (PPDMASYNCCOMPLETIONEPCLASSFILE)pVM->pUVM->pdm.s.apAsyncCompletionEndpointClass[PDMASYNCCOMPLETIONEPCLASSTYPE_FILE];
709
710 /* Syntax is "read|write|flush|any <filename> <delay> [reqs]" */
711 PDMACFILEREQTYPEDELAY enmDelayType = PDMACFILEREQTYPEDELAY_ANY;
712 if (!RTStrCmp(pArgs[0].u.pszString, "read"))
713 enmDelayType = PDMACFILEREQTYPEDELAY_READ;
714 else if (!RTStrCmp(pArgs[0].u.pszString, "write"))
715 enmDelayType = PDMACFILEREQTYPEDELAY_WRITE;
716 else if (!RTStrCmp(pArgs[0].u.pszString, "flush"))
717 enmDelayType = PDMACFILEREQTYPEDELAY_FLUSH;
718 else if (!RTStrCmp(pArgs[0].u.pszString, "any"))
719 enmDelayType = PDMACFILEREQTYPEDELAY_ANY;
720 else
721 return DBGCCmdHlpFail(pCmdHlp, pCmd, "invalid transfer direction '%s'", pArgs[0].u.pszString);
722
723 uint32_t msDelay = (uint32_t)pArgs[2].u.u64Number;
724 if ((uint64_t)msDelay != pArgs[2].u.u64Number)
725 return DBGCCmdHlpFail(pCmdHlp, pCmd, "The delay '%lld' is out of range", pArgs[0].u.u64Number);
726
727 uint32_t cReqsDelay = 1;
728 if (cArgs == 4)
729 cReqsDelay = (uint32_t)pArgs[3].u.u64Number;
730
731 /*
732 * Search for the matching endpoint.
733 */
734 RTCritSectEnter(&pEpClassFile->Core.CritSect);
735
736 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEpClassFile->Core.pEndpointsHead;
737 while (pEpFile)
738 {
739 if (!RTStrCmp(pArgs[1].u.pszString, RTPathFilename(pEpFile->Core.pszUri)))
740 break;
741 pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEpFile->Core.pNext;
742 }
743
744 if (pEpFile)
745 {
746 ASMAtomicWriteSize(&pEpFile->enmTypeDelay, enmDelayType);
747 ASMAtomicWriteU32(&pEpFile->msDelay, msDelay);
748 ASMAtomicWriteU32(&pEpFile->cReqsDelay, cReqsDelay);
749
750 DBGCCmdHlpPrintf(pCmdHlp, "Injected delay for the next %u requests of %u ms into '%s' for %s\n",
751 cReqsDelay, msDelay, pArgs[1].u.pszString, pArgs[0].u.pszString);
752 }
753
754 RTCritSectLeave(&pEpClassFile->Core.CritSect);
755
756 if (!pEpFile)
757 return DBGCCmdHlpFail(pCmdHlp, pCmd, "No file with name '%s' found", pArgs[1].u.pszString);
758 return VINF_SUCCESS;
759}
760# endif /* PDM_ASYNC_COMPLETION_FILE_WITH_DELAY */
761
762#endif /* VBOX_WITH_DEBUGGER */
763
764static int pdmacFileInitialize(PPDMASYNCCOMPLETIONEPCLASS pClassGlobals, PCFGMNODE pCfgNode)
765{
766 PPDMASYNCCOMPLETIONEPCLASSFILE pEpClassFile = (PPDMASYNCCOMPLETIONEPCLASSFILE)pClassGlobals;
767 RTFILEAIOLIMITS AioLimits; /** < Async I/O limitations. */
768
769 int rc = RTFileAioGetLimits(&AioLimits);
770#ifdef DEBUG
771 if (RT_SUCCESS(rc) && RTEnvExist("VBOX_ASYNC_IO_FAILBACK"))
772 rc = VERR_ENV_VAR_NOT_FOUND;
773#endif
774 if (RT_FAILURE(rc))
775 {
776 LogRel(("AIO: Async I/O manager not supported (rc=%Rrc). Falling back to simple manager\n",
777 rc));
778 pEpClassFile->enmMgrTypeOverride = PDMACEPFILEMGRTYPE_SIMPLE;
779 pEpClassFile->enmEpBackendDefault = PDMACFILEEPBACKEND_BUFFERED;
780 }
781 else
782 {
783 pEpClassFile->uBitmaskAlignment = AioLimits.cbBufferAlignment ? ~((RTR3UINTPTR)AioLimits.cbBufferAlignment - 1) : RTR3UINTPTR_MAX;
784 pEpClassFile->cReqsOutstandingMax = AioLimits.cReqsOutstandingMax;
785
786 if (pCfgNode)
787 {
788 /* Query the default manager type */
789 char *pszVal = NULL;
790 rc = CFGMR3QueryStringAllocDef(pCfgNode, "IoMgr", &pszVal, "Async");
791 AssertLogRelRCReturn(rc, rc);
792
793 rc = pdmacFileMgrTypeFromName(pszVal, &pEpClassFile->enmMgrTypeOverride);
794 MMR3HeapFree(pszVal);
795 if (RT_FAILURE(rc))
796 return rc;
797
798 LogRel(("AIOMgr: Default manager type is \"%s\"\n", pdmacFileMgrTypeToName(pEpClassFile->enmMgrTypeOverride)));
799
800 /* Query default backend type */
801 rc = CFGMR3QueryStringAllocDef(pCfgNode, "FileBackend", &pszVal, "NonBuffered");
802 AssertLogRelRCReturn(rc, rc);
803
804 rc = pdmacFileBackendTypeFromName(pszVal, &pEpClassFile->enmEpBackendDefault);
805 MMR3HeapFree(pszVal);
806 if (RT_FAILURE(rc))
807 return rc;
808
809 LogRel(("AIOMgr: Default file backend is \"%s\"\n", pdmacFileBackendTypeToName(pEpClassFile->enmEpBackendDefault)));
810
811#ifdef RT_OS_LINUX
812 if ( pEpClassFile->enmMgrTypeOverride == PDMACEPFILEMGRTYPE_ASYNC
813 && pEpClassFile->enmEpBackendDefault == PDMACFILEEPBACKEND_BUFFERED)
814 {
815 LogRel(("AIOMgr: Linux does not support buffered async I/O, changing to non buffered\n"));
816 pEpClassFile->enmEpBackendDefault = PDMACFILEEPBACKEND_NON_BUFFERED;
817 }
818#endif
819 }
820 else
821 {
822 /* No configuration supplied, set defaults */
823 pEpClassFile->enmEpBackendDefault = PDMACFILEEPBACKEND_NON_BUFFERED;
824 pEpClassFile->enmMgrTypeOverride = PDMACEPFILEMGRTYPE_ASYNC;
825 }
826 }
827
828 /* Init critical section. */
829 rc = RTCritSectInit(&pEpClassFile->CritSect);
830
831#ifdef VBOX_WITH_DEBUGGER
832 /* Install the error injection handler. */
833 if (RT_SUCCESS(rc))
834 {
835 rc = DBGCRegisterCommands(&g_aCmds[0], RT_ELEMENTS(g_aCmds));
836 AssertRC(rc);
837 }
838#endif
839
840 return rc;
841}
842
843static void pdmacFileTerminate(PPDMASYNCCOMPLETIONEPCLASS pClassGlobals)
844{
845 PPDMASYNCCOMPLETIONEPCLASSFILE pEpClassFile = (PPDMASYNCCOMPLETIONEPCLASSFILE)pClassGlobals;
846
847 /* All endpoints should be closed at this point. */
848 AssertMsg(!pEpClassFile->Core.pEndpointsHead, ("There are still endpoints left\n"));
849
850 /* Destroy all left async I/O managers. */
851 while (pEpClassFile->pAioMgrHead)
852 pdmacFileAioMgrDestroy(pEpClassFile, pEpClassFile->pAioMgrHead);
853
854 RTCritSectDelete(&pEpClassFile->CritSect);
855}
856
857static int pdmacFileEpInitialize(PPDMASYNCCOMPLETIONENDPOINT pEndpoint,
858 const char *pszUri, uint32_t fFlags)
859{
860 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
861 PPDMASYNCCOMPLETIONEPCLASSFILE pEpClassFile = (PPDMASYNCCOMPLETIONEPCLASSFILE)pEndpoint->pEpClass;
862 PDMACEPFILEMGRTYPE enmMgrType = pEpClassFile->enmMgrTypeOverride;
863 PDMACFILEEPBACKEND enmEpBackend = pEpClassFile->enmEpBackendDefault;
864
865 AssertMsgReturn((fFlags & ~(PDMACEP_FILE_FLAGS_READ_ONLY | PDMACEP_FILE_FLAGS_DONT_LOCK | PDMACEP_FILE_FLAGS_HOST_CACHE_ENABLED)) == 0,
866 ("PDMAsyncCompletion: Invalid flag specified\n"), VERR_INVALID_PARAMETER);
867
868 unsigned fFileFlags = RTFILE_O_OPEN;
869
870 /*
871 * Revert to the simple manager and the buffered backend if
872 * the host cache should be enabled.
873 */
874 if (fFlags & PDMACEP_FILE_FLAGS_HOST_CACHE_ENABLED)
875 {
876 enmMgrType = PDMACEPFILEMGRTYPE_SIMPLE;
877 enmEpBackend = PDMACFILEEPBACKEND_BUFFERED;
878 }
879
880 if (fFlags & PDMACEP_FILE_FLAGS_READ_ONLY)
881 fFileFlags |= RTFILE_O_READ | RTFILE_O_DENY_NONE;
882 else
883 {
884 fFileFlags |= RTFILE_O_READWRITE;
885
886 /*
887 * Opened in read/write mode. Check whether the caller wants to
888 * avoid the lock. Return an error in case caching is enabled
889 * because this can lead to data corruption.
890 */
891 if (fFlags & PDMACEP_FILE_FLAGS_DONT_LOCK)
892 fFileFlags |= RTFILE_O_DENY_NONE;
893 else
894 fFileFlags |= RTFILE_O_DENY_WRITE;
895 }
896
897 if (enmMgrType == PDMACEPFILEMGRTYPE_ASYNC)
898 fFileFlags |= RTFILE_O_ASYNC_IO;
899
900 int rc;
901 if (enmEpBackend == PDMACFILEEPBACKEND_NON_BUFFERED)
902 {
903 /*
904 * We only disable the cache if the size of the file is a multiple of 512.
905 * Certain hosts like Windows, Linux and Solaris require that transfer sizes
906 * are aligned to the volume sector size.
907 * If not we just make sure that the data is written to disk with RTFILE_O_WRITE_THROUGH
908 * which will trash the host cache but ensures that the host cache will not
909 * contain dirty buffers.
910 */
911 RTFILE hFile;
912 rc = RTFileOpen(&hFile, pszUri, RTFILE_O_READ | RTFILE_O_OPEN | RTFILE_O_DENY_NONE);
913 if (RT_SUCCESS(rc))
914 {
915 uint64_t cbSize;
916
917 rc = pdmacFileEpNativeGetSize(hFile, &cbSize);
918
919 if (RT_SUCCESS(rc) && ((cbSize % 512) == 0))
920 fFileFlags |= RTFILE_O_NO_CACHE;
921 else
922 {
923 /* Downgrade to the buffered backend */
924 enmEpBackend = PDMACFILEEPBACKEND_BUFFERED;
925
926#ifdef RT_OS_LINUX
927 fFileFlags &= ~RTFILE_O_ASYNC_IO;
928 enmMgrType = PDMACEPFILEMGRTYPE_SIMPLE;
929#endif
930 }
931 RTFileClose(hFile);
932 }
933 }
934
935 /* Open with final flags. */
936 rc = RTFileOpen(&pEpFile->hFile, pszUri, fFileFlags);
937 if ( rc == VERR_INVALID_FUNCTION
938 || rc == VERR_INVALID_PARAMETER)
939 {
940 LogRel(("pdmacFileEpInitialize: RTFileOpen %s / %08x failed with %Rrc\n",
941 pszUri, fFileFlags, rc));
942 /*
943 * Solaris doesn't support directio on ZFS so far. :-\
944 * Trying to enable it returns VERR_INVALID_FUNCTION
945 * (ENOTTY). Remove it and hope for the best.
946 * ZFS supports write throttling in case applications
947 * write more data than can be synced to the disk
948 * without blocking the whole application.
949 *
950 * On Linux we have the same problem with cifs.
951 * Have to disable async I/O here too because it requires O_DIRECT.
952 */
953 fFileFlags &= ~RTFILE_O_NO_CACHE;
954 enmEpBackend = PDMACFILEEPBACKEND_BUFFERED;
955
956#ifdef RT_OS_LINUX
957 fFileFlags &= ~RTFILE_O_ASYNC_IO;
958 enmMgrType = PDMACEPFILEMGRTYPE_SIMPLE;
959#endif
960
961 /* Open again. */
962 rc = RTFileOpen(&pEpFile->hFile, pszUri, fFileFlags);
963
964 if (RT_FAILURE(rc))
965 {
966 LogRel(("pdmacFileEpInitialize: RTFileOpen %s / %08x failed AGAIN(!) with %Rrc\n",
967 pszUri, fFileFlags, rc));
968 }
969 }
970
971 if (RT_SUCCESS(rc))
972 {
973 pEpFile->fFlags = fFileFlags;
974
975 rc = pdmacFileEpNativeGetSize(pEpFile->hFile, (uint64_t *)&pEpFile->cbFile);
976 if (RT_SUCCESS(rc))
977 {
978 /* Initialize the segment cache */
979 rc = MMR3HeapAllocZEx(pEpClassFile->Core.pVM, MM_TAG_PDM_ASYNC_COMPLETION,
980 sizeof(PDMACTASKFILE),
981 (void **)&pEpFile->pTasksFreeHead);
982 if (RT_SUCCESS(rc))
983 {
984 PPDMACEPFILEMGR pAioMgr = NULL;
985
986 pEpFile->pTasksFreeTail = pEpFile->pTasksFreeHead;
987 pEpFile->cTasksCached = 0;
988 pEpFile->enmBackendType = enmEpBackend;
989 /*
990 * Disable async flushes on Solaris for now.
991 * They cause weird hangs which needs more investigations.
992 */
993#ifndef RT_OS_SOLARIS
994 pEpFile->fAsyncFlushSupported = true;
995#else
996 pEpFile->fAsyncFlushSupported = false;
997#endif
998
999 if (enmMgrType == PDMACEPFILEMGRTYPE_SIMPLE)
1000 {
1001 /* Simple mode. Every file has its own async I/O manager. */
1002 rc = pdmacFileAioMgrCreate(pEpClassFile, &pAioMgr, PDMACEPFILEMGRTYPE_SIMPLE);
1003 AssertRC(rc);
1004 }
1005 else
1006 {
1007 pAioMgr = pEpClassFile->pAioMgrHead;
1008
1009 /* Check for an idling manager of the same type */
1010 while (pAioMgr)
1011 {
1012 if (pAioMgr->enmMgrType == enmMgrType)
1013 break;
1014 pAioMgr = pAioMgr->pNext;
1015 }
1016
1017 if (!pAioMgr)
1018 {
1019 rc = pdmacFileAioMgrCreate(pEpClassFile, &pAioMgr, enmMgrType);
1020 AssertRC(rc);
1021 }
1022 }
1023
1024 pEpFile->AioMgr.pTreeRangesLocked = (PAVLRFOFFTREE)RTMemAllocZ(sizeof(AVLRFOFFTREE));
1025 if (!pEpFile->AioMgr.pTreeRangesLocked)
1026 rc = VERR_NO_MEMORY;
1027 else
1028 {
1029 pEpFile->enmState = PDMASYNCCOMPLETIONENDPOINTFILESTATE_ACTIVE;
1030
1031 /* Assign the endpoint to the thread. */
1032 rc = pdmacFileAioMgrAddEndpoint(pAioMgr, pEpFile);
1033 if (RT_FAILURE(rc))
1034 {
1035 RTMemFree(pEpFile->AioMgr.pTreeRangesLocked);
1036 MMR3HeapFree(pEpFile->pTasksFreeHead);
1037 }
1038 }
1039 }
1040 }
1041
1042 if (RT_FAILURE(rc))
1043 RTFileClose(pEpFile->hFile);
1044 }
1045
1046#ifdef VBOX_WITH_STATISTICS
1047 if (RT_SUCCESS(rc))
1048 {
1049 STAMR3RegisterF(pEpClassFile->Core.pVM, &pEpFile->StatRead,
1050 STAMTYPE_PROFILE_ADV, STAMVISIBILITY_ALWAYS,
1051 STAMUNIT_TICKS_PER_CALL, "Time taken to read from the endpoint",
1052 "/PDM/AsyncCompletion/File/%s/Read", RTPathFilename(pEpFile->Core.pszUri));
1053
1054 STAMR3RegisterF(pEpClassFile->Core.pVM, &pEpFile->StatWrite,
1055 STAMTYPE_PROFILE_ADV, STAMVISIBILITY_ALWAYS,
1056 STAMUNIT_TICKS_PER_CALL, "Time taken to write to the endpoint",
1057 "/PDM/AsyncCompletion/File/%s/Write", RTPathFilename(pEpFile->Core.pszUri));
1058 }
1059#endif
1060
1061 if (RT_SUCCESS(rc))
1062 LogRel(("AIOMgr: Endpoint for file '%s' (flags %08x) created successfully\n", pszUri, pEpFile->fFlags));
1063
1064 return rc;
1065}
1066
1067static int pdmacFileEpRangesLockedDestroy(PAVLRFOFFNODECORE pNode, void *pvUser)
1068{
1069 NOREF(pNode); NOREF(pvUser);
1070 AssertMsgFailed(("The locked ranges tree should be empty at that point\n"));
1071 return VINF_SUCCESS;
1072}
1073
1074static int pdmacFileEpClose(PPDMASYNCCOMPLETIONENDPOINT pEndpoint)
1075{
1076 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
1077 PPDMASYNCCOMPLETIONEPCLASSFILE pEpClassFile = (PPDMASYNCCOMPLETIONEPCLASSFILE)pEndpoint->pEpClass;
1078
1079 /* Make sure that all tasks finished for this endpoint. */
1080 int rc = pdmacFileAioMgrCloseEndpoint(pEpFile->pAioMgr, pEpFile);
1081 AssertRC(rc);
1082
1083 /*
1084 * If the async I/O manager is in failsafe mode this is the only endpoint
1085 * he processes and thus can be destroyed now.
1086 */
1087 if (pEpFile->pAioMgr->enmMgrType == PDMACEPFILEMGRTYPE_SIMPLE)
1088 pdmacFileAioMgrDestroy(pEpClassFile, pEpFile->pAioMgr);
1089
1090 /* Free cached tasks. */
1091 PPDMACTASKFILE pTask = pEpFile->pTasksFreeHead;
1092
1093 while (pTask)
1094 {
1095 PPDMACTASKFILE pTaskFree = pTask;
1096 pTask = pTask->pNext;
1097 MMR3HeapFree(pTaskFree);
1098 }
1099
1100 /* Destroy the locked ranges tree now. */
1101 RTAvlrFileOffsetDestroy(pEpFile->AioMgr.pTreeRangesLocked, pdmacFileEpRangesLockedDestroy, NULL);
1102
1103 RTFileClose(pEpFile->hFile);
1104
1105#ifdef VBOX_WITH_STATISTICS
1106 STAMR3Deregister(pEpClassFile->Core.pVM, &pEpFile->StatRead);
1107 STAMR3Deregister(pEpClassFile->Core.pVM, &pEpFile->StatWrite);
1108#endif
1109
1110 return VINF_SUCCESS;
1111}
1112
1113static int pdmacFileEpRead(PPDMASYNCCOMPLETIONTASK pTask,
1114 PPDMASYNCCOMPLETIONENDPOINT pEndpoint, RTFOFF off,
1115 PCRTSGSEG paSegments, size_t cSegments,
1116 size_t cbRead)
1117{
1118 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
1119
1120 LogFlowFunc(("pTask=%#p pEndpoint=%#p off=%RTfoff paSegments=%#p cSegments=%zu cbRead=%zu\n",
1121 pTask, pEndpoint, off, paSegments, cSegments, cbRead));
1122
1123 if (RT_UNLIKELY((uint64_t)off + cbRead > pEpFile->cbFile))
1124 return VERR_EOF;
1125
1126 STAM_PROFILE_ADV_START(&pEpFile->StatRead, Read);
1127 pdmacFileEpTaskInit(pTask, cbRead);
1128 int rc = pdmacFileEpTaskInitiate(pTask, pEndpoint, off, paSegments, cSegments, cbRead,
1129 PDMACTASKFILETRANSFER_READ);
1130 STAM_PROFILE_ADV_STOP(&pEpFile->StatRead, Read);
1131
1132 return rc;
1133}
1134
1135static int pdmacFileEpWrite(PPDMASYNCCOMPLETIONTASK pTask,
1136 PPDMASYNCCOMPLETIONENDPOINT pEndpoint, RTFOFF off,
1137 PCRTSGSEG paSegments, size_t cSegments,
1138 size_t cbWrite)
1139{
1140 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
1141
1142 if (RT_UNLIKELY(pEpFile->fReadonly))
1143 return VERR_NOT_SUPPORTED;
1144
1145 STAM_PROFILE_ADV_START(&pEpFile->StatWrite, Write);
1146
1147 pdmacFileEpTaskInit(pTask, cbWrite);
1148
1149 int rc = pdmacFileEpTaskInitiate(pTask, pEndpoint, off, paSegments, cSegments, cbWrite,
1150 PDMACTASKFILETRANSFER_WRITE);
1151
1152 STAM_PROFILE_ADV_STOP(&pEpFile->StatWrite, Write);
1153
1154 return rc;
1155}
1156
1157static int pdmacFileEpFlush(PPDMASYNCCOMPLETIONTASK pTask,
1158 PPDMASYNCCOMPLETIONENDPOINT pEndpoint)
1159{
1160 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
1161 PPDMASYNCCOMPLETIONTASKFILE pTaskFile = (PPDMASYNCCOMPLETIONTASKFILE)pTask;
1162
1163 if (RT_UNLIKELY(pEpFile->fReadonly))
1164 return VERR_NOT_SUPPORTED;
1165
1166 pdmacFileEpTaskInit(pTask, 0);
1167
1168 PPDMACTASKFILE pIoTask = pdmacFileTaskAlloc(pEpFile);
1169 if (RT_UNLIKELY(!pIoTask))
1170 return VERR_NO_MEMORY;
1171
1172 pIoTask->pEndpoint = pEpFile;
1173 pIoTask->enmTransferType = PDMACTASKFILETRANSFER_FLUSH;
1174 pIoTask->pvUser = pTaskFile;
1175 pIoTask->pfnCompleted = pdmacFileEpTaskCompleted;
1176 pdmacFileEpAddTask(pEpFile, pIoTask);
1177
1178 return VINF_AIO_TASK_PENDING;
1179}
1180
1181static int pdmacFileEpGetSize(PPDMASYNCCOMPLETIONENDPOINT pEndpoint, uint64_t *pcbSize)
1182{
1183 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
1184
1185 *pcbSize = ASMAtomicReadU64(&pEpFile->cbFile);
1186
1187 return VINF_SUCCESS;
1188}
1189
1190static int pdmacFileEpSetSize(PPDMASYNCCOMPLETIONENDPOINT pEndpoint, uint64_t cbSize)
1191{
1192 int rc;
1193 PPDMASYNCCOMPLETIONENDPOINTFILE pEpFile = (PPDMASYNCCOMPLETIONENDPOINTFILE)pEndpoint;
1194
1195 rc = RTFileSetSize(pEpFile->hFile, cbSize);
1196 if (RT_SUCCESS(rc))
1197 ASMAtomicWriteU64(&pEpFile->cbFile, cbSize);
1198
1199 return rc;
1200}
1201
1202const PDMASYNCCOMPLETIONEPCLASSOPS g_PDMAsyncCompletionEndpointClassFile =
1203{
1204 /* u32Version */
1205 PDMAC_EPCLASS_OPS_VERSION,
1206 /* pcszName */
1207 "File",
1208 /* enmClassType */
1209 PDMASYNCCOMPLETIONEPCLASSTYPE_FILE,
1210 /* cbEndpointClassGlobal */
1211 sizeof(PDMASYNCCOMPLETIONEPCLASSFILE),
1212 /* cbEndpoint */
1213 sizeof(PDMASYNCCOMPLETIONENDPOINTFILE),
1214 /* cbTask */
1215 sizeof(PDMASYNCCOMPLETIONTASKFILE),
1216 /* pfnInitialize */
1217 pdmacFileInitialize,
1218 /* pfnTerminate */
1219 pdmacFileTerminate,
1220 /* pfnEpInitialize. */
1221 pdmacFileEpInitialize,
1222 /* pfnEpClose */
1223 pdmacFileEpClose,
1224 /* pfnEpRead */
1225 pdmacFileEpRead,
1226 /* pfnEpWrite */
1227 pdmacFileEpWrite,
1228 /* pfnEpFlush */
1229 pdmacFileEpFlush,
1230 /* pfnEpGetSize */
1231 pdmacFileEpGetSize,
1232 /* pfnEpSetSize */
1233 pdmacFileEpSetSize,
1234 /* u32VersionEnd */
1235 PDMAC_EPCLASS_OPS_VERSION
1236};
1237
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