VirtualBox

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

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

AsyncCompletion/Cache: Allow more than one active flush at the same time and use the async flush API if the host supports it

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