VirtualBox

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

Last change on this file since 22793 was 22277, checked in by vboxsync, 15 years ago

PDMDRVREG change (big changeset).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 46.6 KB
Line 
1/* $Id: DrvVD.cpp 22277 2009-08-16 21:12:50Z 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 * @copydoc FNPDMDRVCONSTRUCT
792 */
793static DECLCALLBACK(int) drvvdConstruct(PPDMDRVINS pDrvIns,
794 PCFGMNODE pCfgHandle,
795 uint32_t fFlags)
796{
797 LogFlow(("%s:\n", __FUNCTION__));
798 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
799 int rc = VINF_SUCCESS;
800 char *pszName = NULL; /**< The path of the disk image file. */
801 char *pszFormat = NULL; /**< The format backed to use for this image. */
802 bool fReadOnly; /**< True if the media is readonly. */
803 bool fHonorZeroWrites; /**< True if zero blocks should be written. */
804
805 /*
806 * Init the static parts.
807 */
808 pDrvIns->IBase.pfnQueryInterface = drvvdQueryInterface;
809 pThis->pDrvIns = pDrvIns;
810 pThis->fTempReadOnly = false;
811 pThis->pDisk = NULL;
812 pThis->fAsyncIOSupported = false;
813
814 /* IMedia */
815 pThis->IMedia.pfnRead = drvvdRead;
816 pThis->IMedia.pfnWrite = drvvdWrite;
817 pThis->IMedia.pfnFlush = drvvdFlush;
818 pThis->IMedia.pfnGetSize = drvvdGetSize;
819 pThis->IMedia.pfnIsReadOnly = drvvdIsReadOnly;
820 pThis->IMedia.pfnBiosGetPCHSGeometry = drvvdBiosGetPCHSGeometry;
821 pThis->IMedia.pfnBiosSetPCHSGeometry = drvvdBiosSetPCHSGeometry;
822 pThis->IMedia.pfnBiosGetLCHSGeometry = drvvdBiosGetLCHSGeometry;
823 pThis->IMedia.pfnBiosSetLCHSGeometry = drvvdBiosSetLCHSGeometry;
824 pThis->IMedia.pfnGetUuid = drvvdGetUuid;
825
826 /* IMediaAsync */
827 pThis->IMediaAsync.pfnStartRead = drvvdStartRead;
828 pThis->IMediaAsync.pfnStartWrite = drvvdStartWrite;
829
830 /* Initialize supported VD interfaces. */
831 pThis->pVDIfsDisk = NULL;
832
833 pThis->VDIErrorCallbacks.cbSize = sizeof(VDINTERFACEERROR);
834 pThis->VDIErrorCallbacks.enmInterface = VDINTERFACETYPE_ERROR;
835 pThis->VDIErrorCallbacks.pfnError = drvvdErrorCallback;
836 pThis->VDIErrorCallbacks.pfnMessage = NULL;
837
838 rc = VDInterfaceAdd(&pThis->VDIError, "DrvVD_VDIError", VDINTERFACETYPE_ERROR,
839 &pThis->VDIErrorCallbacks, pDrvIns, &pThis->pVDIfsDisk);
840 AssertRC(rc);
841
842#ifdef VBOX_WITH_PDM_ASYNC_COMPLETION
843 pThis->VDIAsyncIOCallbacks.cbSize = sizeof(VDINTERFACEASYNCIO);
844 pThis->VDIAsyncIOCallbacks.enmInterface = VDINTERFACETYPE_ASYNCIO;
845 pThis->VDIAsyncIOCallbacks.pfnOpen = drvvdAsyncIOOpen;
846 pThis->VDIAsyncIOCallbacks.pfnClose = drvvdAsyncIOClose;
847 pThis->VDIAsyncIOCallbacks.pfnGetSize = drvvdAsyncIOGetSize;
848 pThis->VDIAsyncIOCallbacks.pfnReadSync = drvvdAsyncIOReadSync;
849 pThis->VDIAsyncIOCallbacks.pfnWriteSync = drvvdAsyncIOWriteSync;
850 pThis->VDIAsyncIOCallbacks.pfnFlushSync = drvvdAsyncIOFlushSync;
851 pThis->VDIAsyncIOCallbacks.pfnReadAsync = drvvdAsyncIOReadAsync;
852 pThis->VDIAsyncIOCallbacks.pfnWriteAsync = drvvdAsyncIOWriteAsync;
853 pThis->VDIAsyncIOCallbacks.pfnFlushAsync = drvvdAsyncIOFlushAsync;
854
855 rc = VDInterfaceAdd(&pThis->VDIAsyncIO, "DrvVD_AsyncIO", VDINTERFACETYPE_ASYNCIO,
856 &pThis->VDIAsyncIOCallbacks, pThis, &pThis->pVDIfsDisk);
857 AssertRC(rc);
858#endif
859
860 /* This is just prepared here, the actual interface is per-image, so it's
861 * added later. No need to have separate callback tables. */
862 pThis->VDIConfigCallbacks.cbSize = sizeof(VDINTERFACECONFIG);
863 pThis->VDIConfigCallbacks.enmInterface = VDINTERFACETYPE_CONFIG;
864 pThis->VDIConfigCallbacks.pfnAreKeysValid = drvvdCfgAreKeysValid;
865 pThis->VDIConfigCallbacks.pfnQuerySize = drvvdCfgQuerySize;
866 pThis->VDIConfigCallbacks.pfnQuery = drvvdCfgQuery;
867
868 /* List of images is empty now. */
869 pThis->pImages = NULL;
870
871 /* Try to attach async media port interface above.*/
872 pThis->pDrvMediaAsyncPort = (PPDMIMEDIAASYNCPORT)pDrvIns->pUpBase->pfnQueryInterface(pDrvIns->pUpBase, PDMINTERFACE_MEDIA_ASYNC_PORT);
873
874 /*
875 * Attach the async transport driver below if the device above us implements the
876 * async interface.
877 */
878 if (pThis->pDrvMediaAsyncPort)
879 {
880 /* Try to attach the driver. */
881 PPDMIBASE pBase;
882
883 rc = PDMDrvHlpAttach(pDrvIns, fFlags, &pBase);
884 if (rc == VERR_PDM_NO_ATTACHED_DRIVER)
885 {
886 /*
887 * Though the device supports async I/O there is no transport driver
888 * which processes async requests.
889 * Revert to non async I/O.
890 */
891 rc = VINF_SUCCESS;
892 pThis->pDrvMediaAsyncPort = NULL;
893 pThis->fAsyncIOSupported = false;
894 }
895 else if (RT_FAILURE(rc))
896 {
897 AssertMsgFailed(("Failed to attach async transport driver below rc=%Rrc\n", rc));
898 }
899 else
900 {
901 /*
902 * The device supports async I/O and we successfully attached the transport driver.
903 * Indicate that async I/O is supported for now as we check if the image backend supports
904 * it later.
905 */
906 pThis->fAsyncIOSupported = true;
907
908 /** @todo: Use PDM async completion manager */
909 }
910 }
911
912 /*
913 * Validate configuration and find all parent images.
914 * It's sort of up side down from the image dependency tree.
915 */
916 bool fHostIP = false;
917 unsigned iLevel = 0;
918 PCFGMNODE pCurNode = pCfgHandle;
919
920 for (;;)
921 {
922 bool fValid;
923
924 if (pCurNode == pCfgHandle)
925 {
926 /* Toplevel configuration additionally contains the global image
927 * open flags. Some might be converted to per-image flags later. */
928 fValid = CFGMR3AreValuesValid(pCurNode,
929 "Format\0Path\0"
930 "ReadOnly\0HonorZeroWrites\0"
931 "HostIPStack\0");
932 }
933 else
934 {
935 /* All other image configurations only contain image name and
936 * the format information. */
937 fValid = CFGMR3AreValuesValid(pCurNode, "Format\0Path\0");
938 }
939 if (!fValid)
940 {
941 rc = PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES,
942 RT_SRC_POS, N_("DrvVD: Configuration error: keys incorrect at level %d"), iLevel);
943 break;
944 }
945
946 if (pCurNode == pCfgHandle)
947 {
948 rc = CFGMR3QueryBool(pCurNode, "HostIPStack", &fHostIP);
949 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
950 {
951 fHostIP = true;
952 rc = VINF_SUCCESS;
953 }
954 else if (RT_FAILURE(rc))
955 {
956 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
957 N_("DrvVD: Configuration error: Querying \"HostIPStack\" as boolean failed"));
958 break;
959 }
960
961 rc = CFGMR3QueryBool(pCurNode, "HonorZeroWrites", &fHonorZeroWrites);
962 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
963 {
964 fHonorZeroWrites = false;
965 rc = VINF_SUCCESS;
966 }
967 else if (RT_FAILURE(rc))
968 {
969 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
970 N_("DrvVD: Configuration error: Querying \"HonorZeroWrites\" as boolean failed"));
971 break;
972 }
973
974 rc = CFGMR3QueryBool(pCurNode, "ReadOnly", &fReadOnly);
975 if (rc == VERR_CFGM_VALUE_NOT_FOUND)
976 {
977 fReadOnly = false;
978 rc = VINF_SUCCESS;
979 }
980 else if (RT_FAILURE(rc))
981 {
982 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
983 N_("DrvVD: Configuration error: Querying \"ReadOnly\" as boolean failed"));
984 break;
985 }
986 }
987
988 PCFGMNODE pParent = CFGMR3GetChild(pCurNode, "Parent");
989 if (!pParent)
990 break;
991 pCurNode = pParent;
992 iLevel++;
993 }
994
995 /*
996 * Open the images.
997 */
998 if (RT_SUCCESS(rc))
999 {
1000 /* First of all figure out what kind of TCP networking stack interface
1001 * to use. This is done unconditionally, as backends which don't need
1002 * it will just ignore it. */
1003 if (fHostIP)
1004 {
1005 pThis->VDITcpNetCallbacks.cbSize = sizeof(VDINTERFACETCPNET);
1006 pThis->VDITcpNetCallbacks.enmInterface = VDINTERFACETYPE_TCPNET;
1007 pThis->VDITcpNetCallbacks.pfnClientConnect = RTTcpClientConnect;
1008 pThis->VDITcpNetCallbacks.pfnClientClose = RTTcpClientClose;
1009 pThis->VDITcpNetCallbacks.pfnSelectOne = RTTcpSelectOne;
1010 pThis->VDITcpNetCallbacks.pfnRead = RTTcpRead;
1011 pThis->VDITcpNetCallbacks.pfnWrite = RTTcpWrite;
1012 pThis->VDITcpNetCallbacks.pfnFlush = RTTcpFlush;
1013 }
1014 else
1015 {
1016#ifndef VBOX_WITH_INIP
1017 rc = PDMDrvHlpVMSetError(pDrvIns, VERR_PDM_DRVINS_UNKNOWN_CFG_VALUES,
1018 RT_SRC_POS, N_("DrvVD: Configuration error: TCP over Internal Networking not compiled in"));
1019#else /* VBOX_WITH_INIP */
1020 pThis->VDITcpNetCallbacks.cbSize = sizeof(VDINTERFACETCPNET);
1021 pThis->VDITcpNetCallbacks.enmInterface = VDINTERFACETYPE_TCPNET;
1022 pThis->VDITcpNetCallbacks.pfnClientConnect = drvvdINIPClientConnect;
1023 pThis->VDITcpNetCallbacks.pfnClientClose = drvvdINIPClientClose;
1024 pThis->VDITcpNetCallbacks.pfnSelectOne = drvvdINIPSelectOne;
1025 pThis->VDITcpNetCallbacks.pfnRead = drvvdINIPRead;
1026 pThis->VDITcpNetCallbacks.pfnWrite = drvvdINIPWrite;
1027 pThis->VDITcpNetCallbacks.pfnFlush = drvvdINIPFlush;
1028#endif /* VBOX_WITH_INIP */
1029 }
1030 if (RT_SUCCESS(rc))
1031 {
1032 rc = VDInterfaceAdd(&pThis->VDITcpNet, "DrvVD_INIP",
1033 VDINTERFACETYPE_TCPNET,
1034 &pThis->VDITcpNetCallbacks, NULL,
1035 &pThis->pVDIfsDisk);
1036 }
1037 if (RT_SUCCESS(rc))
1038 {
1039 rc = VDCreate(pThis->pVDIfsDisk, &pThis->pDisk);
1040 /* Error message is already set correctly. */
1041 }
1042 }
1043
1044 while (pCurNode && RT_SUCCESS(rc))
1045 {
1046 /* Allocate per-image data. */
1047 PVBOXIMAGE pImage = drvvdNewImage(pThis);
1048 if (!pImage)
1049 {
1050 rc = VERR_NO_MEMORY;
1051 break;
1052 }
1053
1054 /*
1055 * Read the image configuration.
1056 */
1057 rc = CFGMR3QueryStringAlloc(pCurNode, "Path", &pszName);
1058 if (RT_FAILURE(rc))
1059 {
1060 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
1061 N_("DrvVD: Configuration error: Querying \"Path\" as string failed"));
1062 break;
1063 }
1064
1065 rc = CFGMR3QueryStringAlloc(pCurNode, "Format", &pszFormat);
1066 if (RT_FAILURE(rc))
1067 {
1068 rc = PDMDRV_SET_ERROR(pDrvIns, rc,
1069 N_("DrvVD: Configuration error: Querying \"Format\" as string failed"));
1070 break;
1071 }
1072
1073 PCFGMNODE pCfg = CFGMR3GetChild(pCurNode, "VDConfig");
1074 rc = VDInterfaceAdd(&pImage->VDIConfig, "DrvVD_Config", VDINTERFACETYPE_CONFIG,
1075 &pThis->VDIConfigCallbacks, pCfg, &pImage->pVDIfsImage);
1076 AssertRC(rc);
1077
1078 /*
1079 * Open the image.
1080 */
1081 unsigned uOpenFlags;
1082 if (fReadOnly || iLevel != 0)
1083 uOpenFlags = VD_OPEN_FLAGS_READONLY;
1084 else
1085 uOpenFlags = VD_OPEN_FLAGS_NORMAL;
1086 if (fHonorZeroWrites)
1087 uOpenFlags |= VD_OPEN_FLAGS_HONOR_ZEROES;
1088 if (pThis->pDrvMediaAsyncPort)
1089 uOpenFlags |= VD_OPEN_FLAGS_ASYNC_IO;
1090
1091 /* Try to open backend in asyc I/O mode first. */
1092 rc = VDOpen(pThis->pDisk, pszFormat, pszName, uOpenFlags, pImage->pVDIfsImage);
1093 if (rc == VERR_NOT_SUPPORTED)
1094 {
1095 /* Seems async I/O is not supported by the backend, open in normal mode. */
1096 uOpenFlags &= ~VD_OPEN_FLAGS_ASYNC_IO;
1097 rc = VDOpen(pThis->pDisk, pszFormat, pszName, uOpenFlags, pImage->pVDIfsImage);
1098 }
1099
1100 if (RT_SUCCESS(rc))
1101 {
1102 Log(("%s: %d - Opened '%s' in %s mode\n", __FUNCTION__,
1103 iLevel, pszName,
1104 VDIsReadOnly(pThis->pDisk) ? "read-only" : "read-write"));
1105 if ( VDIsReadOnly(pThis->pDisk)
1106 && !fReadOnly
1107 && iLevel == 0)
1108 {
1109 rc = PDMDrvHlpVMSetError(pDrvIns, VERR_VD_IMAGE_READ_ONLY, RT_SRC_POS,
1110 N_("Failed to open image '%s' for writing due to wrong "
1111 "permissions"), pszName);
1112 break;
1113 }
1114 }
1115 else
1116 {
1117 rc = PDMDrvHlpVMSetError(pDrvIns, rc, RT_SRC_POS,
1118 N_("Failed to open image '%s' in %s mode rc=%Rrc"), pszName,
1119 (uOpenFlags & VD_OPEN_FLAGS_READONLY) ? "readonly" : "read-write", rc);
1120 break;
1121 }
1122
1123
1124 MMR3HeapFree(pszName);
1125 pszName = NULL;
1126 MMR3HeapFree(pszFormat);
1127 pszFormat = NULL;
1128
1129 /* next */
1130 iLevel--;
1131 pCurNode = CFGMR3GetParent(pCurNode);
1132 }
1133
1134 if (RT_FAILURE(rc))
1135 {
1136 if (VALID_PTR(pThis->pDisk))
1137 {
1138 VDDestroy(pThis->pDisk);
1139 pThis->pDisk = NULL;
1140 }
1141 drvvdFreeImages(pThis);
1142 if (VALID_PTR(pszName))
1143 MMR3HeapFree(pszName);
1144 if (VALID_PTR(pszFormat))
1145 MMR3HeapFree(pszFormat);
1146
1147 return rc;
1148 }
1149 else
1150 {
1151 /*
1152 * Check if every opened image supports async I/O.
1153 * If not we revert to non async I/O.
1154 */
1155 if (pThis->fAsyncIOSupported)
1156 {
1157 for (unsigned i = 0; i < VDGetCount(pThis->pDisk); i++)
1158 {
1159 VDBACKENDINFO vdBackendInfo;
1160
1161 rc = VDBackendInfoSingle(pThis->pDisk, i, &vdBackendInfo);
1162 AssertRC(rc);
1163
1164 if (vdBackendInfo.uBackendCaps & VD_CAP_ASYNC)
1165 {
1166 /*
1167 * Backend indicates support for at least some files.
1168 * Check if current file is supported with async I/O)
1169 */
1170 rc = VDImageIsAsyncIOSupported(pThis->pDisk, i, &pThis->fAsyncIOSupported);
1171 AssertRC(rc);
1172
1173 /*
1174 * Check if current image is supported.
1175 * If not we can stop checking because
1176 * at least one does not support it.
1177 */
1178 if (!pThis->fAsyncIOSupported)
1179 break;
1180 }
1181 else
1182 {
1183 pThis->fAsyncIOSupported = false;
1184 break;
1185 }
1186 }
1187 }
1188
1189 /* Switch to runtime error facility. */
1190 pThis->fErrorUseRuntime = true;
1191 }
1192
1193 LogFlow(("%s: returns %Rrc\n", __FUNCTION__, rc));
1194 return rc;
1195}
1196
1197/**
1198 * Destruct a driver instance.
1199 *
1200 * Most VM resources are freed by the VM. This callback is provided so that any non-VM
1201 * resources can be freed correctly.
1202 *
1203 * @param pDrvIns The driver instance data.
1204 */
1205static DECLCALLBACK(void) drvvdDestruct(PPDMDRVINS pDrvIns)
1206{
1207 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1208 LogFlow(("%s:\n", __FUNCTION__));
1209
1210 drvvdFreeImages(pThis);
1211}
1212
1213
1214/**
1215 * When the VM has been suspended we'll change the image mode to read-only
1216 * so that main and others can read the VDIs. This is important when
1217 * saving state and so forth.
1218 *
1219 * @param pDrvIns The driver instance data.
1220 */
1221static DECLCALLBACK(void) drvvdSuspend(PPDMDRVINS pDrvIns)
1222{
1223 LogFlow(("%s:\n", __FUNCTION__));
1224 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1225 if (!VDIsReadOnly(pThis->pDisk))
1226 {
1227 unsigned uOpenFlags;
1228 int rc = VDGetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, &uOpenFlags);
1229 AssertRC(rc);
1230 uOpenFlags |= VD_OPEN_FLAGS_READONLY;
1231 rc = VDSetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, uOpenFlags);
1232 AssertRC(rc);
1233 pThis->fTempReadOnly = true;
1234 }
1235}
1236
1237/**
1238 * Before the VM resumes we'll have to undo the read-only mode change
1239 * done in drvvdSuspend.
1240 *
1241 * @param pDrvIns The driver instance data.
1242 */
1243static DECLCALLBACK(void) drvvdResume(PPDMDRVINS pDrvIns)
1244{
1245 LogFlow(("%s:\n", __FUNCTION__));
1246 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1247 if (pThis->fTempReadOnly)
1248 {
1249 unsigned uOpenFlags;
1250 int rc = VDGetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, &uOpenFlags);
1251 AssertRC(rc);
1252 uOpenFlags &= ~VD_OPEN_FLAGS_READONLY;
1253 rc = VDSetOpenFlags(pThis->pDisk, VD_LAST_IMAGE, uOpenFlags);
1254 AssertRC(rc);
1255 pThis->fTempReadOnly = false;
1256 }
1257}
1258
1259static DECLCALLBACK(void) drvvdPowerOff(PPDMDRVINS pDrvIns)
1260{
1261 LogFlow(("%s:\n", __FUNCTION__));
1262 PVBOXDISK pThis = PDMINS_2_DATA(pDrvIns, PVBOXDISK);
1263
1264 /*
1265 * We must close the disk here to ensure that
1266 * the backend closes all files before the
1267 * async transport driver is destructed.
1268 */
1269 int rc = VDCloseAll(pThis->pDisk);
1270 AssertRC(rc);
1271}
1272
1273/**
1274 * VBox disk container media driver registration record.
1275 */
1276const PDMDRVREG g_DrvVD =
1277{
1278 /* u32Version */
1279 PDM_DRVREG_VERSION,
1280 /* szDriverName */
1281 "VD",
1282 /* pszDescription */
1283 "Generic VBox disk media driver.",
1284 /* fFlags */
1285 PDM_DRVREG_FLAGS_HOST_BITS_DEFAULT,
1286 /* fClass. */
1287 PDM_DRVREG_CLASS_MEDIA,
1288 /* cMaxInstances */
1289 ~0,
1290 /* cbInstance */
1291 sizeof(VBOXDISK),
1292 /* pfnConstruct */
1293 drvvdConstruct,
1294 /* pfnDestruct */
1295 drvvdDestruct,
1296 /* pfnIOCtl */
1297 NULL,
1298 /* pfnPowerOn */
1299 NULL,
1300 /* pfnReset */
1301 NULL,
1302 /* pfnSuspend */
1303 drvvdSuspend,
1304 /* pfnResume */
1305 drvvdResume,
1306 /* pfnAttach */
1307 NULL,
1308 /* pfnDetach */
1309 NULL,
1310 /* pfnPowerOff */
1311 drvvdPowerOff,
1312 /* pfnSoftReset */
1313 NULL,
1314 /* u32EndVersion */
1315 PDM_DRVREG_VERSION
1316};
1317
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