VirtualBox

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

Last change on this file since 25963 was 25893, checked in by vboxsync, 15 years ago

PDMDrv,*: multi context drivers, part 2.

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