VirtualBox

source: vbox/trunk/include/VBox/vd-ifs.h@ 67247

Last change on this file since 67247 was 67247, checked in by vboxsync, 8 years ago

Main,VD: Compressed the Medium::i_exportFile method (VBOX_WITH_NEW_TAR_CREATOR) and hooked it up to the export progress object.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 62.0 KB
Line 
1/** @file
2 * VD Container API - interfaces.
3 */
4
5/*
6 * Copyright (C) 2011-2016 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___VBox_vd_ifs_h
27#define ___VBox_vd_ifs_h
28
29#include <iprt/assert.h>
30#include <iprt/string.h>
31#include <iprt/mem.h>
32#include <iprt/file.h>
33#include <iprt/net.h>
34#include <iprt/sg.h>
35#include <VBox/cdefs.h>
36#include <VBox/types.h>
37#include <VBox/err.h>
38
39RT_C_DECLS_BEGIN
40
41/** Interface header magic. */
42#define VDINTERFACE_MAGIC UINT32_C(0x19701015)
43
44/**
45 * Supported interface types.
46 */
47typedef enum VDINTERFACETYPE
48{
49 /** First valid interface. */
50 VDINTERFACETYPE_FIRST = 0,
51 /** Interface to pass error message to upper layers. Per-disk. */
52 VDINTERFACETYPE_ERROR = VDINTERFACETYPE_FIRST,
53 /** Interface for I/O operations. Per-image. */
54 VDINTERFACETYPE_IO,
55 /** Interface for progress notification. Per-operation. */
56 VDINTERFACETYPE_PROGRESS,
57 /** Interface for configuration information. Per-image. */
58 VDINTERFACETYPE_CONFIG,
59 /** Interface for TCP network stack. Per-image. */
60 VDINTERFACETYPE_TCPNET,
61 /** Interface for getting parent image state. Per-operation. */
62 VDINTERFACETYPE_PARENTSTATE,
63 /** Interface for synchronizing accesses from several threads. Per-disk. */
64 VDINTERFACETYPE_THREADSYNC,
65 /** Interface for I/O between the generic VBoxHDD code and the backend. Per-image (internal).
66 * This interface is completely internal and must not be used elsewhere. */
67 VDINTERFACETYPE_IOINT,
68 /** Interface to query the use of block ranges on the disk. Per-operation. */
69 VDINTERFACETYPE_QUERYRANGEUSE,
70 /** Interface for the metadata traverse callback. Per-operation. */
71 VDINTERFACETYPE_TRAVERSEMETADATA,
72 /** Interface for crypto operations. Per-filter. */
73 VDINTERFACETYPE_CRYPTO,
74 /** invalid interface. */
75 VDINTERFACETYPE_INVALID
76} VDINTERFACETYPE;
77
78/**
79 * Common structure for all interfaces and at the beginning of all types.
80 */
81typedef struct VDINTERFACE
82{
83 uint32_t u32Magic;
84 /** Human readable interface name. */
85 const char *pszInterfaceName;
86 /** Pointer to the next common interface structure. */
87 struct VDINTERFACE *pNext;
88 /** Interface type. */
89 VDINTERFACETYPE enmInterface;
90 /** Size of the interface. */
91 size_t cbSize;
92 /** Opaque user data which is passed on every call. */
93 void *pvUser;
94} VDINTERFACE;
95/** Pointer to a VDINTERFACE. */
96typedef VDINTERFACE *PVDINTERFACE;
97/** Pointer to a const VDINTERFACE. */
98typedef const VDINTERFACE *PCVDINTERFACE;
99
100/**
101 * Helper functions to handle interface lists.
102 *
103 * @note These interface lists are used consistently to pass per-disk,
104 * per-image and/or per-operation callbacks. Those three purposes are strictly
105 * separate. See the individual interface declarations for what context they
106 * apply to. The caller is responsible for ensuring that the lifetime of the
107 * interface descriptors is appropriate for the category of interface.
108 */
109
110/**
111 * Get a specific interface from a list of interfaces specified by the type.
112 *
113 * @return Pointer to the matching interface or NULL if none was found.
114 * @param pVDIfs Pointer to the VD interface list.
115 * @param enmInterface Interface to search for.
116 */
117DECLINLINE(PVDINTERFACE) VDInterfaceGet(PVDINTERFACE pVDIfs, VDINTERFACETYPE enmInterface)
118{
119 AssertMsgReturn( enmInterface >= VDINTERFACETYPE_FIRST
120 && enmInterface < VDINTERFACETYPE_INVALID,
121 ("enmInterface=%u", enmInterface), NULL);
122
123 while (pVDIfs)
124 {
125 AssertMsgBreak(pVDIfs->u32Magic == VDINTERFACE_MAGIC,
126 ("u32Magic=%#x\n", pVDIfs->u32Magic));
127
128 if (pVDIfs->enmInterface == enmInterface)
129 return pVDIfs;
130 pVDIfs = pVDIfs->pNext;
131 }
132
133 /* No matching interface was found. */
134 return NULL;
135}
136
137/**
138 * Add an interface to a list of interfaces.
139 *
140 * @return VBox status code.
141 * @param pInterface Pointer to an unitialized common interface structure.
142 * @param pszName Name of the interface.
143 * @param enmInterface Type of the interface.
144 * @param pvUser Opaque user data passed on every function call.
145 * @param cbInterface The interface size.
146 * @param ppVDIfs Pointer to the VD interface list.
147 */
148DECLINLINE(int) VDInterfaceAdd(PVDINTERFACE pInterface, const char *pszName, VDINTERFACETYPE enmInterface, void *pvUser,
149 size_t cbInterface, PVDINTERFACE *ppVDIfs)
150{
151 /* Argument checks. */
152 AssertMsgReturn( enmInterface >= VDINTERFACETYPE_FIRST
153 && enmInterface < VDINTERFACETYPE_INVALID,
154 ("enmInterface=%u", enmInterface), VERR_INVALID_PARAMETER);
155
156 AssertMsgReturn(VALID_PTR(ppVDIfs),
157 ("pInterfaceList=%#p", ppVDIfs),
158 VERR_INVALID_PARAMETER);
159
160 /* Fill out interface descriptor. */
161 pInterface->u32Magic = VDINTERFACE_MAGIC;
162 pInterface->cbSize = cbInterface;
163 pInterface->pszInterfaceName = pszName;
164 pInterface->enmInterface = enmInterface;
165 pInterface->pvUser = pvUser;
166 pInterface->pNext = *ppVDIfs;
167
168 /* Remember the new start of the list. */
169 *ppVDIfs = pInterface;
170
171 return VINF_SUCCESS;
172}
173
174/**
175 * Removes an interface from a list of interfaces.
176 *
177 * @return VBox status code
178 * @param pInterface Pointer to an initialized common interface structure to remove.
179 * @param ppVDIfs Pointer to the VD interface list to remove from.
180 */
181DECLINLINE(int) VDInterfaceRemove(PVDINTERFACE pInterface, PVDINTERFACE *ppVDIfs)
182{
183 int rc = VERR_NOT_FOUND;
184
185 /* Argument checks. */
186 AssertMsgReturn(VALID_PTR(pInterface),
187 ("pInterface=%#p", pInterface),
188 VERR_INVALID_PARAMETER);
189
190 AssertMsgReturn(VALID_PTR(ppVDIfs),
191 ("pInterfaceList=%#p", ppVDIfs),
192 VERR_INVALID_PARAMETER);
193
194 if (*ppVDIfs)
195 {
196 PVDINTERFACE pPrev = NULL;
197 PVDINTERFACE pCurr = *ppVDIfs;
198
199 while ( pCurr
200 && (pCurr != pInterface))
201 {
202 pPrev = pCurr;
203 pCurr = pCurr->pNext;
204 }
205
206 /* First interface */
207 if (!pPrev)
208 {
209 *ppVDIfs = pCurr->pNext;
210 rc = VINF_SUCCESS;
211 }
212 else if (pCurr)
213 {
214 pPrev = pCurr->pNext;
215 rc = VINF_SUCCESS;
216 }
217 }
218
219 return rc;
220}
221
222/**
223 * Interface to deliver error messages (and also informational messages)
224 * to upper layers.
225 *
226 * Per-disk interface. Optional, but think twice if you want to miss the
227 * opportunity of reporting better human-readable error messages.
228 */
229typedef struct VDINTERFACEERROR
230{
231 /**
232 * Common interface header.
233 */
234 VDINTERFACE Core;
235
236 /**
237 * Error message callback. Must be able to accept special IPRT format
238 * strings.
239 *
240 * @param pvUser The opaque data passed on container creation.
241 * @param rc The VBox error code.
242 * @param SRC_POS Use RT_SRC_POS.
243 * @param pszFormat Error message format string.
244 * @param va Error message arguments.
245 */
246 DECLR3CALLBACKMEMBER(void, pfnError, (void *pvUser, int rc, RT_SRC_POS_DECL,
247 const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(6, 0));
248
249 /**
250 * Informational message callback. May be NULL. Used e.g. in
251 * VDDumpImages(). Must be able to accept special IPRT format strings.
252 *
253 * @return VBox status code.
254 * @param pvUser The opaque data passed on container creation.
255 * @param pszFormat Message format string.
256 * @param va Message arguments.
257 */
258 DECLR3CALLBACKMEMBER(int, pfnMessage, (void *pvUser, const char *pszFormat, va_list va) RT_IPRT_FORMAT_ATTR(2, 0));
259
260} VDINTERFACEERROR, *PVDINTERFACEERROR;
261
262/**
263 * Get error interface from interface list.
264 *
265 * @return Pointer to the first error interface in the list.
266 * @param pVDIfs Pointer to the interface list.
267 */
268DECLINLINE(PVDINTERFACEERROR) VDIfErrorGet(PVDINTERFACE pVDIfs)
269{
270 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_ERROR);
271
272 /* Check that the interface descriptor is a progress interface. */
273 AssertMsgReturn( !pIf
274 || ( (pIf->enmInterface == VDINTERFACETYPE_ERROR)
275 && (pIf->cbSize == sizeof(VDINTERFACEERROR))),
276 ("Not an error interface\n"), NULL);
277
278 return (PVDINTERFACEERROR)pIf;
279}
280
281/**
282 * Signal an error to the frontend.
283 *
284 * @returns VBox status code.
285 * @param pIfError The error interface.
286 * @param rc The status code.
287 * @param SRC_POS The position in the source code.
288 * @param pszFormat The format string to pass.
289 * @param ... Arguments to the format string.
290 */
291DECLINLINE(int) RT_IPRT_FORMAT_ATTR(6, 7) vdIfError(PVDINTERFACEERROR pIfError, int rc, RT_SRC_POS_DECL,
292 const char *pszFormat, ...)
293{
294 va_list va;
295 va_start(va, pszFormat);
296 if (pIfError)
297 pIfError->pfnError(pIfError->Core.pvUser, rc, RT_SRC_POS_ARGS, pszFormat, va);
298 va_end(va);
299 return rc;
300}
301
302/**
303 * Signal an informational message to the frontend.
304 *
305 * @returns VBox status code.
306 * @param pIfError The error interface.
307 * @param pszFormat The format string to pass.
308 * @param ... Arguments to the format string.
309 */
310DECLINLINE(int) RT_IPRT_FORMAT_ATTR(2, 3) vdIfErrorMessage(PVDINTERFACEERROR pIfError, const char *pszFormat, ...)
311{
312 int rc = VINF_SUCCESS;
313 va_list va;
314 va_start(va, pszFormat);
315 if (pIfError && pIfError->pfnMessage)
316 rc = pIfError->pfnMessage(pIfError->Core.pvUser, pszFormat, va);
317 va_end(va);
318 return rc;
319}
320
321/**
322 * Completion callback which is called by the interface owner
323 * to inform the backend that a task finished.
324 *
325 * @return VBox status code.
326 * @param pvUser Opaque user data which is passed on request submission.
327 * @param rcReq Status code of the completed request.
328 */
329typedef DECLCALLBACK(int) FNVDCOMPLETED(void *pvUser, int rcReq);
330/** Pointer to FNVDCOMPLETED() */
331typedef FNVDCOMPLETED *PFNVDCOMPLETED;
332
333/**
334 * Support interface for I/O
335 *
336 * Per-image. Optional as input.
337 */
338typedef struct VDINTERFACEIO
339{
340 /**
341 * Common interface header.
342 */
343 VDINTERFACE Core;
344
345 /**
346 * Open callback
347 *
348 * @return VBox status code.
349 * @param pvUser The opaque data passed on container creation.
350 * @param pszLocation Name of the location to open.
351 * @param fOpen Flags for opening the backend.
352 * See RTFILE_O_* \#defines, inventing another set
353 * of open flags is not worth the mapping effort.
354 * @param pfnCompleted The callback which is called whenever a task
355 * completed. The backend has to pass the user data
356 * of the request initiator (ie the one who calls
357 * VDAsyncRead or VDAsyncWrite) in pvCompletion
358 * if this is NULL.
359 * @param ppvStorage Where to store the opaque storage handle.
360 */
361 DECLR3CALLBACKMEMBER(int, pfnOpen, (void *pvUser, const char *pszLocation,
362 uint32_t fOpen,
363 PFNVDCOMPLETED pfnCompleted,
364 void **ppvStorage));
365
366 /**
367 * Close callback.
368 *
369 * @return VBox status code.
370 * @param pvUser The opaque data passed on container creation.
371 * @param pvStorage The opaque storage handle to close.
372 */
373 DECLR3CALLBACKMEMBER(int, pfnClose, (void *pvUser, void *pvStorage));
374
375 /**
376 * Delete callback.
377 *
378 * @return VBox status code.
379 * @param pvUser The opaque data passed on container creation.
380 * @param pcszFilename Name of the file to delete.
381 */
382 DECLR3CALLBACKMEMBER(int, pfnDelete, (void *pvUser, const char *pcszFilename));
383
384 /**
385 * Move callback.
386 *
387 * @return VBox status code.
388 * @param pvUser The opaque data passed on container creation.
389 * @param pcszSrc The path to the source file.
390 * @param pcszDst The path to the destination file.
391 * This file will be created.
392 * @param fMove A combination of the RTFILEMOVE_* flags.
393 */
394 DECLR3CALLBACKMEMBER(int, pfnMove, (void *pvUser, const char *pcszSrc, const char *pcszDst, unsigned fMove));
395
396 /**
397 * Returns the free space on a disk.
398 *
399 * @return VBox status code.
400 * @param pvUser The opaque data passed on container creation.
401 * @param pcszFilename Name of a file to identify the disk.
402 * @param pcbFreeSpace Where to store the free space of the disk.
403 */
404 DECLR3CALLBACKMEMBER(int, pfnGetFreeSpace, (void *pvUser, const char *pcszFilename, int64_t *pcbFreeSpace));
405
406 /**
407 * Returns the last modification timestamp of a file.
408 *
409 * @return VBox status code.
410 * @param pvUser The opaque data passed on container creation.
411 * @param pcszFilename Name of a file to identify the disk.
412 * @param pModificationTime Where to store the timestamp of the file.
413 */
414 DECLR3CALLBACKMEMBER(int, pfnGetModificationTime, (void *pvUser, const char *pcszFilename, PRTTIMESPEC pModificationTime));
415
416 /**
417 * Returns the size of the opened storage backend.
418 *
419 * @return VBox status code.
420 * @param pvUser The opaque data passed on container creation.
421 * @param pvStorage The opaque storage handle to get the size from.
422 * @param pcb Where to store the size of the storage backend.
423 */
424 DECLR3CALLBACKMEMBER(int, pfnGetSize, (void *pvUser, void *pvStorage, uint64_t *pcb));
425
426 /**
427 * Sets the size of the opened storage backend if possible.
428 *
429 * @return VBox status code.
430 * @retval VERR_NOT_SUPPORTED if the backend does not support this operation.
431 * @param pvUser The opaque data passed on container creation.
432 * @param pvStorage The opaque storage handle to set the size for.
433 * @param cb The new size of the image.
434 *
435 * @note Depending on the host the underlying storage (backing file, etc.)
436 * might not have all required storage allocated (sparse file) which
437 * can delay writes or fail with a not enough free space error if there
438 * is not enough space on the storage medium when writing to the range for
439 * the first time.
440 * Use VDINTERFACEIO::pfnSetAllocationSize to make sure the storage is
441 * really alloacted.
442 */
443 DECLR3CALLBACKMEMBER(int, pfnSetSize, (void *pvUser, void *pvStorage, uint64_t cb));
444
445 /**
446 * Sets the size of the opened storage backend making sure the given size
447 * is really allocated.
448 *
449 * @return VBox status code.
450 * @retval VERR_NOT_SUPPORTED if the implementer of the interface doesn't support
451 * this method.
452 * @param pvUser The opaque data passed on container creation.
453 * @param pvStorage The storage handle.
454 * @param cbSize The new size of the image.
455 * @param fFlags Flags for controlling the allocation strategy.
456 * Reserved for future use, MBZ.
457 */
458 DECLR3CALLBACKMEMBER(int, pfnSetAllocationSize, (void *pvUser, void *pvStorage,
459 uint64_t cbSize, uint32_t fFlags));
460
461 /**
462 * Synchronous write callback.
463 *
464 * @return VBox status code.
465 * @param pvUser The opaque data passed on container creation.
466 * @param pvStorage The storage handle to use.
467 * @param off The offset to start from.
468 * @param pvBuf Pointer to the bits need to be written.
469 * @param cbToWrite How many bytes to write.
470 * @param pcbWritten Where to store how many bytes were actually written.
471 */
472 DECLR3CALLBACKMEMBER(int, pfnWriteSync, (void *pvUser, void *pvStorage, uint64_t off,
473 const void *pvBuf, size_t cbToWrite, size_t *pcbWritten));
474
475 /**
476 * Synchronous read callback.
477 *
478 * @return VBox status code.
479 * @param pvUser The opaque data passed on container creation.
480 * @param pvStorage The storage handle to use.
481 * @param off The offset to start from.
482 * @param pvBuf Where to store the read bits.
483 * @param cbToRead How many bytes to read.
484 * @param pcbRead Where to store how many bytes were actually read.
485 */
486 DECLR3CALLBACKMEMBER(int, pfnReadSync, (void *pvUser, void *pvStorage, uint64_t off,
487 void *pvBuf, size_t cbToRead, size_t *pcbRead));
488
489 /**
490 * Flush data to the storage backend.
491 *
492 * @return VBox status code.
493 * @param pvUser The opaque data passed on container creation.
494 * @param pvStorage The storage handle to flush.
495 */
496 DECLR3CALLBACKMEMBER(int, pfnFlushSync, (void *pvUser, void *pvStorage));
497
498 /**
499 * Initiate an asynchronous read request.
500 *
501 * @return VBox status code.
502 * @param pvUser The opaque user data passed on container creation.
503 * @param pvStorage The storage handle.
504 * @param uOffset The offset to start reading from.
505 * @param paSegments Scatter gather list to store the data in.
506 * @param cSegments Number of segments in the list.
507 * @param cbRead How many bytes to read.
508 * @param pvCompletion The opaque user data which is returned upon completion.
509 * @param ppTask Where to store the opaque task handle.
510 */
511 DECLR3CALLBACKMEMBER(int, pfnReadAsync, (void *pvUser, void *pvStorage, uint64_t uOffset,
512 PCRTSGSEG paSegments, size_t cSegments,
513 size_t cbRead, void *pvCompletion,
514 void **ppTask));
515
516 /**
517 * Initiate an asynchronous write request.
518 *
519 * @return VBox status code.
520 * @param pvUser The opaque user data passed on conatiner creation.
521 * @param pvStorage The storage handle.
522 * @param uOffset The offset to start writing to.
523 * @param paSegments Scatter gather list of the data to write
524 * @param cSegments Number of segments in the list.
525 * @param cbWrite How many bytes to write.
526 * @param pvCompletion The opaque user data which is returned upon completion.
527 * @param ppTask Where to store the opaque task handle.
528 */
529 DECLR3CALLBACKMEMBER(int, pfnWriteAsync, (void *pvUser, void *pvStorage, uint64_t uOffset,
530 PCRTSGSEG paSegments, size_t cSegments,
531 size_t cbWrite, void *pvCompletion,
532 void **ppTask));
533
534 /**
535 * Initiates an async flush request.
536 *
537 * @return VBox status code.
538 * @param pvUser The opaque data passed on container creation.
539 * @param pvStorage The storage handle to flush.
540 * @param pvCompletion The opaque user data which is returned upon completion.
541 * @param ppTask Where to store the opaque task handle.
542 */
543 DECLR3CALLBACKMEMBER(int, pfnFlushAsync, (void *pvUser, void *pvStorage,
544 void *pvCompletion, void **ppTask));
545
546} VDINTERFACEIO, *PVDINTERFACEIO;
547
548/**
549 * Get I/O interface from interface list.
550 *
551 * @return Pointer to the first I/O interface in the list.
552 * @param pVDIfs Pointer to the interface list.
553 */
554DECLINLINE(PVDINTERFACEIO) VDIfIoGet(PVDINTERFACE pVDIfs)
555{
556 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_IO);
557
558 /* Check that the interface descriptor is a progress interface. */
559 AssertMsgReturn( !pIf
560 || ( (pIf->enmInterface == VDINTERFACETYPE_IO)
561 && (pIf->cbSize == sizeof(VDINTERFACEIO))),
562 ("Not a I/O interface"), NULL);
563
564 return (PVDINTERFACEIO)pIf;
565}
566
567DECLINLINE(int) vdIfIoFileOpen(PVDINTERFACEIO pIfIo, const char *pszFilename,
568 uint32_t fOpen, PFNVDCOMPLETED pfnCompleted,
569 void **ppStorage)
570{
571 return pIfIo->pfnOpen(pIfIo->Core.pvUser, pszFilename, fOpen, pfnCompleted, ppStorage);
572}
573
574DECLINLINE(int) vdIfIoFileClose(PVDINTERFACEIO pIfIo, void *pStorage)
575{
576 return pIfIo->pfnClose(pIfIo->Core.pvUser, pStorage);
577}
578
579DECLINLINE(int) vdIfIoFileDelete(PVDINTERFACEIO pIfIo, const char *pszFilename)
580{
581 return pIfIo->pfnDelete(pIfIo->Core.pvUser, pszFilename);
582}
583
584DECLINLINE(int) vdIfIoFileMove(PVDINTERFACEIO pIfIo, const char *pszSrc,
585 const char *pszDst, unsigned fMove)
586{
587 return pIfIo->pfnMove(pIfIo->Core.pvUser, pszSrc, pszDst, fMove);
588}
589
590DECLINLINE(int) vdIfIoFileGetFreeSpace(PVDINTERFACEIO pIfIo, const char *pszFilename,
591 int64_t *pcbFree)
592{
593 return pIfIo->pfnGetFreeSpace(pIfIo->Core.pvUser, pszFilename, pcbFree);
594}
595
596DECLINLINE(int) vdIfIoFileGetModificationTime(PVDINTERFACEIO pIfIo, const char *pcszFilename,
597 PRTTIMESPEC pModificationTime)
598{
599 return pIfIo->pfnGetModificationTime(pIfIo->Core.pvUser, pcszFilename,
600 pModificationTime);
601}
602
603DECLINLINE(int) vdIfIoFileGetSize(PVDINTERFACEIO pIfIo, void *pStorage,
604 uint64_t *pcbSize)
605{
606 return pIfIo->pfnGetSize(pIfIo->Core.pvUser, pStorage, pcbSize);
607}
608
609DECLINLINE(int) vdIfIoFileSetSize(PVDINTERFACEIO pIfIo, void *pStorage,
610 uint64_t cbSize)
611{
612 return pIfIo->pfnSetSize(pIfIo->Core.pvUser, pStorage, cbSize);
613}
614
615DECLINLINE(int) vdIfIoFileWriteSync(PVDINTERFACEIO pIfIo, void *pStorage,
616 uint64_t uOffset, const void *pvBuffer, size_t cbBuffer,
617 size_t *pcbWritten)
618{
619 return pIfIo->pfnWriteSync(pIfIo->Core.pvUser, pStorage, uOffset,
620 pvBuffer, cbBuffer, pcbWritten);
621}
622
623DECLINLINE(int) vdIfIoFileReadSync(PVDINTERFACEIO pIfIo, void *pStorage,
624 uint64_t uOffset, void *pvBuffer, size_t cbBuffer,
625 size_t *pcbRead)
626{
627 return pIfIo->pfnReadSync(pIfIo->Core.pvUser, pStorage, uOffset,
628 pvBuffer, cbBuffer, pcbRead);
629}
630
631DECLINLINE(int) vdIfIoFileFlushSync(PVDINTERFACEIO pIfIo, void *pStorage)
632{
633 return pIfIo->pfnFlushSync(pIfIo->Core.pvUser, pStorage);
634}
635
636/**
637 * Create a VFS stream handle around a VD I/O interface.
638 *
639 * The I/O interface will not be closed or free by the stream, the caller will
640 * do so after it is done with the stream and has released the instances of the
641 * I/O stream object returned by this API.
642 *
643 * @return VBox status code.
644 * @param pVDIfsIo Pointer to the VD I/O interface.
645 * @param pvStorage The storage argument to pass to the interface
646 * methods.
647 * @param fFlags RTFILE_O_XXX, access mask requied.
648 * @param phVfsIos Where to return the VFS I/O stream handle on
649 * success.
650 */
651VBOXDDU_DECL(int) VDIfCreateVfsStream(PVDINTERFACEIO pVDIfsIo, void *pvStorage, uint32_t fFlags, PRTVFSIOSTREAM phVfsIos);
652
653/**
654 * Create a VFS file handle around a VD I/O interface.
655 *
656 * The I/O interface will not be closed or free by the VFS file, the caller will
657 * do so after it is done with the VFS file and has released the instances of
658 * the VFS object returned by this API.
659 *
660 * @return VBox status code.
661 * @param pVDIfs Pointer to the VD I/O interface. If NULL, then @a
662 * pVDIfsInt must be specified.
663 * @param pVDIfsInt Pointer to the internal VD I/O interface. If NULL,
664 * then @ pVDIfs must be specified.
665 * @param pvStorage The storage argument to pass to the interface
666 * methods.
667 * @param fFlags RTFILE_O_XXX, access mask requied.
668 * @param phVfsFile Where to return the VFS file handle on success.
669 */
670VBOXDDU_DECL(int) VDIfCreateVfsFile(PVDINTERFACEIO pVDIfs, struct VDINTERFACEIOINT *pVDIfsInt, void *pvStorage, uint32_t fFlags, PRTVFSFILE phVfsFile);
671
672/**
673 * Creates an VD I/O interface wrapper around an IPRT VFS I/O stream.
674 *
675 * @return VBox status code.
676 * @param hVfsIos The IPRT VFS I/O stream handle. The handle will be
677 * retained by the returned I/O interface (released on
678 * close or destruction).
679 * @param fAccessMode The access mode (RTFILE_O_ACCESS_MASK) to accept.
680 * @param ppIoIf Where to return the pointer to the VD I/O interface.
681 * This must be passed to VDIfDestroyFromVfsStream().
682 */
683VBOXDDU_DECL(int) VDIfCreateFromVfsStream(RTVFSIOSTREAM hVfsIos, uint32_t fAccessMode, PVDINTERFACEIO *ppIoIf);
684
685/**
686 * Destroys the VD I/O interface returned by VDIfCreateFromVfsStream.
687 *
688 * @returns VBox status code.
689 * @param pIoIf The I/O interface pointer returned by
690 * VDIfCreateFromVfsStream. NULL will be quietly
691 * ignored.
692 */
693VBOXDDU_DECL(int) VDIfDestroyFromVfsStream(PVDINTERFACEIO pIoIf);
694
695
696/**
697 * Callback which provides progress information about a currently running
698 * lengthy operation.
699 *
700 * @return VBox status code.
701 * @param pvUser The opaque user data associated with this interface.
702 * @param uPercentage Completion percentage.
703 */
704typedef DECLCALLBACK(int) FNVDPROGRESS(void *pvUser, unsigned uPercentage);
705/** Pointer to FNVDPROGRESS() */
706typedef FNVDPROGRESS *PFNVDPROGRESS;
707
708/**
709 * Progress notification interface
710 *
711 * Per-operation. Optional.
712 */
713typedef struct VDINTERFACEPROGRESS
714{
715 /**
716 * Common interface header.
717 */
718 VDINTERFACE Core;
719
720 /**
721 * Progress notification callbacks.
722 */
723 PFNVDPROGRESS pfnProgress;
724
725} VDINTERFACEPROGRESS, *PVDINTERFACEPROGRESS;
726
727/** Initializer for VDINTERFACEPROGRESS. */
728#define VDINTERFACEPROGRESS_INITALIZER(a_pfnProgress) { { 0, NULL, NULL, VDINTERFACETYPE_INVALID, 0, NULL }, a_pfnProgress }
729
730/**
731 * Get progress interface from interface list.
732 *
733 * @return Pointer to the first progress interface in the list.
734 * @param pVDIfs Pointer to the interface list.
735 */
736DECLINLINE(PVDINTERFACEPROGRESS) VDIfProgressGet(PVDINTERFACE pVDIfs)
737{
738 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_PROGRESS);
739
740 /* Check that the interface descriptor is a progress interface. */
741 AssertMsgReturn( !pIf
742 || ( (pIf->enmInterface == VDINTERFACETYPE_PROGRESS)
743 && (pIf->cbSize == sizeof(VDINTERFACEPROGRESS))),
744 ("Not a progress interface"), NULL);
745
746 return (PVDINTERFACEPROGRESS)pIf;
747}
748
749/**
750 * Signal new progress information to the frontend.
751 *
752 * @returns VBox status code.
753 * @param pIfProgress The progress interface.
754 * @param uPercentage Completion percentage.
755 */
756DECLINLINE(int) vdIfProgress(PVDINTERFACEPROGRESS pIfProgress, unsigned uPercentage)
757{
758 if (pIfProgress)
759 return pIfProgress->pfnProgress(pIfProgress->Core.pvUser, uPercentage);
760 return VINF_SUCCESS;
761}
762
763/**
764 * Configuration information interface
765 *
766 * Per-image. Optional for most backends, but mandatory for images which do
767 * not operate on files (including standard block or character devices).
768 */
769typedef struct VDINTERFACECONFIG
770{
771 /**
772 * Common interface header.
773 */
774 VDINTERFACE Core;
775
776 /**
777 * Validates that the keys are within a set of valid names.
778 *
779 * @return true if all key names are found in pszzAllowed.
780 * @return false if not.
781 * @param pvUser The opaque user data associated with this interface.
782 * @param pszzValid List of valid key names separated by '\\0' and ending with
783 * a double '\\0'.
784 */
785 DECLR3CALLBACKMEMBER(bool, pfnAreKeysValid, (void *pvUser, const char *pszzValid));
786
787 /**
788 * Retrieves the length of the string value associated with a key (including
789 * the terminator, for compatibility with CFGMR3QuerySize).
790 *
791 * @return VBox status code.
792 * VERR_CFGM_VALUE_NOT_FOUND means that the key is not known.
793 * @param pvUser The opaque user data associated with this interface.
794 * @param pszName Name of the key to query.
795 * @param pcbValue Where to store the value length. Non-NULL.
796 */
797 DECLR3CALLBACKMEMBER(int, pfnQuerySize, (void *pvUser, const char *pszName, size_t *pcbValue));
798
799 /**
800 * Query the string value associated with a key.
801 *
802 * @return VBox status code.
803 * VERR_CFGM_VALUE_NOT_FOUND means that the key is not known.
804 * VERR_CFGM_NOT_ENOUGH_SPACE means that the buffer is not big enough.
805 * @param pvUser The opaque user data associated with this interface.
806 * @param pszName Name of the key to query.
807 * @param pszValue Pointer to buffer where to store value.
808 * @param cchValue Length of value buffer.
809 */
810 DECLR3CALLBACKMEMBER(int, pfnQuery, (void *pvUser, const char *pszName, char *pszValue, size_t cchValue));
811
812 /**
813 * Query the bytes value associated with a key.
814 *
815 * @return VBox status code.
816 * VERR_CFGM_VALUE_NOT_FOUND means that the key is not known.
817 * VERR_CFGM_NOT_ENOUGH_SPACE means that the buffer is not big enough.
818 * @param pvUser The opaque user data associated with this interface.
819 * @param pszName Name of the key to query.
820 * @param ppvData Pointer to buffer where to store the data.
821 * @param cbData Length of data buffer.
822 */
823 DECLR3CALLBACKMEMBER(int, pfnQueryBytes, (void *pvUser, const char *pszName, void *ppvData, size_t cbData));
824
825} VDINTERFACECONFIG, *PVDINTERFACECONFIG;
826
827/**
828 * Get configuration information interface from interface list.
829 *
830 * @return Pointer to the first configuration information interface in the list.
831 * @param pVDIfs Pointer to the interface list.
832 */
833DECLINLINE(PVDINTERFACECONFIG) VDIfConfigGet(PVDINTERFACE pVDIfs)
834{
835 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_CONFIG);
836
837 /* Check that the interface descriptor is a progress interface. */
838 AssertMsgReturn( !pIf
839 || ( (pIf->enmInterface == VDINTERFACETYPE_CONFIG)
840 && (pIf->cbSize == sizeof(VDINTERFACECONFIG))),
841 ("Not a config interface"), NULL);
842
843 return (PVDINTERFACECONFIG)pIf;
844}
845
846/**
847 * Query configuration, validates that the keys are within a set of valid names.
848 *
849 * @return true if all key names are found in pszzAllowed.
850 * @return false if not.
851 * @param pCfgIf Pointer to configuration callback table.
852 * @param pszzValid List of valid names separated by '\\0' and ending with
853 * a double '\\0'.
854 */
855DECLINLINE(bool) VDCFGAreKeysValid(PVDINTERFACECONFIG pCfgIf, const char *pszzValid)
856{
857 return pCfgIf->pfnAreKeysValid(pCfgIf->Core.pvUser, pszzValid);
858}
859
860/**
861 * Checks whether a given key is existing.
862 *
863 * @return true if the key exists.
864 * @return false if the key does not exist.
865 * @param pCfgIf Pointer to configuration callback table.
866 * @param pszName Name of the key.
867 */
868DECLINLINE(bool) VDCFGIsKeyExisting(PVDINTERFACECONFIG pCfgIf, const char *pszName)
869{
870 size_t cb = 0;
871 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
872 return rc == VERR_CFGM_VALUE_NOT_FOUND ? false : true;
873}
874
875/**
876 * Query configuration, unsigned 64-bit integer value with default.
877 *
878 * @return VBox status code.
879 * @param pCfgIf Pointer to configuration callback table.
880 * @param pszName Name of an integer value
881 * @param pu64 Where to store the value. Set to default on failure.
882 * @param u64Def The default value.
883 */
884DECLINLINE(int) VDCFGQueryU64Def(PVDINTERFACECONFIG pCfgIf,
885 const char *pszName, uint64_t *pu64,
886 uint64_t u64Def)
887{
888 char aszBuf[32];
889 int rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, aszBuf, sizeof(aszBuf));
890 if (RT_SUCCESS(rc))
891 {
892 rc = RTStrToUInt64Full(aszBuf, 0, pu64);
893 }
894 else if (rc == VERR_CFGM_VALUE_NOT_FOUND)
895 {
896 rc = VINF_SUCCESS;
897 *pu64 = u64Def;
898 }
899 return rc;
900}
901
902/**
903 * Query configuration, unsigned 64-bit integer value.
904 *
905 * @return VBox status code.
906 * @param pCfgIf Pointer to configuration callback table.
907 * @param pszName Name of an integer value
908 * @param pu64 Where to store the value.
909 */
910DECLINLINE(int) VDCFGQueryU64(PVDINTERFACECONFIG pCfgIf, const char *pszName,
911 uint64_t *pu64)
912{
913 char aszBuf[32];
914 int rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, aszBuf, sizeof(aszBuf));
915 if (RT_SUCCESS(rc))
916 {
917 rc = RTStrToUInt64Full(aszBuf, 0, pu64);
918 }
919
920 return rc;
921}
922
923/**
924 * Query configuration, unsigned 32-bit integer value with default.
925 *
926 * @return VBox status code.
927 * @param pCfgIf Pointer to configuration callback table.
928 * @param pszName Name of an integer value
929 * @param pu32 Where to store the value. Set to default on failure.
930 * @param u32Def The default value.
931 */
932DECLINLINE(int) VDCFGQueryU32Def(PVDINTERFACECONFIG pCfgIf,
933 const char *pszName, uint32_t *pu32,
934 uint32_t u32Def)
935{
936 uint64_t u64;
937 int rc = VDCFGQueryU64Def(pCfgIf, pszName, &u64, u32Def);
938 if (RT_SUCCESS(rc))
939 {
940 if (!(u64 & UINT64_C(0xffffffff00000000)))
941 *pu32 = (uint32_t)u64;
942 else
943 rc = VERR_CFGM_INTEGER_TOO_BIG;
944 }
945 return rc;
946}
947
948/**
949 * Query configuration, bool value with default.
950 *
951 * @return VBox status code.
952 * @param pCfgIf Pointer to configuration callback table.
953 * @param pszName Name of an integer value
954 * @param pf Where to store the value. Set to default on failure.
955 * @param fDef The default value.
956 */
957DECLINLINE(int) VDCFGQueryBoolDef(PVDINTERFACECONFIG pCfgIf,
958 const char *pszName, bool *pf,
959 bool fDef)
960{
961 uint64_t u64;
962 int rc = VDCFGQueryU64Def(pCfgIf, pszName, &u64, fDef);
963 if (RT_SUCCESS(rc))
964 *pf = u64 ? true : false;
965 return rc;
966}
967
968/**
969 * Query configuration, bool value.
970 *
971 * @return VBox status code.
972 * @param pCfgIf Pointer to configuration callback table.
973 * @param pszName Name of an integer value
974 * @param pf Where to store the value.
975 */
976DECLINLINE(int) VDCFGQueryBool(PVDINTERFACECONFIG pCfgIf, const char *pszName,
977 bool *pf)
978{
979 uint64_t u64;
980 int rc = VDCFGQueryU64(pCfgIf, pszName, &u64);
981 if (RT_SUCCESS(rc))
982 *pf = u64 ? true : false;
983 return rc;
984}
985
986/**
987 * Query configuration, dynamically allocated (RTMemAlloc) zero terminated
988 * character value.
989 *
990 * @return VBox status code.
991 * @param pCfgIf Pointer to configuration callback table.
992 * @param pszName Name of an zero terminated character value
993 * @param ppszString Where to store the string pointer. Not set on failure.
994 * Free this using RTMemFree().
995 */
996DECLINLINE(int) VDCFGQueryStringAlloc(PVDINTERFACECONFIG pCfgIf,
997 const char *pszName, char **ppszString)
998{
999 size_t cb;
1000 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
1001 if (RT_SUCCESS(rc))
1002 {
1003 char *pszString = (char *)RTMemAlloc(cb);
1004 if (pszString)
1005 {
1006 rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, pszString, cb);
1007 if (RT_SUCCESS(rc))
1008 *ppszString = pszString;
1009 else
1010 RTMemFree(pszString);
1011 }
1012 else
1013 rc = VERR_NO_MEMORY;
1014 }
1015 return rc;
1016}
1017
1018/**
1019 * Query configuration, dynamically allocated (RTMemAlloc) zero terminated
1020 * character value with default.
1021 *
1022 * @return VBox status code.
1023 * @param pCfgIf Pointer to configuration callback table.
1024 * @param pszName Name of an zero terminated character value
1025 * @param ppszString Where to store the string pointer. Not set on failure.
1026 * Free this using RTMemFree().
1027 * @param pszDef The default value.
1028 */
1029DECLINLINE(int) VDCFGQueryStringAllocDef(PVDINTERFACECONFIG pCfgIf,
1030 const char *pszName,
1031 char **ppszString,
1032 const char *pszDef)
1033{
1034 size_t cb;
1035 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
1036 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
1037 {
1038 cb = strlen(pszDef) + 1;
1039 rc = VINF_SUCCESS;
1040 }
1041 if (RT_SUCCESS(rc))
1042 {
1043 char *pszString = (char *)RTMemAlloc(cb);
1044 if (pszString)
1045 {
1046 rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, pszString, cb);
1047 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
1048 {
1049 memcpy(pszString, pszDef, cb);
1050 rc = VINF_SUCCESS;
1051 }
1052 if (RT_SUCCESS(rc))
1053 *ppszString = pszString;
1054 else
1055 RTMemFree(pszString);
1056 }
1057 else
1058 rc = VERR_NO_MEMORY;
1059 }
1060 return rc;
1061}
1062
1063/**
1064 * Query configuration, dynamically allocated (RTMemAlloc) byte string value.
1065 *
1066 * @return VBox status code.
1067 * @param pCfgIf Pointer to configuration callback table.
1068 * @param pszName Name of an zero terminated character value
1069 * @param ppvData Where to store the byte string pointer. Not set on failure.
1070 * Free this using RTMemFree().
1071 * @param pcbData Where to store the byte string length.
1072 */
1073DECLINLINE(int) VDCFGQueryBytesAlloc(PVDINTERFACECONFIG pCfgIf,
1074 const char *pszName, void **ppvData, size_t *pcbData)
1075{
1076 size_t cb;
1077 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
1078 if (RT_SUCCESS(rc))
1079 {
1080 char *pbData;
1081 Assert(cb);
1082
1083 pbData = (char *)RTMemAlloc(cb);
1084 if (pbData)
1085 {
1086 if(pCfgIf->pfnQueryBytes)
1087 rc = pCfgIf->pfnQueryBytes(pCfgIf->Core.pvUser, pszName, pbData, cb);
1088 else
1089 rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, pbData, cb);
1090
1091 if (RT_SUCCESS(rc))
1092 {
1093 *ppvData = pbData;
1094 /* Exclude terminator if the byte data was obtained using the string query callback. */
1095 *pcbData = cb;
1096 if (!pCfgIf->pfnQueryBytes)
1097 (*pcbData)--;
1098 }
1099 else
1100 RTMemFree(pbData);
1101 }
1102 else
1103 rc = VERR_NO_MEMORY;
1104 }
1105 return rc;
1106}
1107
1108/** Forward declaration of a VD socket. */
1109typedef struct VDSOCKETINT *VDSOCKET;
1110/** Pointer to a VD socket. */
1111typedef VDSOCKET *PVDSOCKET;
1112/** Nil socket handle. */
1113#define NIL_VDSOCKET ((VDSOCKET)0)
1114
1115/** Connect flag to indicate that the backend wants to use the extended
1116 * socket I/O multiplexing call. This might not be supported on all configurations
1117 * (internal networking and iSCSI)
1118 * and the backend needs to take appropriate action.
1119 */
1120#define VD_INTERFACETCPNET_CONNECT_EXTENDED_SELECT RT_BIT_32(0)
1121
1122/** @name Select events
1123 * @{ */
1124/** Readable without blocking. */
1125#define VD_INTERFACETCPNET_EVT_READ RT_BIT_32(0)
1126/** Writable without blocking. */
1127#define VD_INTERFACETCPNET_EVT_WRITE RT_BIT_32(1)
1128/** Error condition, hangup, exception or similar. */
1129#define VD_INTERFACETCPNET_EVT_ERROR RT_BIT_32(2)
1130/** Hint for the select that getting interrupted while waiting is more likely.
1131 * The interface implementation can optimize the waiting strategy based on this.
1132 * It is assumed that it is more likely to get one of the above socket events
1133 * instead of being interrupted if the flag is not set. */
1134#define VD_INTERFACETCPNET_HINT_INTERRUPT RT_BIT_32(3)
1135/** Mask of the valid bits. */
1136#define VD_INTERFACETCPNET_EVT_VALID_MASK UINT32_C(0x0000000f)
1137/** @} */
1138
1139/**
1140 * TCP network stack interface
1141 *
1142 * Per-image. Mandatory for backends which have the VD_CAP_TCPNET bit set.
1143 */
1144typedef struct VDINTERFACETCPNET
1145{
1146 /**
1147 * Common interface header.
1148 */
1149 VDINTERFACE Core;
1150
1151 /**
1152 * Creates a socket. The socket is not connected if this succeeds.
1153 *
1154 * @return iprt status code.
1155 * @retval VERR_NOT_SUPPORTED if the combination of flags is not supported.
1156 * @param fFlags Combination of the VD_INTERFACETCPNET_CONNECT_* \#defines.
1157 * @param phVdSock Where to store the handle.
1158 */
1159 DECLR3CALLBACKMEMBER(int, pfnSocketCreate, (uint32_t fFlags, PVDSOCKET phVdSock));
1160
1161 /**
1162 * Destroys the socket.
1163 *
1164 * @return iprt status code.
1165 * @param hVdSock Socket handle (/ pointer).
1166 */
1167 DECLR3CALLBACKMEMBER(int, pfnSocketDestroy, (VDSOCKET hVdSock));
1168
1169 /**
1170 * Connect as a client to a TCP port.
1171 *
1172 * @return iprt status code.
1173 * @param hVdSock Socket handle (/ pointer)..
1174 * @param pszAddress The address to connect to.
1175 * @param uPort The port to connect to.
1176 * @param cMillies Number of milliseconds to wait for the connect attempt to complete.
1177 * Use RT_INDEFINITE_WAIT to wait for ever.
1178 * Use RT_SOCKETCONNECT_DEFAULT_WAIT to wait for the default time
1179 * configured on the running system.
1180 */
1181 DECLR3CALLBACKMEMBER(int, pfnClientConnect, (VDSOCKET hVdSock, const char *pszAddress, uint32_t uPort,
1182 RTMSINTERVAL cMillies));
1183
1184 /**
1185 * Close a TCP connection.
1186 *
1187 * @return iprt status code.
1188 * @param hVdSock Socket handle (/ pointer).
1189 */
1190 DECLR3CALLBACKMEMBER(int, pfnClientClose, (VDSOCKET hVdSock));
1191
1192 /**
1193 * Returns whether the socket is currently connected to the client.
1194 *
1195 * @returns true if the socket is connected.
1196 * false otherwise.
1197 * @param hVdSock Socket handle (/ pointer).
1198 */
1199 DECLR3CALLBACKMEMBER(bool, pfnIsClientConnected, (VDSOCKET hVdSock));
1200
1201 /**
1202 * Socket I/O multiplexing.
1203 * Checks if the socket is ready for reading.
1204 *
1205 * @return iprt status code.
1206 * @param hVdSock Socket handle (/ pointer).
1207 * @param cMillies Number of milliseconds to wait for the socket.
1208 * Use RT_INDEFINITE_WAIT to wait for ever.
1209 */
1210 DECLR3CALLBACKMEMBER(int, pfnSelectOne, (VDSOCKET hVdSock, RTMSINTERVAL cMillies));
1211
1212 /**
1213 * Receive data from a socket.
1214 *
1215 * @return iprt status code.
1216 * @param hVdSock Socket handle (/ pointer).
1217 * @param pvBuffer Where to put the data we read.
1218 * @param cbBuffer Read buffer size.
1219 * @param pcbRead Number of bytes read.
1220 * If NULL the entire buffer will be filled upon successful return.
1221 * If not NULL a partial read can be done successfully.
1222 */
1223 DECLR3CALLBACKMEMBER(int, pfnRead, (VDSOCKET hVdSock, void *pvBuffer, size_t cbBuffer, size_t *pcbRead));
1224
1225 /**
1226 * Send data to a socket.
1227 *
1228 * @return iprt status code.
1229 * @param hVdSock Socket handle (/ pointer).
1230 * @param pvBuffer Buffer to write data to socket.
1231 * @param cbBuffer How much to write.
1232 */
1233 DECLR3CALLBACKMEMBER(int, pfnWrite, (VDSOCKET hVdSock, const void *pvBuffer, size_t cbBuffer));
1234
1235 /**
1236 * Send data from scatter/gather buffer to a socket.
1237 *
1238 * @return iprt status code.
1239 * @param hVdSock Socket handle (/ pointer).
1240 * @param pSgBuf Scatter/gather buffer to write data to socket.
1241 */
1242 DECLR3CALLBACKMEMBER(int, pfnSgWrite, (VDSOCKET hVdSock, PCRTSGBUF pSgBuf));
1243
1244 /**
1245 * Receive data from a socket - not blocking.
1246 *
1247 * @return iprt status code.
1248 * @param hVdSock Socket handle (/ pointer).
1249 * @param pvBuffer Where to put the data we read.
1250 * @param cbBuffer Read buffer size.
1251 * @param pcbRead Number of bytes read.
1252 */
1253 DECLR3CALLBACKMEMBER(int, pfnReadNB, (VDSOCKET hVdSock, void *pvBuffer, size_t cbBuffer, size_t *pcbRead));
1254
1255 /**
1256 * Send data to a socket - not blocking.
1257 *
1258 * @return iprt status code.
1259 * @param hVdSock Socket handle (/ pointer).
1260 * @param pvBuffer Buffer to write data to socket.
1261 * @param cbBuffer How much to write.
1262 * @param pcbWritten Number of bytes written.
1263 */
1264 DECLR3CALLBACKMEMBER(int, pfnWriteNB, (VDSOCKET hVdSock, const void *pvBuffer, size_t cbBuffer, size_t *pcbWritten));
1265
1266 /**
1267 * Send data from scatter/gather buffer to a socket - not blocking.
1268 *
1269 * @return iprt status code.
1270 * @param hVdSock Socket handle (/ pointer).
1271 * @param pSgBuf Scatter/gather buffer to write data to socket.
1272 * @param pcbWritten Number of bytes written.
1273 */
1274 DECLR3CALLBACKMEMBER(int, pfnSgWriteNB, (VDSOCKET hVdSock, PRTSGBUF pSgBuf, size_t *pcbWritten));
1275
1276 /**
1277 * Flush socket write buffers.
1278 *
1279 * @return iprt status code.
1280 * @param hVdSock Socket handle (/ pointer).
1281 */
1282 DECLR3CALLBACKMEMBER(int, pfnFlush, (VDSOCKET hVdSock));
1283
1284 /**
1285 * Enables or disables delaying sends to coalesce packets.
1286 *
1287 * @return iprt status code.
1288 * @param hVdSock Socket handle (/ pointer).
1289 * @param fEnable When set to true enables coalescing.
1290 */
1291 DECLR3CALLBACKMEMBER(int, pfnSetSendCoalescing, (VDSOCKET hVdSock, bool fEnable));
1292
1293 /**
1294 * Gets the address of the local side.
1295 *
1296 * @return iprt status code.
1297 * @param hVdSock Socket handle (/ pointer).
1298 * @param pAddr Where to store the local address on success.
1299 */
1300 DECLR3CALLBACKMEMBER(int, pfnGetLocalAddress, (VDSOCKET hVdSock, PRTNETADDR pAddr));
1301
1302 /**
1303 * Gets the address of the other party.
1304 *
1305 * @return iprt status code.
1306 * @param hVdSock Socket handle (/ pointer).
1307 * @param pAddr Where to store the peer address on success.
1308 */
1309 DECLR3CALLBACKMEMBER(int, pfnGetPeerAddress, (VDSOCKET hVdSock, PRTNETADDR pAddr));
1310
1311 /**
1312 * Socket I/O multiplexing - extended version which can be woken up.
1313 * Checks if the socket is ready for reading or writing.
1314 *
1315 * @return iprt status code.
1316 * @retval VERR_INTERRUPTED if the thread was woken up by a pfnPoke call.
1317 * @param hVdSock VD Socket handle(/pointer).
1318 * @param fEvents Mask of events to wait for.
1319 * @param pfEvents Where to store the received events.
1320 * @param cMillies Number of milliseconds to wait for the socket.
1321 * Use RT_INDEFINITE_WAIT to wait for ever.
1322 */
1323 DECLR3CALLBACKMEMBER(int, pfnSelectOneEx, (VDSOCKET hVdSock, uint32_t fEvents,
1324 uint32_t *pfEvents, RTMSINTERVAL cMillies));
1325
1326 /**
1327 * Wakes up the thread waiting in pfnSelectOneEx.
1328 *
1329 * @return iprt status code.
1330 * @param hVdSock VD Socket handle(/pointer).
1331 */
1332 DECLR3CALLBACKMEMBER(int, pfnPoke, (VDSOCKET hVdSock));
1333
1334} VDINTERFACETCPNET, *PVDINTERFACETCPNET;
1335
1336/**
1337 * Get TCP network stack interface from interface list.
1338 *
1339 * @return Pointer to the first TCP network stack interface in the list.
1340 * @param pVDIfs Pointer to the interface list.
1341 */
1342DECLINLINE(PVDINTERFACETCPNET) VDIfTcpNetGet(PVDINTERFACE pVDIfs)
1343{
1344 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_TCPNET);
1345
1346 /* Check that the interface descriptor is a progress interface. */
1347 AssertMsgReturn( !pIf
1348 || ( (pIf->enmInterface == VDINTERFACETYPE_TCPNET)
1349 && (pIf->cbSize == sizeof(VDINTERFACETCPNET))),
1350 ("Not a TCP net interface"), NULL);
1351
1352 return (PVDINTERFACETCPNET)pIf;
1353}
1354
1355
1356/**
1357 * Interface to synchronize concurrent accesses by several threads.
1358 *
1359 * @note The scope of this interface is to manage concurrent accesses after
1360 * the HDD container has been created, and they must stop before destroying the
1361 * container. Opening or closing images is covered by the synchronization, but
1362 * that does not mean it is safe to close images while a thread executes
1363 * #VDMerge or #VDCopy operating on these images. Making them safe would require
1364 * the lock to be held during the entire operation, which prevents other
1365 * concurrent acitivities.
1366 *
1367 * @note Right now this is kept as simple as possible, and does not even
1368 * attempt to provide enough information to allow e.g. concurrent write
1369 * accesses to different areas of the disk. The reason is that it is very
1370 * difficult to predict which area of a disk is affected by a write,
1371 * especially when different image formats are mixed. Maybe later a more
1372 * sophisticated interface will be provided which has the necessary information
1373 * about worst case affected areas.
1374 *
1375 * Per-disk interface. Optional, needed if the disk is accessed concurrently
1376 * by several threads, e.g. when merging diff images while a VM is running.
1377 */
1378typedef struct VDINTERFACETHREADSYNC
1379{
1380 /**
1381 * Common interface header.
1382 */
1383 VDINTERFACE Core;
1384
1385 /**
1386 * Start a read operation.
1387 */
1388 DECLR3CALLBACKMEMBER(int, pfnStartRead, (void *pvUser));
1389
1390 /**
1391 * Finish a read operation.
1392 */
1393 DECLR3CALLBACKMEMBER(int, pfnFinishRead, (void *pvUser));
1394
1395 /**
1396 * Start a write operation.
1397 */
1398 DECLR3CALLBACKMEMBER(int, pfnStartWrite, (void *pvUser));
1399
1400 /**
1401 * Finish a write operation.
1402 */
1403 DECLR3CALLBACKMEMBER(int, pfnFinishWrite, (void *pvUser));
1404
1405} VDINTERFACETHREADSYNC, *PVDINTERFACETHREADSYNC;
1406
1407/**
1408 * Get thread synchronization interface from interface list.
1409 *
1410 * @return Pointer to the first thread synchronization interface in the list.
1411 * @param pVDIfs Pointer to the interface list.
1412 */
1413DECLINLINE(PVDINTERFACETHREADSYNC) VDIfThreadSyncGet(PVDINTERFACE pVDIfs)
1414{
1415 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_THREADSYNC);
1416
1417 /* Check that the interface descriptor is a progress interface. */
1418 AssertMsgReturn( !pIf
1419 || ( (pIf->enmInterface == VDINTERFACETYPE_THREADSYNC)
1420 && (pIf->cbSize == sizeof(VDINTERFACETHREADSYNC))),
1421 ("Not a thread synchronization interface"), NULL);
1422
1423 return (PVDINTERFACETHREADSYNC)pIf;
1424}
1425
1426/**
1427 * Interface to query usage of disk ranges.
1428 *
1429 * Per-operation interface. Optional.
1430 */
1431typedef struct VDINTERFACEQUERYRANGEUSE
1432{
1433 /**
1434 * Common interface header.
1435 */
1436 VDINTERFACE Core;
1437
1438 /**
1439 * Query use of a disk range.
1440 */
1441 DECLR3CALLBACKMEMBER(int, pfnQueryRangeUse, (void *pvUser, uint64_t off, uint64_t cb,
1442 bool *pfUsed));
1443
1444} VDINTERFACEQUERYRANGEUSE, *PVDINTERFACEQUERYRANGEUSE;
1445
1446/**
1447 * Get query range use interface from interface list.
1448 *
1449 * @return Pointer to the first thread synchronization interface in the list.
1450 * @param pVDIfs Pointer to the interface list.
1451 */
1452DECLINLINE(PVDINTERFACEQUERYRANGEUSE) VDIfQueryRangeUseGet(PVDINTERFACE pVDIfs)
1453{
1454 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_QUERYRANGEUSE);
1455
1456 /* Check that the interface descriptor is a progress interface. */
1457 AssertMsgReturn( !pIf
1458 || ( (pIf->enmInterface == VDINTERFACETYPE_QUERYRANGEUSE)
1459 && (pIf->cbSize == sizeof(VDINTERFACEQUERYRANGEUSE))),
1460 ("Not a query range use interface"), NULL);
1461
1462 return (PVDINTERFACEQUERYRANGEUSE)pIf;
1463}
1464
1465DECLINLINE(int) vdIfQueryRangeUse(PVDINTERFACEQUERYRANGEUSE pIfQueryRangeUse, uint64_t off, uint64_t cb,
1466 bool *pfUsed)
1467{
1468 return pIfQueryRangeUse->pfnQueryRangeUse(pIfQueryRangeUse->Core.pvUser, off, cb, pfUsed);
1469}
1470
1471
1472/**
1473 * Interface used to retrieve keys for cryptographic operations.
1474 *
1475 * Per-module interface. Optional but cryptographic modules might fail and
1476 * return an error if this is not present.
1477 */
1478typedef struct VDINTERFACECRYPTO
1479{
1480 /**
1481 * Common interface header.
1482 */
1483 VDINTERFACE Core;
1484
1485 /**
1486 * Retains a key identified by the ID. The caller will only hold a reference
1487 * to the key and must not modify the key buffer in any way.
1488 *
1489 * @returns VBox status code.
1490 * @param pvUser The opaque user data associated with this interface.
1491 * @param pszId The alias/id for the key to retrieve.
1492 * @param ppbKey Where to store the pointer to the key buffer on success.
1493 * @param pcbKey Where to store the size of the key in bytes on success.
1494 */
1495 DECLR3CALLBACKMEMBER(int, pfnKeyRetain, (void *pvUser, const char *pszId, const uint8_t **ppbKey, size_t *pcbKey));
1496
1497 /**
1498 * Releases one reference of the key identified by the given identifier.
1499 * The caller must not access the key buffer after calling this operation.
1500 *
1501 * @returns VBox status code.
1502 * @param pvUser The opaque user data associated with this interface.
1503 * @param pszId The alias/id for the key to release.
1504 *
1505 * @note It is advised to release the key whenever it is not used anymore so
1506 * the entity storing the key can do anything to make retrieving the key
1507 * from memory more difficult like scrambling the memory buffer for
1508 * instance.
1509 */
1510 DECLR3CALLBACKMEMBER(int, pfnKeyRelease, (void *pvUser, const char *pszId));
1511
1512 /**
1513 * Gets a reference to the password identified by the given ID to open a key store supplied through the config interface.
1514 *
1515 * @returns VBox status code.
1516 * @param pvUser The opaque user data associated with this interface.
1517 * @param pszId The alias/id for the password to retain.
1518 * @param ppszPassword Where to store the password to unlock the key store on success.
1519 */
1520 DECLR3CALLBACKMEMBER(int, pfnKeyStorePasswordRetain, (void *pvUser, const char *pszId, const char **ppszPassword));
1521
1522 /**
1523 * Releases a reference of the password previously acquired with VDINTERFACECRYPTO::pfnKeyStorePasswordRetain()
1524 * identified by the given ID.
1525 *
1526 * @returns VBox status code.
1527 * @param pvUser The opaque user data associated with this interface.
1528 * @param pszId The alias/id for the password to release.
1529 */
1530 DECLR3CALLBACKMEMBER(int, pfnKeyStorePasswordRelease, (void *pvUser, const char *pszId));
1531
1532 /**
1533 * Saves a key store.
1534 *
1535 * @returns VBox status code.
1536 * @param pvUser The opaque user data associated with this interface.
1537 * @param pvKeyStore The key store to save.
1538 * @param cbKeyStore Size of the key store in bytes.
1539 *
1540 * @note The format is filter specific and should be treated as binary data.
1541 */
1542 DECLR3CALLBACKMEMBER(int, pfnKeyStoreSave, (void *pvUser, const void *pvKeyStore, size_t cbKeyStore));
1543
1544 /**
1545 * Returns the parameters after the key store was loaded successfully.
1546 *
1547 * @returns VBox status code.
1548 * @param pvUser The opaque user data associated with this interface.
1549 * @param pszCipher The cipher identifier the DEK is used for.
1550 * @param pbDek The raw DEK which was contained in the key store loaded by
1551 * VDINTERFACECRYPTO::pfnKeyStoreLoad().
1552 * @param cbDek The size of the DEK.
1553 *
1554 * @note The provided pointer to the DEK is only valid until this call returns.
1555 * The content might change afterwards with out notice (when scrambling the key
1556 * for further protection for example) or might be even freed.
1557 *
1558 * @note This method is optional and can be NULL if the caller does not require the
1559 * parameters.
1560 */
1561 DECLR3CALLBACKMEMBER(int, pfnKeyStoreReturnParameters, (void *pvUser, const char *pszCipher,
1562 const uint8_t *pbDek, size_t cbDek));
1563
1564} VDINTERFACECRYPTO, *PVDINTERFACECRYPTO;
1565
1566
1567/**
1568 * Get error interface from interface list.
1569 *
1570 * @return Pointer to the first error interface in the list.
1571 * @param pVDIfs Pointer to the interface list.
1572 */
1573DECLINLINE(PVDINTERFACECRYPTO) VDIfCryptoGet(PVDINTERFACE pVDIfs)
1574{
1575 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_CRYPTO);
1576
1577 /* Check that the interface descriptor is a crypto interface. */
1578 AssertMsgReturn( !pIf
1579 || ( (pIf->enmInterface == VDINTERFACETYPE_CRYPTO)
1580 && (pIf->cbSize == sizeof(VDINTERFACECRYPTO))),
1581 ("Not an crypto interface\n"), NULL);
1582
1583 return (PVDINTERFACECRYPTO)pIf;
1584}
1585
1586/**
1587 * Retains a key identified by the ID. The caller will only hold a reference
1588 * to the key and must not modify the key buffer in any way.
1589 *
1590 * @returns VBox status code.
1591 * @param pIfCrypto Pointer to the crypto interface.
1592 * @param pszId The alias/id for the key to retrieve.
1593 * @param ppbKey Where to store the pointer to the key buffer on success.
1594 * @param pcbKey Where to store the size of the key in bytes on success.
1595 */
1596DECLINLINE(int) vdIfCryptoKeyRetain(PVDINTERFACECRYPTO pIfCrypto, const char *pszId, const uint8_t **ppbKey, size_t *pcbKey)
1597{
1598 return pIfCrypto->pfnKeyRetain(pIfCrypto->Core.pvUser, pszId, ppbKey, pcbKey);
1599}
1600
1601/**
1602 * Releases one reference of the key identified by the given identifier.
1603 * The caller must not access the key buffer after calling this operation.
1604 *
1605 * @returns VBox status code.
1606 * @param pIfCrypto Pointer to the crypto interface.
1607 * @param pszId The alias/id for the key to release.
1608 *
1609 * @note It is advised to release the key whenever it is not used anymore so
1610 * the entity storing the key can do anything to make retrieving the key
1611 * from memory more difficult like scrambling the memory buffer for
1612 * instance.
1613 */
1614DECLINLINE(int) vdIfCryptoKeyRelease(PVDINTERFACECRYPTO pIfCrypto, const char *pszId)
1615{
1616 return pIfCrypto->pfnKeyRelease(pIfCrypto->Core.pvUser, pszId);
1617}
1618
1619/**
1620 * Gets a reference to the password identified by the given ID to open a key store supplied through the config interface.
1621 *
1622 * @returns VBox status code.
1623 * @param pIfCrypto Pointer to the crypto interface.
1624 * @param pszId The alias/id for the password to retain.
1625 * @param ppszPassword Where to store the password to unlock the key store on success.
1626 */
1627DECLINLINE(int) vdIfCryptoKeyStorePasswordRetain(PVDINTERFACECRYPTO pIfCrypto, const char *pszId, const char **ppszPassword)
1628{
1629 return pIfCrypto->pfnKeyStorePasswordRetain(pIfCrypto->Core.pvUser, pszId, ppszPassword);
1630}
1631
1632/**
1633 * Releases a reference of the password previously acquired with VDINTERFACECRYPTO::pfnKeyStorePasswordRetain()
1634 * identified by the given ID.
1635 *
1636 * @returns VBox status code.
1637 * @param pIfCrypto Pointer to the crypto interface.
1638 * @param pszId The alias/id for the password to release.
1639 */
1640DECLINLINE(int) vdIfCryptoKeyStorePasswordRelease(PVDINTERFACECRYPTO pIfCrypto, const char *pszId)
1641{
1642 return pIfCrypto->pfnKeyStorePasswordRelease(pIfCrypto->Core.pvUser, pszId);
1643}
1644
1645/**
1646 * Saves a key store.
1647 *
1648 * @returns VBox status code.
1649 * @param pIfCrypto Pointer to the crypto interface.
1650 * @param pvKeyStore The key store to save.
1651 * @param cbKeyStore Size of the key store in bytes.
1652 *
1653 * @note The format is filter specific and should be treated as binary data.
1654 */
1655DECLINLINE(int) vdIfCryptoKeyStoreSave(PVDINTERFACECRYPTO pIfCrypto, const void *pvKeyStore, size_t cbKeyStore)
1656{
1657 return pIfCrypto->pfnKeyStoreSave(pIfCrypto->Core.pvUser, pvKeyStore, cbKeyStore);
1658}
1659
1660/**
1661 * Returns the parameters after the key store was loaded successfully.
1662 *
1663 * @returns VBox status code.
1664 * @param pIfCrypto Pointer to the crypto interface.
1665 * @param pszCipher The cipher identifier the DEK is used for.
1666 * @param pbDek The raw DEK which was contained in the key store loaded by
1667 * VDINTERFACECRYPTO::pfnKeyStoreLoad().
1668 * @param cbDek The size of the DEK.
1669 *
1670 * @note The provided pointer to the DEK is only valid until this call returns.
1671 * The content might change afterwards with out notice (when scrambling the key
1672 * for further protection for example) or might be even freed.
1673 *
1674 * @note This method is optional and can be NULL if the caller does not require the
1675 * parameters.
1676 */
1677DECLINLINE(int) vdIfCryptoKeyStoreReturnParameters(PVDINTERFACECRYPTO pIfCrypto, const char *pszCipher,
1678 const uint8_t *pbDek, size_t cbDek)
1679{
1680 if (pIfCrypto->pfnKeyStoreReturnParameters)
1681 return pIfCrypto->pfnKeyStoreReturnParameters(pIfCrypto->Core.pvUser, pszCipher, pbDek, cbDek);
1682
1683 return VINF_SUCCESS;
1684}
1685
1686
1687RT_C_DECLS_END
1688
1689/** @} */
1690
1691#endif
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