VirtualBox

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

Last change on this file since 22264 was 21806, checked in by vboxsync, 15 years ago

Storage/VBoxHDD: resurrect the facility to dump information about disk images, and bare minimum fix for creating diff images.

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