VirtualBox

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

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

VBoxHDD: introduce new function pointers to the TCP interface for getting the local and peer address of a socket

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