VirtualBox

source: vbox/trunk/src/VBox/VMM/PDMAsyncCompletionFileFailsafe.cpp@ 22997

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

PDMAsyncCompletion: Add missing state transitions in failsafe manager

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 7.0 KB
Line 
1/* $Id: PDMAsyncCompletionFileFailsafe.cpp 22979 2009-09-13 16:39:21Z vboxsync $ */
2/** @file
3 * PDM Async I/O - Transport data asynchronous in R3 using EMT.
4 * Failsafe File I/O manager.
5 */
6
7/*
8 * Copyright (C) 2006-2008 Sun Microsystems, Inc.
9 *
10 * This file is part of VirtualBox Open Source Edition (OSE), as
11 * available from http://www.virtualbox.org. This file is free software;
12 * you can redistribute it and/or modify it under the terms of the GNU
13 * General Public License (GPL) as published by the Free Software
14 * Foundation, in version 2 as it comes in the "COPYING" file of the
15 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
16 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
17 *
18 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
19 * Clara, CA 95054 USA or visit http://www.sun.com if you need
20 * additional information or have any questions.
21 */
22#define LOG_GROUP LOG_GROUP_PDM_ASYNC_COMPLETION
23#include <iprt/types.h>
24#include <VBox/log.h>
25
26#include "PDMAsyncCompletionFileInternal.h"
27
28static int pdmacFileAioMgrFailsafeProcessEndpoint(PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint)
29{
30 int rc = VINF_SUCCESS;
31 PPDMACTASKFILE pTasks = pdmacFileEpGetNewTasks(pEndpoint);
32
33 while (pTasks)
34 {
35 PPDMACTASKFILE pCurr = pTasks;
36
37 pTasks = pTasks->pNext;
38
39 switch (pCurr->enmTransferType)
40 {
41 case PDMACTASKFILETRANSFER_FLUSH:
42 {
43 rc = RTFileFlush(pEndpoint->File);
44 break;
45 }
46 case PDMACTASKFILETRANSFER_READ:
47 case PDMACTASKFILETRANSFER_WRITE:
48 {
49 if (pCurr->enmTransferType == PDMACTASKFILETRANSFER_READ)
50 {
51 rc = RTFileReadAt(pEndpoint->File, pCurr->Off,
52 pCurr->DataSeg.pvSeg,
53 pCurr->DataSeg.cbSeg,
54 NULL);
55 }
56 else
57 {
58 if (RT_UNLIKELY((pCurr->Off + pCurr->DataSeg.cbSeg) > pEndpoint->cbFile))
59 {
60 ASMAtomicWriteU64(&pEndpoint->cbFile, pCurr->Off + pCurr->DataSeg.cbSeg);
61 RTFileSetSize(pEndpoint->File, pCurr->Off + pCurr->DataSeg.cbSeg);
62 }
63
64 rc = RTFileWriteAt(pEndpoint->File, pCurr->Off,
65 pCurr->DataSeg.pvSeg,
66 pCurr->DataSeg.cbSeg,
67 NULL);
68 }
69
70 break;
71 }
72 default:
73 AssertMsgFailed(("Invalid transfer type %d\n", pTasks->enmTransferType));
74 }
75
76 AssertRC(rc);
77
78 pCurr->pfnCompleted(pCurr, pCurr->pvUser);
79 pdmacFileTaskFree(pEndpoint, pCurr);
80 }
81
82 return rc;
83}
84
85/**
86 * A fallback method in case something goes wrong with the normal
87 * I/O manager.
88 */
89int pdmacFileAioMgrFailsafe(RTTHREAD ThreadSelf, void *pvUser)
90{
91 int rc = VINF_SUCCESS;
92 PPDMACEPFILEMGR pAioMgr = (PPDMACEPFILEMGR)pvUser;
93
94 while ( (pAioMgr->enmState == PDMACEPFILEMGRSTATE_RUNNING)
95 || (pAioMgr->enmState == PDMACEPFILEMGRSTATE_SUSPENDING))
96 {
97 if (!ASMAtomicReadBool(&pAioMgr->fWokenUp))
98 {
99 ASMAtomicWriteBool(&pAioMgr->fWaitingEventSem, true);
100 rc = RTSemEventWait(pAioMgr->EventSem, RT_INDEFINITE_WAIT);
101 ASMAtomicWriteBool(&pAioMgr->fWaitingEventSem, false);
102 AssertRC(rc);
103 }
104 ASMAtomicXchgBool(&pAioMgr->fWokenUp, false);
105
106 /* Process endpoint events first. */
107 PPDMASYNCCOMPLETIONENDPOINTFILE pEndpoint = pAioMgr->pEndpointsHead;
108 while (pEndpoint)
109 {
110 rc = pdmacFileAioMgrFailsafeProcessEndpoint(pEndpoint);
111 AssertRC(rc);
112 pEndpoint = pEndpoint->AioMgr.pEndpointNext;
113 }
114
115 /* Now check for an external blocking event. */
116 if (pAioMgr->fBlockingEventPending)
117 {
118 switch (pAioMgr->enmBlockingEvent)
119 {
120 case PDMACEPFILEAIOMGRBLOCKINGEVENT_ADD_ENDPOINT:
121 {
122 PPDMASYNCCOMPLETIONENDPOINTFILE pEndpointNew = pAioMgr->BlockingEventData.AddEndpoint.pEndpoint;
123 AssertMsg(VALID_PTR(pEndpointNew), ("Adding endpoint event without a endpoint to add\n"));
124
125 pEndpointNew->enmState = PDMASYNCCOMPLETIONENDPOINTFILESTATE_ACTIVE;
126
127 pEndpointNew->AioMgr.pEndpointNext = pAioMgr->pEndpointsHead;
128 pEndpointNew->AioMgr.pEndpointPrev = NULL;
129 if (pAioMgr->pEndpointsHead)
130 pAioMgr->pEndpointsHead->AioMgr.pEndpointPrev = pEndpointNew;
131 pAioMgr->pEndpointsHead = pEndpointNew;
132 break;
133 }
134 case PDMACEPFILEAIOMGRBLOCKINGEVENT_REMOVE_ENDPOINT:
135 {
136 PPDMASYNCCOMPLETIONENDPOINTFILE pEndpointRemove = pAioMgr->BlockingEventData.RemoveEndpoint.pEndpoint;
137 AssertMsg(VALID_PTR(pEndpointRemove), ("Removing endpoint event without a endpoint to remove\n"));
138
139 pEndpointRemove->enmState = PDMASYNCCOMPLETIONENDPOINTFILESTATE_REMOVING;
140
141 PPDMASYNCCOMPLETIONENDPOINTFILE pPrev = pEndpointRemove->AioMgr.pEndpointPrev;
142 PPDMASYNCCOMPLETIONENDPOINTFILE pNext = pEndpointRemove->AioMgr.pEndpointNext;
143
144 if (pPrev)
145 pPrev->AioMgr.pEndpointNext = pNext;
146 else
147 pAioMgr->pEndpointsHead = pNext;
148
149 if (pNext)
150 pNext->AioMgr.pEndpointPrev = pPrev;
151 break;
152 }
153 case PDMACEPFILEAIOMGRBLOCKINGEVENT_CLOSE_ENDPOINT:
154 {
155 PPDMASYNCCOMPLETIONENDPOINTFILE pEndpointClose = pAioMgr->BlockingEventData.CloseEndpoint.pEndpoint;
156 AssertMsg(VALID_PTR(pEndpointClose), ("Close endpoint event without a endpoint to Close\n"));
157
158 pEndpointClose->enmState = PDMASYNCCOMPLETIONENDPOINTFILESTATE_CLOSING;
159
160 /* Make sure all tasks finished. */
161 rc = pdmacFileAioMgrFailsafeProcessEndpoint(pEndpointClose);
162 AssertRC(rc);
163 break;
164 }
165 case PDMACEPFILEAIOMGRBLOCKINGEVENT_SHUTDOWN:
166 pAioMgr->enmState = PDMACEPFILEMGRSTATE_SHUTDOWN;
167 break;
168 case PDMACEPFILEAIOMGRBLOCKINGEVENT_SUSPEND:
169 pAioMgr->enmState = PDMACEPFILEMGRSTATE_SUSPENDING;
170 break;
171 case PDMACEPFILEAIOMGRBLOCKINGEVENT_RESUME:
172 pAioMgr->enmState = PDMACEPFILEMGRSTATE_RUNNING;
173 break;
174 default:
175 AssertMsgFailed(("Invalid event type %d\n", pAioMgr->enmBlockingEvent));
176 }
177
178 /* Release the waiting thread. */
179 rc = RTSemEventSignal(pAioMgr->EventSemBlock);
180 AssertRC(rc);
181 }
182 }
183
184 return rc;
185}
186
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