VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/DrvVD.cpp@ 20833

Last change on this file since 20833 was 20374, checked in by vboxsync, 16 years ago

*: s/RT_\(BEGIN|END\)_DECLS/RT_C_DECLS_\1/g

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 46.8 KB
Line 
1/* $Id: DrvVD.cpp 20374 2009-06-08 00:43:21Z vboxsync $ */
2/** @file
3 * DrvVD - Generic VBox disk media driver.
4 */
5
6/*
7 * Copyright (C) 2006-2008 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header files *
25*******************************************************************************/
26#define LOG_GROUP LOG_GROUP_DRV_VD
27#include <VBox/VBoxHDD.h>
28#include <VBox/pdmdrv.h>
29#include <VBox/pdmasynccompletion.h>
30#include <iprt/alloc.h>
31#include <iprt/assert.h>
32#include <iprt/uuid.h>
33#include <iprt/file.h>
34#include <iprt/string.h>
35#include <iprt/cache.h>
36#include <iprt/tcp.h>
37#include <iprt/semaphore.h>
38
39#ifdef VBOX_WITH_INIP
40/* All lwip header files are not C++ safe. So hack around this. */
41RT_C_DECLS_BEGIN
42#include <lwip/inet.h>
43#include <lwip/tcp.h>
44#include <lwip/sockets.h>
45RT_C_DECLS_END
46#endif /* VBOX_WITH_INIP */
47
48#include "Builtins.h"
49
50#ifdef VBOX_WITH_INIP
51/* Small hack to get at lwIP initialized status */
52extern bool DevINIPConfigured(void);
53#endif /* VBOX_WITH_INIP */
54
55
56/*******************************************************************************
57* Defined types, constants and macros *
58*******************************************************************************/
59
60/** Converts a pointer to VDIDISK::IMedia to a PVBOXDISK. */
61#define PDMIMEDIA_2_VBOXDISK(pInterface) \
62 ( (PVBOXDISK)((uintptr_t)pInterface - RT_OFFSETOF(VBOXDISK, IMedia)) )
63
64/** Converts a pointer to PDMDRVINS::IBase to a PPDMDRVINS. */
65#define PDMIBASE_2_DRVINS(pInterface) \
66 ( (PPDMDRVINS)((uintptr_t)pInterface - RT_OFFSETOF(PDMDRVINS, IBase)) )
67
68/** Converts a pointer to PDMDRVINS::IBase to a PVBOXDISK. */
69#define PDMIBASE_2_VBOXDISK(pInterface) \
70 ( PDMINS_2_DATA(PDMIBASE_2_DRVINS(pInterface), PVBOXDISK) )
71
72/** Converts a pointer to VBOXDISK::IMediaAsync to a PVBOXDISK. */
73#define PDMIMEDIAASYNC_2_VBOXDISK(pInterface) \
74 ( (PVBOXDISK)((uintptr_t)pInterface - RT_OFFSETOF(VBOXDISK, IMediaAsync)) )
75
76/**
77 * VBox disk container, image information, private part.
78 */
79
80typedef struct VBOXIMAGE
81{
82 /** Pointer to next image. */
83 struct VBOXIMAGE *pNext;
84 /** Pointer to list of VD interfaces. Per-image. */
85 PVDINTERFACE pVDIfsImage;
86 /** Common structure for the configuration information interface. */
87 VDINTERFACE VDIConfig;
88} VBOXIMAGE, *PVBOXIMAGE;
89
90/**
91 * Storage backend data.
92 */
93typedef struct DRVVDSTORAGEBACKEND
94{
95 /** PDM async completion end point. */
96 PPDMASYNCCOMPLETIONENDPOINT pEndpoint;
97 /** The template. */
98 PPDMASYNCCOMPLETIONTEMPLATE pTemplate;
99 /** Event semaphore for synchronous operations. */
100 RTSEMEVENT EventSem;
101 /** Flag whether a synchronous operation is currently pending. */
102 volatile bool fSyncIoPending;
103 /** Callback routine */
104 PFNVDCOMPLETED pfnCompleted;
105} DRVVDSTORAGEBACKEND, *PDRVVDSTORAGEBACKEND;
106
107/**
108 * VBox disk container media main structure, private part.
109 */
110typedef struct VBOXDISK
111{
112 /** The VBox disk container. */
113 PVBOXHDD pDisk;
114 /** The media interface. */
115 PDMIMEDIA IMedia;
116 /** Pointer to the driver instance. */
117 PPDMDRVINS pDrvIns;
118 /** Flag whether suspend has changed image open mode to read only. */
119 bool fTempReadOnly;
120 /** Flag whether to use the runtime (true) or startup error facility. */
121 bool fErrorUseRuntime;
122 /** Pointer to list of VD interfaces. Per-disk. */
123 PVDINTERFACE pVDIfsDisk;
124 /** Common structure for the supported error interface. */
125 VDINTERFACE VDIError;
126 /** Callback table for error interface. */
127 VDINTERFACEERROR VDIErrorCallbacks;
128 /** Common structure for the supported TCP network stack interface. */
129 VDINTERFACE VDITcpNet;
130 /** Callback table for TCP network stack interface. */
131 VDINTERFACETCPNET VDITcpNetCallbacks;
132 /** Common structure for the supported async I/O interface. */
133 VDINTERFACE VDIAsyncIO;
134 /** Callback table for async I/O interface. */
135 VDINTERFACEASYNCIO VDIAsyncIOCallbacks;
136 /** Callback table for the configuration information interface. */
137 VDINTERFACECONFIG VDIConfigCallbacks;
138 /** Flag whether opened disk suppports async I/O operations. */
139 bool fAsyncIOSupported;
140 /** The async media interface. */
141 PDMIMEDIAASYNC IMediaAsync;
142 /** The async media port interface above. */
143 PPDMIMEDIAASYNCPORT pDrvMediaAsyncPort;
144 /** Pointer to the list of data we need to keep per image. */
145 PVBOXIMAGE pImages;
146} VBOXDISK, *PVBOXDISK;
147
148/*******************************************************************************
149* Error reporting callback *
150*******************************************************************************/
151
152static void drvvdErrorCallback(void *pvUser, int rc, RT_SRC_POS_DECL,
153 const char *pszFormat, va_list va)
154{
155 PPDMDRVINS pDrvIns = (PPDMDRVINS)pvUser;
156 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
157 if (pThis->fErrorUseRuntime)
158 /* We must not pass VMSETRTERR_FLAGS_FATAL as it could lead to a
159 * deadlock: We are probably executed in a thread context != EMT
160 * and the EM thread would wait until every thread is suspended
161 * but we would wait for the EM thread ... */
162 pDrvIns->pDrvHlp->pfnVMSetRuntimeErrorV(pDrvIns, /* fFlags=*/ 0, "DrvVD", pszFormat, va);
163 else
164 pDrvIns->pDrvHlp->pfnVMSetErrorV(pDrvIns, rc, RT_SRC_POS_ARGS, pszFormat, va);
165}
166
167
168/**
169 * Internal: allocate new image descriptor and put it in the list
170 */
171static PVBOXIMAGE drvvdNewImage(PVBOXDISK pThis)
172{
173 AssertPtr(pThis);
174 PVBOXIMAGE pImage = (PVBOXIMAGE)RTMemAllocZ(sizeof(VBOXIMAGE));
175 if (pImage)
176 {
177 pImage->pVDIfsImage = NULL;
178 PVBOXIMAGE *pp = &pThis->pImages;
179 while (*pp != NULL)
180 pp = &(*pp)->pNext;
181 *pp = pImage;
182 pImage->pNext = NULL;
183 }
184
185 return pImage;
186}
187
188/**
189 * Internal: free the list of images descriptors.
190 */
191static void drvvdFreeImages(PVBOXDISK pThis)
192{
193 while (pThis->pImages != NULL)
194 {
195 PVBOXIMAGE p = pThis->pImages;
196 pThis->pImages = pThis->pImages->pNext;
197 RTMemFree(p);
198 }
199}
200
201/*******************************************************************************
202* VD Async I/O interface implementation *
203*******************************************************************************/
204
205#ifdef VBOX_WITH_PDM_ASYNC_COMPLETION
206
207static DECLCALLBACK(void) drvvdAsyncTaskCompleted(PPDMDRVINS pDrvIns, void *pvTemplateUser, void *pvUser)
208{
209 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
210 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pvTemplateUser;
211 int rc = VINF_VD_ASYNC_IO_FINISHED;
212 void *pvCallerUser = NULL;
213
214 if (pStorageBackend->fSyncIoPending)
215 {
216 pStorageBackend->fSyncIoPending;
217 RTSemEventSignal(pStorageBackend->EventSem);
218 }
219 else if (pStorageBackend->pfnCompleted)
220 rc = pStorageBackend->pfnCompleted(pvUser, &pvCallerUser);
221 else
222 pvCallerUser = pvUser;
223
224 if (rc == VINF_VD_ASYNC_IO_FINISHED)
225 {
226 rc = pThis->pDrvMediaAsyncPort->pfnTransferCompleteNotify(pThis->pDrvMediaAsyncPort, pvCallerUser);
227 AssertRC(rc);
228 }
229 else
230 AssertMsg(rc == VERR_VD_ASYNC_IO_IN_PROGRESS, ("Invalid return code from disk backend rc=%Rrc\n", rc));
231}
232
233static DECLCALLBACK(int) drvvdAsyncIOOpen(void *pvUser, const char *pszLocation, bool fReadonly,
234 PFNVDCOMPLETED pfnCompleted, void **ppStorage)
235{
236 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
237 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)RTMemAllocZ(sizeof(DRVVDSTORAGEBACKEND));
238 int rc = VINF_SUCCESS;
239
240 if (pStorageBackend)
241 {
242 pStorageBackend->fSyncIoPending = false;
243 pStorageBackend->pfnCompleted = pfnCompleted;
244
245 int rc = RTSemEventCreate(&pStorageBackend->EventSem);
246 if (RT_SUCCESS(rc))
247 {
248 rc = PDMDrvHlpPDMAsyncCompletionTemplateCreate(pDrvVD->pDrvIns, &pStorageBackend->pTemplate,
249 drvvdAsyncTaskCompleted, pStorageBackend, "AsyncTaskCompleted");
250 if (RT_SUCCESS(rc))
251 {
252 rc = PDMR3AsyncCompletionEpCreateForFile(&pStorageBackend->pEndpoint, pszLocation,
253 fReadonly
254 ? PDMACEP_FILE_FLAGS_READ_ONLY | PDMACEP_FILE_FLAGS_CACHING
255 : PDMACEP_FILE_FLAGS_CACHING,
256 pStorageBackend->pTemplate);
257 if (RT_SUCCESS(rc))
258 {
259 *ppStorage = pStorageBackend;
260 return VINF_SUCCESS;
261 }
262
263 PDMR3AsyncCompletionTemplateDestroy(pStorageBackend->pTemplate);
264 }
265 RTSemEventDestroy(pStorageBackend->EventSem);
266 }
267 RTMemFree(pStorageBackend);
268 }
269 else
270 rc = VERR_NO_MEMORY;
271
272 return rc;
273}
274
275static DECLCALLBACK(int) drvvdAsyncIOClose(void *pvUser, void *pStorage)
276{
277 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
278 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
279
280 PDMR3AsyncCompletionEpClose(pStorageBackend->pEndpoint);
281 PDMR3AsyncCompletionTemplateDestroy(pStorageBackend->pTemplate);
282 RTSemEventDestroy(pStorageBackend->EventSem);
283 RTMemFree(pStorageBackend);
284
285 return VINF_SUCCESS;;
286}
287
288static DECLCALLBACK(int) drvvdAsyncIOGetSize(void *pvUser, void *pStorage, uint64_t *pcbSize)
289{
290 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
291 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
292
293 return PDMR3AsyncCompletionEpGetSize(pStorageBackend->pEndpoint, pcbSize);
294}
295
296static DECLCALLBACK(int) drvvdAsyncIOReadSync(void *pvUser, void *pStorage, uint64_t uOffset,
297 size_t cbRead, void *pvBuf, size_t *pcbRead)
298{
299 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
300 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
301 PDMDATASEG DataSeg;
302 PPDMASYNCCOMPLETIONTASK pTask;
303
304 Assert(!pStorageBackend->fSyncIoPending);
305 pStorageBackend->fSyncIoPending = true;
306 DataSeg.cbSeg = cbRead;
307 DataSeg.pvSeg = pvBuf;
308
309 int rc = PDMR3AsyncCompletionEpRead(pStorageBackend->pEndpoint, uOffset, &DataSeg, 1, cbRead, NULL, &pTask);
310 if (RT_FAILURE(rc))
311 return rc;
312
313 /* Wait */
314 rc = RTSemEventWait(pStorageBackend->EventSem, RT_INDEFINITE_WAIT);
315 AssertRC(rc);
316
317 if (pcbRead)
318 *pcbRead = cbRead;
319
320 return VINF_SUCCESS;
321}
322
323static DECLCALLBACK(int) drvvdAsyncIOWriteSync(void *pvUser, void *pStorage, uint64_t uOffset,
324 size_t cbWrite, const void *pvBuf, size_t *pcbWritten)
325{
326 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
327 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
328 PDMDATASEG DataSeg;
329 PPDMASYNCCOMPLETIONTASK pTask;
330
331 Assert(!pStorageBackend->fSyncIoPending);
332 pStorageBackend->fSyncIoPending = true;
333 DataSeg.cbSeg = cbWrite;
334 DataSeg.pvSeg = (void *)pvBuf;
335
336 int rc = PDMR3AsyncCompletionEpWrite(pStorageBackend->pEndpoint, uOffset, &DataSeg, 1, cbWrite, NULL, &pTask);
337 if (RT_FAILURE(rc))
338 return rc;
339
340 /* Wait */
341 rc = RTSemEventWait(pStorageBackend->EventSem, RT_INDEFINITE_WAIT);
342 AssertRC(rc);
343
344 if (pcbWritten)
345 *pcbWritten = cbWrite;
346
347 return VINF_SUCCESS;
348}
349
350static DECLCALLBACK(int) drvvdAsyncIOFlushSync(void *pvUser, void *pStorage)
351{
352 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
353 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
354 PPDMASYNCCOMPLETIONTASK pTask;
355
356 Assert(!pStorageBackend->fSyncIoPending);
357 pStorageBackend->fSyncIoPending = true;
358
359 int rc = PDMR3AsyncCompletionEpFlush(pStorageBackend->pEndpoint, NULL, &pTask);
360 if (RT_FAILURE(rc))
361 return rc;
362
363 /* Wait */
364 rc = RTSemEventWait(pStorageBackend->EventSem, RT_INDEFINITE_WAIT);
365 AssertRC(rc);
366
367 return VINF_SUCCESS;
368}
369
370static DECLCALLBACK(int) drvvdAsyncIOReadAsync(void *pvUser, void *pStorage, uint64_t uOffset,
371 PCPDMDATASEG paSegments, size_t cSegments,
372 size_t cbRead, void *pvCompletion,
373 void **ppTask)
374{
375 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
376 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
377
378 return PDMR3AsyncCompletionEpRead(pStorageBackend->pEndpoint, uOffset, paSegments, cSegments, cbRead,
379 pvCompletion, (PPPDMASYNCCOMPLETIONTASK)ppTask);
380}
381
382static DECLCALLBACK(int) drvvdAsyncIOWriteAsync(void *pvUser, void *pStorage, uint64_t uOffset,
383 PCPDMDATASEG paSegments, size_t cSegments,
384 size_t cbWrite, void *pvCompletion,
385 void **ppTask)
386{
387 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
388 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
389
390 return PDMR3AsyncCompletionEpWrite(pStorageBackend->pEndpoint, uOffset, paSegments, cSegments, cbWrite,
391 pvCompletion, (PPPDMASYNCCOMPLETIONTASK)ppTask);
392}
393
394static DECLCALLBACK(int) drvvdAsyncIOFlushAsync(void *pvUser, void *pStorage,
395 void *pvCompletion, void **ppTask)
396{
397 PVBOXDISK pDrvVD = (PVBOXDISK)pvUser;
398 PDRVVDSTORAGEBACKEND pStorageBackend = (PDRVVDSTORAGEBACKEND)pStorage;
399
400 return PDMR3AsyncCompletionEpFlush(pStorageBackend->pEndpoint, pvCompletion,
401 (PPPDMASYNCCOMPLETIONTASK)ppTask);
402}
403
404#endif /* VBOX_WITH_PDM_ASYNC_COMPLETION */
405
406
407/*******************************************************************************
408* VD Configuration interface implementation *
409*******************************************************************************/
410
411static bool drvvdCfgAreKeysValid(void *pvUser, const char *pszzValid)
412{
413 return CFGMR3AreValuesValid((PCFGMNODE)pvUser, pszzValid);
414}
415
416static int drvvdCfgQuerySize(void *pvUser, const char *pszName, size_t *pcb)
417{
418 return CFGMR3QuerySize((PCFGMNODE)pvUser, pszName, pcb);
419}
420
421static int drvvdCfgQuery(void *pvUser, const char *pszName, char *pszString, size_t cchString)
422{
423 return CFGMR3QueryString((PCFGMNODE)pvUser, pszName, pszString, cchString);
424}
425
426
427#ifdef VBOX_WITH_INIP
428/*******************************************************************************
429* VD TCP network stack interface implementation - INIP case *
430*******************************************************************************/
431
432/** @copydoc VDINTERFACETCPNET::pfnClientConnect */
433static DECLCALLBACK(int) drvvdINIPClientConnect(const char *pszAddress, uint32_t uPort, PRTSOCKET pSock)
434{
435 int rc = VINF_SUCCESS;
436 /* First check whether lwIP is set up in this VM instance. */
437 if (!DevINIPConfigured())
438 {
439 LogRelFunc(("no IP stack\n"));
440 return VERR_NET_HOST_UNREACHABLE;
441 }
442 /* Resolve hostname. As there is no standard resolver for lwIP yet,
443 * just accept numeric IP addresses for now. */
444 struct in_addr ip;
445 if (!lwip_inet_aton(pszAddress, &ip))
446 {
447 LogRelFunc(("cannot resolve IP %s\n", pszAddress));
448 return VERR_NET_HOST_UNREACHABLE;
449 }
450 /* Create socket and connect. */
451 RTSOCKET Sock = lwip_socket(PF_INET, SOCK_STREAM, 0);
452 if (Sock != -1)
453 {
454 struct sockaddr_in InAddr = {0};
455 InAddr.sin_family = AF_INET;
456 InAddr.sin_port = htons(uPort);
457 InAddr.sin_addr = ip;
458 if (!lwip_connect(Sock, (struct sockaddr *)&InAddr, sizeof(InAddr)))
459 {
460 *pSock = Sock;
461 return VINF_SUCCESS;
462 }
463 rc = VERR_NET_CONNECTION_REFUSED; /* @todo real solution needed */
464 lwip_close(Sock);
465 }
466 else
467 rc = VERR_NET_CONNECTION_REFUSED; /* @todo real solution needed */
468 return rc;
469}
470
471/** @copydoc VDINTERFACETCPNET::pfnClientClose */
472static DECLCALLBACK(int) drvvdINIPClientClose(RTSOCKET Sock)
473{
474 lwip_close(Sock);
475 return VINF_SUCCESS; /** @todo real solution needed */
476}
477
478/** @copydoc VDINTERFACETCPNET::pfnSelectOne */
479static DECLCALLBACK(int) drvvdINIPSelectOne(RTSOCKET Sock, unsigned cMillies)
480{
481 fd_set fdsetR;
482 FD_ZERO(&fdsetR);
483 FD_SET(Sock, &fdsetR);
484 fd_set fdsetE = fdsetR;
485
486 int rc;
487 if (cMillies == RT_INDEFINITE_WAIT)
488 rc = lwip_select(Sock + 1, &fdsetR, NULL, &fdsetE, NULL);
489 else
490 {
491 struct timeval timeout;
492 timeout.tv_sec = cMillies / 1000;
493 timeout.tv_usec = (cMillies % 1000) * 1000;
494 rc = lwip_select(Sock + 1, &fdsetR, NULL, &fdsetE, &timeout);
495 }
496 if (rc > 0)
497 return VINF_SUCCESS;
498 if (rc == 0)
499 return VERR_TIMEOUT;
500 return VERR_NET_CONNECTION_REFUSED; /** @todo real solution needed */
501}
502
503/** @copydoc VDINTERFACETCPNET::pfnRead */
504static DECLCALLBACK(int) drvvdINIPRead(RTSOCKET Sock, void *pvBuffer, size_t cbBuffer, size_t *pcbRead)
505{
506 /* Do params checking */
507 if (!pvBuffer || !cbBuffer)
508 {
509 AssertMsgFailed(("Invalid params\n"));
510 return VERR_INVALID_PARAMETER;
511 }
512
513 /*
514 * Read loop.
515 * If pcbRead is NULL we have to fill the entire buffer!
516 */
517 size_t cbRead = 0;
518 size_t cbToRead = cbBuffer;
519 for (;;)
520 {
521 /** @todo this clipping here is just in case (the send function
522 * needed it, so I added it here, too). Didn't investigate if this
523 * really has issues. Better be safe than sorry. */
524 ssize_t cbBytesRead = lwip_recv(Sock, (char *)pvBuffer + cbRead,
525 RT_MIN(cbToRead, 32768), 0);
526 if (cbBytesRead < 0)
527 return VERR_NET_CONNECTION_REFUSED; /** @todo real solution */
528 if (cbBytesRead == 0 && errno)
529 return VERR_NET_CONNECTION_REFUSED; /** @todo real solution */
530 if (pcbRead)
531 {
532 /* return partial data */
533 *pcbRead = cbBytesRead;
534 break;
535 }
536
537 /* read more? */
538 cbRead += cbBytesRead;
539 if (cbRead == cbBuffer)
540 break;
541
542 /* next */
543 cbToRead = cbBuffer - cbRead;
544 }
545
546 return VINF_SUCCESS;
547}
548
549/** @copydoc VDINTERFACETCPNET::pfnWrite */
550static DECLCALLBACK(int) drvvdINIPWrite(RTSOCKET Sock, const void *pvBuffer, size_t cbBuffer)
551{
552 do
553 {
554 /** @todo lwip send only supports up to 65535 bytes in a single
555 * send (stupid limitation buried in the code), so make sure we
556 * don't get any wraparounds. This should be moved to DevINIP
557 * stack interface once that's implemented. */
558 ssize_t cbWritten = lwip_send(Sock, (void *)pvBuffer,
559 RT_MIN(cbBuffer, 32768), 0);
560 if (cbWritten < 0)
561 return VERR_NET_CONNECTION_REFUSED; /** @todo real solution needed */
562 AssertMsg(cbBuffer >= (size_t)cbWritten, ("Wrote more than we requested!!! cbWritten=%d cbBuffer=%d\n",
563 cbWritten, cbBuffer));
564 cbBuffer -= cbWritten;
565 pvBuffer = (const char *)pvBuffer + cbWritten;
566 } while (cbBuffer);
567
568 return VINF_SUCCESS;
569}
570
571/** @copydoc VDINTERFACETCPNET::pfnFlush */
572static DECLCALLBACK(int) drvvdINIPFlush(RTSOCKET Sock)
573{
574 int fFlag = 1;
575 lwip_setsockopt(Sock, IPPROTO_TCP, TCP_NODELAY,
576 (const char *)&fFlag, sizeof(fFlag));
577 fFlag = 0;
578 lwip_setsockopt(Sock, IPPROTO_TCP, TCP_NODELAY,
579 (const char *)&fFlag, sizeof(fFlag));
580 return VINF_SUCCESS;
581}
582#endif /* VBOX_WITH_INIP */
583
584
585/*******************************************************************************
586* Media interface methods *
587*******************************************************************************/
588
589/** @copydoc PDMIMEDIA::pfnRead */
590static DECLCALLBACK(int) drvvdRead(PPDMIMEDIA pInterface,
591 uint64_t off, void *pvBuf, size_t cbRead)
592{
593 LogFlow(("%s: off=%#llx pvBuf=%p cbRead=%d\n", __FUNCTION__,
594 off, pvBuf, cbRead));
595 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
596 int rc = VDRead(pThis->pDisk, off, pvBuf, cbRead);
597 if (RT_SUCCESS(rc))
598 Log2(("%s: off=%#llx pvBuf=%p cbRead=%d %.*Rhxd\n", __FUNCTION__,
599 off, pvBuf, cbRead, cbRead, pvBuf));
600 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
601 return rc;
602}
603
604/** @copydoc PDMIMEDIA::pfnWrite */
605static DECLCALLBACK(int) drvvdWrite(PPDMIMEDIA pInterface,
606 uint64_t off, const void *pvBuf,
607 size_t cbWrite)
608{
609 LogFlow(("%s: off=%#llx pvBuf=%p cbWrite=%d\n", __FUNCTION__,
610 off, pvBuf, cbWrite));
611 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
612 Log2(("%s: off=%#llx pvBuf=%p cbWrite=%d %.*Rhxd\n", __FUNCTION__,
613 off, pvBuf, cbWrite, cbWrite, pvBuf));
614 int rc = VDWrite(pThis->pDisk, off, pvBuf, cbWrite);
615 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
616 return rc;
617}
618
619/** @copydoc PDMIMEDIA::pfnFlush */
620static DECLCALLBACK(int) drvvdFlush(PPDMIMEDIA pInterface)
621{
622 LogFlow(("%s:\n", __FUNCTION__));
623 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
624 int rc = VDFlush(pThis->pDisk);
625 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
626 return rc;
627}
628
629/** @copydoc PDMIMEDIA::pfnGetSize */
630static DECLCALLBACK(uint64_t) drvvdGetSize(PPDMIMEDIA pInterface)
631{
632 LogFlow(("%s:\n", __FUNCTION__));
633 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
634 uint64_t cb = VDGetSize(pThis->pDisk, VD_LAST_IMAGE);
635 LogFlow(("%s: returns %#llx (%llu)\n", __FUNCTION__, cb, cb));
636 return cb;
637}
638
639/** @copydoc PDMIMEDIA::pfnIsReadOnly */
640static DECLCALLBACK(bool) drvvdIsReadOnly(PPDMIMEDIA pInterface)
641{
642 LogFlow(("%s:\n", __FUNCTION__));
643 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
644 bool f = VDIsReadOnly(pThis->pDisk);
645 LogFlow(("%s: returns %d\n", __FUNCTION__, f));
646 return f;
647}
648
649/** @copydoc PDMIMEDIA::pfnBiosGetPCHSGeometry */
650static DECLCALLBACK(int) drvvdBiosGetPCHSGeometry(PPDMIMEDIA pInterface,
651 PPDMMEDIAGEOMETRY pPCHSGeometry)
652{
653 LogFlow(("%s:\n", __FUNCTION__));
654 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
655 int rc = VDGetPCHSGeometry(pThis->pDisk, VD_LAST_IMAGE, pPCHSGeometry);
656 if (RT_FAILURE(rc))
657 {
658 Log(("%s: geometry not available.\n", __FUNCTION__));
659 rc = VERR_PDM_GEOMETRY_NOT_SET;
660 }
661 LogFlow(("%s: returns %Rrc (CHS=%d/%d/%d)\n", __FUNCTION__,
662 rc, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
663 return rc;
664}
665
666/** @copydoc PDMIMEDIA::pfnBiosSetPCHSGeometry */
667static DECLCALLBACK(int) drvvdBiosSetPCHSGeometry(PPDMIMEDIA pInterface,
668 PCPDMMEDIAGEOMETRY pPCHSGeometry)
669{
670 LogFlow(("%s: CHS=%d/%d/%d\n", __FUNCTION__,
671 pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
672 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
673 int rc = VDSetPCHSGeometry(pThis->pDisk, VD_LAST_IMAGE, pPCHSGeometry);
674 if (rc == VERR_VD_GEOMETRY_NOT_SET)
675 rc = VERR_PDM_GEOMETRY_NOT_SET;
676 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
677 return rc;
678}
679
680/** @copydoc PDMIMEDIA::pfnBiosGetLCHSGeometry */
681static DECLCALLBACK(int) drvvdBiosGetLCHSGeometry(PPDMIMEDIA pInterface,
682 PPDMMEDIAGEOMETRY pLCHSGeometry)
683{
684 LogFlow(("%s:\n", __FUNCTION__));
685 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
686 int rc = VDGetLCHSGeometry(pThis->pDisk, VD_LAST_IMAGE, pLCHSGeometry);
687 if (RT_FAILURE(rc))
688 {
689 Log(("%s: geometry not available.\n", __FUNCTION__));
690 rc = VERR_PDM_GEOMETRY_NOT_SET;
691 }
692 LogFlow(("%s: returns %Rrc (CHS=%d/%d/%d)\n", __FUNCTION__,
693 rc, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
694 return rc;
695}
696
697/** @copydoc PDMIMEDIA::pfnBiosSetLCHSGeometry */
698static DECLCALLBACK(int) drvvdBiosSetLCHSGeometry(PPDMIMEDIA pInterface,
699 PCPDMMEDIAGEOMETRY pLCHSGeometry)
700{
701 LogFlow(("%s: CHS=%d/%d/%d\n", __FUNCTION__,
702 pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
703 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
704 int rc = VDSetLCHSGeometry(pThis->pDisk, VD_LAST_IMAGE, pLCHSGeometry);
705 if (rc == VERR_VD_GEOMETRY_NOT_SET)
706 rc = VERR_PDM_GEOMETRY_NOT_SET;
707 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
708 return rc;
709}
710
711/** @copydoc PDMIMEDIA::pfnGetUuid */
712static DECLCALLBACK(int) drvvdGetUuid(PPDMIMEDIA pInterface, PRTUUID pUuid)
713{
714 LogFlow(("%s:\n", __FUNCTION__));
715 PVBOXDISK pThis = PDMIMEDIA_2_VBOXDISK(pInterface);
716 int rc = VDGetUuid(pThis->pDisk, 0, pUuid);
717 LogFlow(("%s: returns %Rrc ({%RTuuid})\n", __FUNCTION__, rc, pUuid));
718 return rc;
719}
720
721/*******************************************************************************
722* Async Media interface methods *
723*******************************************************************************/
724
725static DECLCALLBACK(int) drvvdStartRead(PPDMIMEDIAASYNC pInterface, uint64_t uOffset,
726 PPDMDATASEG paSeg, unsigned cSeg,
727 size_t cbRead, void *pvUser)
728{
729 LogFlow(("%s: uOffset=%#llx paSeg=%#p cSeg=%u cbRead=%d\n pvUser=%#p", __FUNCTION__,
730 uOffset, paSeg, cSeg, cbRead, pvUser));
731 PVBOXDISK pThis = PDMIMEDIAASYNC_2_VBOXDISK(pInterface);
732 int rc = VDAsyncRead(pThis->pDisk, uOffset, cbRead, paSeg, cSeg, pvUser);
733 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
734 return rc;
735}
736
737static DECLCALLBACK(int) drvvdStartWrite(PPDMIMEDIAASYNC pInterface, uint64_t uOffset,
738 PPDMDATASEG paSeg, unsigned cSeg,
739 size_t cbWrite, void *pvUser)
740{
741 LogFlow(("%s: uOffset=%#llx paSeg=%#p cSeg=%u cbWrite=%d\n pvUser=%#p", __FUNCTION__,
742 uOffset, paSeg, cSeg, cbWrite, pvUser));
743 PVBOXDISK pThis = PDMIMEDIAASYNC_2_VBOXDISK(pInterface);
744 int rc = VDAsyncWrite(pThis->pDisk, uOffset, cbWrite, paSeg, cSeg, pvUser);
745 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
746 return rc;
747}
748
749/*******************************************************************************
750* Async transport port interface methods *
751*******************************************************************************/
752
753static DECLCALLBACK(int) drvvdTasksCompleteNotify(PPDMDRVINS pDrvIns, void *pvUser)
754{
755 return VERR_NOT_IMPLEMENTED;
756}
757
758
759/*******************************************************************************
760* Base interface methods *
761*******************************************************************************/
762
763/** @copydoc PDMIBASE::pfnQueryInterface */
764static DECLCALLBACK(void *) drvvdQueryInterface(PPDMIBASE pInterface,
765 PDMINTERFACE enmInterface)
766{
767 PPDMDRVINS pDrvIns = PDMIBASE_2_DRVINS(pInterface);
768 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
769 switch (enmInterface)
770 {
771 case PDMINTERFACE_BASE:
772 return &pDrvIns->IBase;
773 case PDMINTERFACE_MEDIA:
774 return &pThis->IMedia;
775 case PDMINTERFACE_MEDIA_ASYNC:
776 return pThis->fAsyncIOSupported ? &pThis->IMediaAsync : NULL;
777 default:
778 return NULL;
779 }
780}
781
782
783/*******************************************************************************
784* Driver methods *
785*******************************************************************************/
786
787
788/**
789 * Construct a VBox disk media driver instance.
790 *
791 * @returns VBox status.
792 * @param pDrvIns The driver instance data.
793 * If the registration structure is needed, pDrvIns->pDrvReg points to it.
794 * @param pCfgHandle Configuration node handle for the driver. Use this to obtain the configuration
795 * of the driver instance. It's also found in pDrvIns->pCfgHandle as it's expected
796 * to be used frequently in this function.
797 */
798static DECLCALLBACK(int) drvvdConstruct(PPDMDRVINS pDrvIns,
799 PCFGMNODE pCfgHandle)
800{
801 LogFlow(("%s:\n", __FUNCTION__));
802 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
803 int rc = VINF_SUCCESS;
804 char *pszName = NULL; /**< The path of the disk image file. */
805 char *pszFormat = NULL; /**< The format backed to use for this image. */
806 bool fReadOnly; /**< True if the media is readonly. */
807 bool fHonorZeroWrites; /**< True if zero blocks should be written. */
808
809 /*
810 * Init the static parts.
811 */
812 pDrvIns->IBase.pfnQueryInterface = drvvdQueryInterface;
813 pThis->pDrvIns = pDrvIns;
814 pThis->fTempReadOnly = false;
815 pThis->pDisk = NULL;
816 pThis->fAsyncIOSupported = false;
817
818 /* IMedia */
819 pThis->IMedia.pfnRead = drvvdRead;
820 pThis->IMedia.pfnWrite = drvvdWrite;
821 pThis->IMedia.pfnFlush = drvvdFlush;
822 pThis->IMedia.pfnGetSize = drvvdGetSize;
823 pThis->IMedia.pfnIsReadOnly = drvvdIsReadOnly;
824 pThis->IMedia.pfnBiosGetPCHSGeometry = drvvdBiosGetPCHSGeometry;
825 pThis->IMedia.pfnBiosSetPCHSGeometry = drvvdBiosSetPCHSGeometry;
826 pThis->IMedia.pfnBiosGetLCHSGeometry = drvvdBiosGetLCHSGeometry;
827 pThis->IMedia.pfnBiosSetLCHSGeometry = drvvdBiosSetLCHSGeometry;
828 pThis->IMedia.pfnGetUuid = drvvdGetUuid;
829
830 /* IMediaAsync */
831 pThis->IMediaAsync.pfnStartRead = drvvdStartRead;
832 pThis->IMediaAsync.pfnStartWrite = drvvdStartWrite;
833
834 /* Initialize supported VD interfaces. */
835 pThis->pVDIfsDisk = NULL;
836
837 pThis->VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
838 pThis->VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
839 pThis->VDIErrorCallbacks.pfnError = drvvdErrorCallback;
840
841 rc = VDInterfaceAdd(&pThis->VDIError, "DrvVD_VDIError", VDINTERFACETYPE_ERROR,
842 &pThis->VDIErrorCallbacks, pDrvIns, &pThis->pVDIfsDisk);
843 AssertRC(rc);
844
845#ifdef VBOX_WITH_PDM_ASYNC_COMPLETION
846 pThis->VDIAsyncIOCallbacks.cbSize = sizeof(VDINTERFACEASYNCIO);
847 pThis->VDIAsyncIOCallbacks.enmInterface = VDINTERFACETYPE_ASYNCIO;
848 pThis->VDIAsyncIOCallbacks.pfnOpen = drvvdAsyncIOOpen;
849 pThis->VDIAsyncIOCallbacks.pfnClose = drvvdAsyncIOClose;
850 pThis->VDIAsyncIOCallbacks.pfnGetSize = drvvdAsyncIOGetSize;
851 pThis->VDIAsyncIOCallbacks.pfnReadSync = drvvdAsyncIOReadSync;
852 pThis->VDIAsyncIOCallbacks.pfnWriteSync = drvvdAsyncIOWriteSync;
853 pThis->VDIAsyncIOCallbacks.pfnFlushSync = drvvdAsyncIOFlushSync;
854 pThis->VDIAsyncIOCallbacks.pfnReadAsync = drvvdAsyncIOReadAsync;
855 pThis->VDIAsyncIOCallbacks.pfnWriteAsync = drvvdAsyncIOWriteAsync;
856 pThis->VDIAsyncIOCallbacks.pfnFlushAsync = drvvdAsyncIOFlushAsync;
857
858 rc = VDInterfaceAdd(&pThis->VDIAsyncIO, "DrvVD_AsyncIO", VDINTERFACETYPE_ASYNCIO,
859 &pThis->VDIAsyncIOCallbacks, pThis, &pThis->pVDIfsDisk);
860 AssertRC(rc);
861#endif
862
863 /* This is just prepared here, the actual interface is per-image, so it's
864 * added later. No need to have separate callback tables. */
865 pThis->VDIConfigCallbacks.cbSize = sizeof(VDINTERFACECONFIG);
866 pThis->VDIConfigCallbacks.enmInterface = VDINTERFACETYPE_CONFIG;
867 pThis->VDIConfigCallbacks.pfnAreKeysValid = drvvdCfgAreKeysValid;
868 pThis->VDIConfigCallbacks.pfnQuerySize = drvvdCfgQuerySize;
869 pThis->VDIConfigCallbacks.pfnQuery = drvvdCfgQuery;
870
871 /* List of images is empty now. */
872 pThis->pImages = NULL;
873
874 /* Try to attach async media port interface above.*/
875 pThis->pDrvMediaAsyncPort = (PPDMIMEDIAASYNCPORT)pDrvIns->pUpBase->pfnQueryInterface(pDrvIns->pUpBase, PDMINTERFACE_MEDIA_ASYNC_PORT);
876
877 /*
878 * Attach the async transport driver below if the device above us implements the
879 * async interface.
880 */
881 if (pThis->pDrvMediaAsyncPort)
882 {
883 /* Try to attach the driver. */
884 PPDMIBASE pBase;
885
886 rc = pDrvIns->pDrvHlp->pfnAttach(pDrvIns, &pBase);
887 if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
888 {
889 /*
890 * Though the device supports async I/O there is no transport driver
891 * which processes async requests.
892 * Revert to non async I/O.
893 */
894 rc = VINF_SUCCESS;
895 pThis->pDrvMediaAsyncPort = NULL;
896 pThis->fAsyncIOSupported = false;
897 }
898 else if (RT_FAILURE(rc))
899 {
900 AssertMsgFailed(("Failed to attach async transport driver below rc=%Rrc\n", rc));
901 }
902 else
903 {
904 /*
905 * The device supports async I/O and we successfully attached the transport driver.
906 * Indicate that async I/O is supported for now as we check if the image backend supports
907 * it later.
908 */
909 pThis->fAsyncIOSupported = true;
910
911 /** @todo: Use PDM async completion manager */
912 }
913 }
914
915 /*
916 * Validate configuration and find all parent images.
917 * It's sort of up side down from the image dependency tree.
918 */
919 bool fHostIP = false;
920 unsigned iLevel = 0;
921 PCFGMNODE pCurNode = pCfgHandle;
922
923 for (;;)
924 {
925 bool fValid;
926
927 if (pCurNode == pCfgHandle)
928 {
929 /* Toplevel configuration additionally contains the global image
930 * open flags. Some might be converted to per-image flags later. */
931 fValid = CFGMR3AreValuesValid(pCurNode,
932 "Format\0Path\0"
933 "ReadOnly\0HonorZeroWrites\0"
934 "HostIPStack\0");
935 }
936 else
937 {
938 /* All other image configurations only contain image name and
939 * the format information. */
940 fValid = CFGMR3AreValuesValid(pCurNode, "Format\0Path\0");
941 }
942 if (!fValid)
943 {
944 rc = PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES,
945 RT_SRC_POS, N_("DrvVD: Configuration error: keys incorrect at level %d"), iLevel);
946 break;
947 }
948
949 if (pCurNode == pCfgHandle)
950 {
951 rc = CFGMR3QueryBool(pCurNode, "HostIPStack", &fHostIP);
952 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
953 {
954 fHostIP = true;
955 rc = VINF_SUCCESS;
956 }
957 else if (RT_FAILURE(rc))
958 {
959 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
960 N_("DrvVD: Configuration error: Querying \"HostIPStack\" as boolean failed"));
961 break;
962 }
963
964 rc = CFGMR3QueryBool(pCurNode, "HonorZeroWrites", &fHonorZeroWrites);
965 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
966 {
967 fHonorZeroWrites = false;
968 rc = VINF_SUCCESS;
969 }
970 else if (RT_FAILURE(rc))
971 {
972 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
973 N_("DrvVD: Configuration error: Querying \"HonorZeroWrites\" as boolean failed"));
974 break;
975 }
976
977 rc = CFGMR3QueryBool(pCurNode, "ReadOnly", &fReadOnly);
978 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
979 {
980 fReadOnly = false;
981 rc = VINF_SUCCESS;
982 }
983 else if (RT_FAILURE(rc))
984 {
985 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
986 N_("DrvVD: Configuration error: Querying \"ReadOnly\" as boolean failed"));
987 break;
988 }
989 }
990
991 PCFGMNODE pParent = CFGMR3GetChild(pCurNode, "Parent");
992 if (!pParent)
993 break;
994 pCurNode = pParent;
995 iLevel++;
996 }
997
998 /*
999 * Open the images.
1000 */
1001 if (RT_SUCCESS(rc))
1002 {
1003 /* First of all figure out what kind of TCP networking stack interface
1004 * to use. This is done unconditionally, as backends which don't need
1005 * it will just ignore it. */
1006 if (fHostIP)
1007 {
1008 pThis->VDITcpNetCallbacks.cbSize = sizeof(VDINTERFACETCPNET);
1009 pThis->VDITcpNetCallbacks.enmInterface = VDINTERFACETYPE_TCPNET;
1010 pThis->VDITcpNetCallbacks.pfnClientConnect = RTTcpClientConnect;
1011 pThis->VDITcpNetCallbacks.pfnClientClose = RTTcpClientClose;
1012 pThis->VDITcpNetCallbacks.pfnSelectOne = RTTcpSelectOne;
1013 pThis->VDITcpNetCallbacks.pfnRead = RTTcpRead;
1014 pThis->VDITcpNetCallbacks.pfnWrite = RTTcpWrite;
1015 pThis->VDITcpNetCallbacks.pfnFlush = RTTcpFlush;
1016 }
1017 else
1018 {
1019#ifndef VBOX_WITH_INIP
1020 rc = PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES,
1021 RT_SRC_POS, N_("DrvVD: Configuration error: TCP over Internal Networking not compiled in"));
1022#else /* VBOX_WITH_INIP */
1023 pThis->VDITcpNetCallbacks.cbSize = sizeof(VDINTERFACETCPNET);
1024 pThis->VDITcpNetCallbacks.enmInterface = VDINTERFACETYPE_TCPNET;
1025 pThis->VDITcpNetCallbacks.pfnClientConnect = drvvdINIPClientConnect;
1026 pThis->VDITcpNetCallbacks.pfnClientClose = drvvdINIPClientClose;
1027 pThis->VDITcpNetCallbacks.pfnSelectOne = drvvdINIPSelectOne;
1028 pThis->VDITcpNetCallbacks.pfnRead = drvvdINIPRead;
1029 pThis->VDITcpNetCallbacks.pfnWrite = drvvdINIPWrite;
1030 pThis->VDITcpNetCallbacks.pfnFlush = drvvdINIPFlush;
1031#endif /* VBOX_WITH_INIP */
1032 }
1033 if (RT_SUCCESS(rc))
1034 {
1035 rc = VDInterfaceAdd(&pThis->VDITcpNet, "DrvVD_INIP",
1036 VDINTERFACETYPE_TCPNET,
1037 &pThis->VDITcpNetCallbacks, NULL,
1038 &pThis->pVDIfsDisk);
1039 }
1040 if (RT_SUCCESS(rc))
1041 {
1042 rc = VDCreate(pThis->pVDIfsDisk, &pThis->pDisk);
1043 /* Error message is already set correctly. */
1044 }
1045 }
1046
1047 while (pCurNode && RT_SUCCESS(rc))
1048 {
1049 /* Allocate per-image data. */
1050 PVBOXIMAGE pImage = drvvdNewImage(pThis);
1051 if (!pImage)
1052 {
1053 rc = VERR_NO_MEMORY;
1054 break;
1055 }
1056
1057 /*
1058 * Read the image configuration.
1059 */
1060 rc = CFGMR3QueryStringAlloc(pCurNode, "Path", &pszName);
1061 if (RT_FAILURE(rc))
1062 {
1063 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
1064 N_("DrvVD: Configuration error: Querying \"Path\" as string failed"));
1065 break;
1066 }
1067
1068 rc = CFGMR3QueryStringAlloc(pCurNode, "Format", &pszFormat);
1069 if (RT_FAILURE(rc))
1070 {
1071 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
1072 N_("DrvVD: Configuration error: Querying \"Format\" as string failed"));
1073 break;
1074 }
1075
1076 PCFGMNODE pCfg = CFGMR3GetChild(pCurNode, "VDConfig");
1077 rc = VDInterfaceAdd(&pImage->VDIConfig, "DrvVD_Config", VDINTERFACETYPE_CONFIG,
1078 &pThis->VDIConfigCallbacks, pCfg, &pImage->pVDIfsImage);
1079 AssertRC(rc);
1080
1081 /*
1082 * Open the image.
1083 */
1084 unsigned uOpenFlags;
1085 if (fReadOnly || iLevel != 0)
1086 uOpenFlags = VD_OPEN_FLAGS_READONLY;
1087 else
1088 uOpenFlags = VD_OPEN_FLAGS_NORMAL;
1089 if (fHonorZeroWrites)
1090 uOpenFlags |= VD_OPEN_FLAGS_HONOR_ZEROES;
1091 if (pThis->pDrvMediaAsyncPort)
1092 uOpenFlags |= VD_OPEN_FLAGS_ASYNC_IO;
1093
1094 /* Try to open backend in asyc I/O mode first. */
1095 rc = VDOpen(pThis->pDisk, pszFormat, pszName, uOpenFlags, pImage->pVDIfsImage);
1096 if (rc == VERR_NOT_SUPPORTED)
1097 {
1098 /* Seems async I/O is not supported by the backend, open in normal mode. */
1099 uOpenFlags &= ~VD_OPEN_FLAGS_ASYNC_IO;
1100 rc = VDOpen(pThis->pDisk, pszFormat, pszName, uOpenFlags, pImage->pVDIfsImage);
1101 }
1102
1103 if (RT_SUCCESS(rc))
1104 {
1105 Log(("%s: %d - Opened '%s' in %s mode\n", __FUNCTION__,
1106 iLevel, pszName,
1107 VDIsReadOnly(pThis->pDisk) ? "read-only" : "read-write"));
1108 if ( VDIsReadOnly(pThis->pDisk)
1109 && !fReadOnly
1110 && iLevel == 0)
1111 {
1112 rc = PDMDrvHlpVMSetError(pDrvIns, VERR_VD_IMAGE_READ_ONLY, RT_SRC_POS,
1113 N_("Failed to open image '%s' for writing due to wrong "
1114 "permissions"), pszName);
1115 break;
1116 }
1117 }
1118 else
1119 {
1120 rc = PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1121 N_("Failed to open image '%s' in %s mode rc=%Rrc"), pszName,
1122 (uOpenFlags & VD_OPEN_FLAGS_READONLY) ? "readonly" : "read-write", rc);
1123 break;
1124 }
1125
1126
1127 MMR3HeapFree(pszName);
1128 pszName = NULL;
1129 MMR3HeapFree(pszFormat);
1130 pszFormat = NULL;
1131
1132 /* next */
1133 iLevel--;
1134 pCurNode = CFGMR3GetParent(pCurNode);
1135 }
1136
1137 if (RT_FAILURE(rc))
1138 {
1139 if (VALID_PTR(pThis->pDisk))
1140 {
1141 VDDestroy(pThis->pDisk);
1142 pThis->pDisk = NULL;
1143 }
1144 drvvdFreeImages(pThis);
1145 if (VALID_PTR(pszName))
1146 MMR3HeapFree(pszName);
1147 if (VALID_PTR(pszFormat))
1148 MMR3HeapFree(pszFormat);
1149
1150 return rc;
1151 }
1152 else
1153 {
1154 /*
1155 * Check if every opened image supports async I/O.
1156 * If not we revert to non async I/O.
1157 */
1158 if (pThis->fAsyncIOSupported)
1159 {
1160 for (unsigned i = 0; i < VDGetCount(pThis->pDisk); i++)
1161 {
1162 VDBACKENDINFO vdBackendInfo;
1163
1164 rc = VDBackendInfoSingle(pThis->pDisk, i, &vdBackendInfo);
1165 AssertRC(rc);
1166
1167 if (vdBackendInfo.uBackendCaps & VD_CAP_ASYNC)
1168 {
1169 /*
1170 * Backend indicates support for at least some files.
1171 * Check if current file is supported with async I/O)
1172 */
1173 rc = VDImageIsAsyncIOSupported(pThis->pDisk, i, &pThis->fAsyncIOSupported);
1174 AssertRC(rc);
1175
1176 /*
1177 * Check if current image is supported.
1178 * If not we can stop checking because
1179 * at least one does not support it.
1180 */
1181 if (!pThis->fAsyncIOSupported)
1182 break;
1183 }
1184 else
1185 {
1186 pThis->fAsyncIOSupported = false;
1187 break;
1188 }
1189 }
1190 }
1191
1192 /* Switch to runtime error facility. */
1193 pThis->fErrorUseRuntime = true;
1194 }
1195
1196 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
1197 return rc;
1198}
1199
1200/**
1201 * Destruct a driver instance.
1202 *
1203 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
1204 * resources can be freed correctly.
1205 *
1206 * @param pDrvIns The driver instance data.
1207 */
1208static DECLCALLBACK(void) drvvdDestruct(PPDMDRVINS pDrvIns)
1209{
1210 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1211 LogFlow(("%s:\n", __FUNCTION__));
1212
1213 drvvdFreeImages(pThis);
1214}
1215
1216
1217/**
1218 * When the VM has been suspended we'll change the image mode to read-only
1219 * so that main and others can read the VDIs. This is important when
1220 * saving state and so forth.
1221 *
1222 * @param pDrvIns The driver instance data.
1223 */
1224static DECLCALLBACK(void) drvvdSuspend(PPDMDRVINS pDrvIns)
1225{
1226 LogFlow(("%s:\n", __FUNCTION__));
1227 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1228 if (!VDIsReadOnly(pThis->pDisk))
1229 {
1230 unsigned uOpenFlags;
1231 int rc = VDGetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, &uOpenFlags);
1232 AssertRC(rc);
1233 uOpenFlags |= VD_OPEN_FLAGS_READONLY;
1234 rc = VDSetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, uOpenFlags);
1235 AssertRC(rc);
1236 pThis->fTempReadOnly = true;
1237 }
1238}
1239
1240/**
1241 * Before the VM resumes we'll have to undo the read-only mode change
1242 * done in drvvdSuspend.
1243 *
1244 * @param pDrvIns The driver instance data.
1245 */
1246static DECLCALLBACK(void) drvvdResume(PPDMDRVINS pDrvIns)
1247{
1248 LogFlow(("%s:\n", __FUNCTION__));
1249 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1250 if (pThis->fTempReadOnly)
1251 {
1252 unsigned uOpenFlags;
1253 int rc = VDGetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, &uOpenFlags);
1254 AssertRC(rc);
1255 uOpenFlags &= ~VD_OPEN_FLAGS_READONLY;
1256 rc = VDSetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, uOpenFlags);
1257 AssertRC(rc);
1258 pThis->fTempReadOnly = false;
1259 }
1260}
1261
1262static DECLCALLBACK(void) drvvdPowerOff(PPDMDRVINS pDrvIns)
1263{
1264 LogFlow(("%s:\n", __FUNCTION__));
1265 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1266
1267 /*
1268 * We must close the disk here to ensure that
1269 * the backend closes all files before the
1270 * async transport driver is destructed.
1271 */
1272 int rc = VDCloseAll(pThis->pDisk);
1273 AssertRC(rc);
1274}
1275
1276/**
1277 * VBox disk container media driver registration record.
1278 */
1279const PDMDRVREG g_DrvVD =
1280{
1281 /* u32Version */
1282 PDM_DRVREG_VERSION,
1283 /* szDriverName */
1284 "VD",
1285 /* pszDescription */
1286 "Generic VBox disk media driver.",
1287 /* fFlags */
1288 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1289 /* fClass. */
1290 PDM_DRVREG_CLASS_MEDIA,
1291 /* cMaxInstances */
1292 ~0,
1293 /* cbInstance */
1294 sizeof(VBOXDISK),
1295 /* pfnConstruct */
1296 drvvdConstruct,
1297 /* pfnDestruct */
1298 drvvdDestruct,
1299 /* pfnIOCtl */
1300 NULL,
1301 /* pfnPowerOn */
1302 NULL,
1303 /* pfnReset */
1304 NULL,
1305 /* pfnSuspend */
1306 drvvdSuspend,
1307 /* pfnResume */
1308 drvvdResume,
1309 /* pfnDetach */
1310 NULL,
1311 /* pfnPowerOff */
1312 drvvdPowerOff
1313};
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