VirtualBox

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

Last change on this file since 56402 was 54591, checked in by vboxsync, 10 years ago

Add support to supply passwords for disk encryption while the VM is running

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 56.4 KB
Line 
1/** @file
2 * VD Container API - interfaces.
3 */
4
5/*
6 * Copyright (C) 2011-2015 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_Interfaces_h
27#define ___VBox_VD_Interfaces_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 ppVDIfs Pointer to the VD interface list.
146 */
147DECLINLINE(int) VDInterfaceAdd(PVDINTERFACE pInterface, const char *pszName,
148 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 RT_SRC_POS_DECL 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, const char *pszFormat, va_list va));
247
248 /**
249 * Informational message callback. May be NULL. Used e.g. in
250 * VDDumpImages(). Must be able to accept special IPRT format strings.
251 *
252 * @return VBox status code.
253 * @param pvUser The opaque data passed on container creation.
254 * @param pszFormat Message format string.
255 * @param va Message arguments.
256 */
257 DECLR3CALLBACKMEMBER(int, pfnMessage, (void *pvUser, const char *pszFormat, va_list va));
258
259} VDINTERFACEERROR, *PVDINTERFACEERROR;
260
261/**
262 * Get error interface from interface list.
263 *
264 * @return Pointer to the first error interface in the list.
265 * @param pVDIfs Pointer to the interface list.
266 */
267DECLINLINE(PVDINTERFACEERROR) VDIfErrorGet(PVDINTERFACE pVDIfs)
268{
269 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_ERROR);
270
271 /* Check that the interface descriptor is a progress interface. */
272 AssertMsgReturn( !pIf
273 || ( (pIf->enmInterface == VDINTERFACETYPE_ERROR)
274 && (pIf->cbSize == sizeof(VDINTERFACEERROR))),
275 ("Not an error interface\n"), NULL);
276
277 return (PVDINTERFACEERROR)pIf;
278}
279
280/**
281 * Signal an error to the frontend.
282 *
283 * @returns VBox status code.
284 * @param pIfError The error interface.
285 * @param rc The status code.
286 * @param RT_SRC_POS_DECL The position in the source code.
287 * @param pszFormat The format string to pass.
288 * @param ... Arguments to the format string.
289 */
290DECLINLINE(int) vdIfError(PVDINTERFACEERROR pIfError, int rc, RT_SRC_POS_DECL,
291 const char *pszFormat, ...)
292{
293 va_list va;
294 va_start(va, pszFormat);
295 if (pIfError)
296 pIfError->pfnError(pIfError->Core.pvUser, rc, RT_SRC_POS_ARGS, pszFormat, va);
297 va_end(va);
298 return rc;
299}
300
301/**
302 * Signal an informational message to the frontend.
303 *
304 * @returns VBox status code.
305 * @param pIfError The error interface.
306 * @param pszFormat The format string to pass.
307 * @param ... Arguments to the format string.
308 */
309DECLINLINE(int) vdIfErrorMessage(PVDINTERFACEERROR pIfError, const char *pszFormat, ...)
310{
311 int rc = VINF_SUCCESS;
312 va_list va;
313 va_start(va, pszFormat);
314 if (pIfError && pIfError->pfnMessage)
315 rc = pIfError->pfnMessage(pIfError->Core.pvUser, pszFormat, va);
316 va_end(va);
317 return rc;
318}
319
320/**
321 * Completion callback which is called by the interface owner
322 * to inform the backend that a task finished.
323 *
324 * @return VBox status code.
325 * @param pvUser Opaque user data which is passed on request submission.
326 * @param rcReq Status code of the completed request.
327 */
328typedef DECLCALLBACK(int) FNVDCOMPLETED(void *pvUser, int rcReq);
329/** Pointer to FNVDCOMPLETED() */
330typedef FNVDCOMPLETED *PFNVDCOMPLETED;
331
332/**
333 * Support interface for I/O
334 *
335 * Per-image. Optional as input.
336 */
337typedef struct VDINTERFACEIO
338{
339 /**
340 * Common interface header.
341 */
342 VDINTERFACE Core;
343
344 /**
345 * Open callback
346 *
347 * @return VBox status code.
348 * @param pvUser The opaque data passed on container creation.
349 * @param pszLocation Name of the location to open.
350 * @param fOpen Flags for opening the backend.
351 * See RTFILE_O_* #defines, inventing another set
352 * of open flags is not worth the mapping effort.
353 * @param pfnCompleted The callback which is called whenever a task
354 * completed. The backend has to pass the user data
355 * of the request initiator (ie the one who calls
356 * VDAsyncRead or VDAsyncWrite) in pvCompletion
357 * if this is NULL.
358 * @param ppStorage Where to store the opaque storage handle.
359 */
360 DECLR3CALLBACKMEMBER(int, pfnOpen, (void *pvUser, const char *pszLocation,
361 uint32_t fOpen,
362 PFNVDCOMPLETED pfnCompleted,
363 void **ppStorage));
364
365 /**
366 * Close callback.
367 *
368 * @return VBox status code.
369 * @param pvUser The opaque data passed on container creation.
370 * @param pStorage The opaque storage handle to close.
371 */
372 DECLR3CALLBACKMEMBER(int, pfnClose, (void *pvUser, void *pStorage));
373
374 /**
375 * Delete callback.
376 *
377 * @return VBox status code.
378 * @param pvUser The opaque data passed on container creation.
379 * @param pcszFilename Name of the file to delete.
380 */
381 DECLR3CALLBACKMEMBER(int, pfnDelete, (void *pvUser, const char *pcszFilename));
382
383 /**
384 * Move callback.
385 *
386 * @return VBox status code.
387 * @param pvUser The opaque data passed on container creation.
388 * @param pcszSrc The path to the source file.
389 * @param pcszDst The path to the destination file.
390 * This file will be created.
391 * @param fMove A combination of the RTFILEMOVE_* flags.
392 */
393 DECLR3CALLBACKMEMBER(int, pfnMove, (void *pvUser, const char *pcszSrc, const char *pcszDst, unsigned fMove));
394
395 /**
396 * Returns the free space on a disk.
397 *
398 * @return VBox status code.
399 * @param pvUser The opaque data passed on container creation.
400 * @param pcszFilename Name of a file to identify the disk.
401 * @param pcbFreeSpace Where to store the free space of the disk.
402 */
403 DECLR3CALLBACKMEMBER(int, pfnGetFreeSpace, (void *pvUser, const char *pcszFilename, int64_t *pcbFreeSpace));
404
405 /**
406 * Returns the last modification timestamp of a file.
407 *
408 * @return VBox status code.
409 * @param pvUser The opaque data passed on container creation.
410 * @param pcszFilename Name of a file to identify the disk.
411 * @param pModificationTime Where to store the timestamp of the file.
412 */
413 DECLR3CALLBACKMEMBER(int, pfnGetModificationTime, (void *pvUser, const char *pcszFilename, PRTTIMESPEC pModificationTime));
414
415 /**
416 * Returns the size of the opened storage backend.
417 *
418 * @return VBox status code.
419 * @param pvUser The opaque data passed on container creation.
420 * @param pStorage The opaque storage handle to close.
421 * @param pcbSize Where to store the size of the storage backend.
422 */
423 DECLR3CALLBACKMEMBER(int, pfnGetSize, (void *pvUser, void *pStorage, uint64_t *pcbSize));
424
425 /**
426 * Sets the size of the opened storage backend if possible.
427 *
428 * @return VBox status code.
429 * @retval VERR_NOT_SUPPORTED if the backend does not support this operation.
430 * @param pvUser The opaque data passed on container creation.
431 * @param pStorage The opaque storage handle to close.
432 * @param cbSize The new size of the image.
433 */
434 DECLR3CALLBACKMEMBER(int, pfnSetSize, (void *pvUser, void *pStorage, uint64_t cbSize));
435
436 /**
437 * Synchronous write callback.
438 *
439 * @return VBox status code.
440 * @param pvUser The opaque data passed on container creation.
441 * @param pStorage The storage handle to use.
442 * @param uOffset The offset to start from.
443 * @param pvBuffer Pointer to the bits need to be written.
444 * @param cbBuffer How many bytes to write.
445 * @param pcbWritten Where to store how many bytes were actually written.
446 */
447 DECLR3CALLBACKMEMBER(int, pfnWriteSync, (void *pvUser, void *pStorage, uint64_t uOffset,
448 const void *pvBuffer, size_t cbBuffer, size_t *pcbWritten));
449
450 /**
451 * Synchronous read callback.
452 *
453 * @return VBox status code.
454 * @param pvUser The opaque data passed on container creation.
455 * @param pStorage The storage handle to use.
456 * @param uOffset The offset to start from.
457 * @param pvBuffer Where to store the read bits.
458 * @param cbBuffer How many bytes to read.
459 * @param pcbRead Where to store how many bytes were actually read.
460 */
461 DECLR3CALLBACKMEMBER(int, pfnReadSync, (void *pvUser, void *pStorage, uint64_t uOffset,
462 void *pvBuffer, size_t cbBuffer, size_t *pcbRead));
463
464 /**
465 * Flush data to the storage backend.
466 *
467 * @return VBox status code.
468 * @param pvUser The opaque data passed on container creation.
469 * @param pStorage The storage handle to flush.
470 */
471 DECLR3CALLBACKMEMBER(int, pfnFlushSync, (void *pvUser, void *pStorage));
472
473 /**
474 * Initiate an asynchronous read request.
475 *
476 * @return VBox status code.
477 * @param pvUser The opaque user data passed on container creation.
478 * @param pStorage The storage handle.
479 * @param uOffset The offset to start reading from.
480 * @param paSegments Scatter gather list to store the data in.
481 * @param cSegments Number of segments in the list.
482 * @param cbRead How many bytes to read.
483 * @param pvCompletion The opaque user data which is returned upon completion.
484 * @param ppTask Where to store the opaque task handle.
485 */
486 DECLR3CALLBACKMEMBER(int, pfnReadAsync, (void *pvUser, void *pStorage, uint64_t uOffset,
487 PCRTSGSEG paSegments, size_t cSegments,
488 size_t cbRead, void *pvCompletion,
489 void **ppTask));
490
491 /**
492 * Initiate an asynchronous write request.
493 *
494 * @return VBox status code.
495 * @param pvUser The opaque user data passed on conatiner creation.
496 * @param pStorage The storage handle.
497 * @param uOffset The offset to start writing to.
498 * @param paSegments Scatter gather list of the data to write
499 * @param cSegments Number of segments in the list.
500 * @param cbWrite How many bytes to write.
501 * @param pvCompletion The opaque user data which is returned upon completion.
502 * @param ppTask Where to store the opaque task handle.
503 */
504 DECLR3CALLBACKMEMBER(int, pfnWriteAsync, (void *pvUser, void *pStorage, uint64_t uOffset,
505 PCRTSGSEG paSegments, size_t cSegments,
506 size_t cbWrite, void *pvCompletion,
507 void **ppTask));
508
509 /**
510 * Initiates an async flush request.
511 *
512 * @return VBox status code.
513 * @param pvUser The opaque data passed on container creation.
514 * @param pStorage The storage handle to flush.
515 * @param pvCompletion The opaque user data which is returned upon completion.
516 * @param ppTask Where to store the opaque task handle.
517 */
518 DECLR3CALLBACKMEMBER(int, pfnFlushAsync, (void *pvUser, void *pStorage,
519 void *pvCompletion, void **ppTask));
520
521} VDINTERFACEIO, *PVDINTERFACEIO;
522
523/**
524 * Get I/O interface from interface list.
525 *
526 * @return Pointer to the first I/O interface in the list.
527 * @param pVDIfs Pointer to the interface list.
528 */
529DECLINLINE(PVDINTERFACEIO) VDIfIoGet(PVDINTERFACE pVDIfs)
530{
531 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_IO);
532
533 /* Check that the interface descriptor is a progress interface. */
534 AssertMsgReturn( !pIf
535 || ( (pIf->enmInterface == VDINTERFACETYPE_IO)
536 && (pIf->cbSize == sizeof(VDINTERFACEIO))),
537 ("Not a I/O interface"), NULL);
538
539 return (PVDINTERFACEIO)pIf;
540}
541
542DECLINLINE(int) vdIfIoFileOpen(PVDINTERFACEIO pIfIo, const char *pszFilename,
543 uint32_t fOpen, PFNVDCOMPLETED pfnCompleted,
544 void **ppStorage)
545{
546 return pIfIo->pfnOpen(pIfIo->Core.pvUser, pszFilename, fOpen, pfnCompleted, ppStorage);
547}
548
549DECLINLINE(int) vdIfIoFileClose(PVDINTERFACEIO pIfIo, void *pStorage)
550{
551 return pIfIo->pfnClose(pIfIo->Core.pvUser, pStorage);
552}
553
554DECLINLINE(int) vdIfIoFileDelete(PVDINTERFACEIO pIfIo, const char *pszFilename)
555{
556 return pIfIo->pfnDelete(pIfIo->Core.pvUser, pszFilename);
557}
558
559DECLINLINE(int) vdIfIoFileMove(PVDINTERFACEIO pIfIo, const char *pszSrc,
560 const char *pszDst, unsigned fMove)
561{
562 return pIfIo->pfnMove(pIfIo->Core.pvUser, pszSrc, pszDst, fMove);
563}
564
565DECLINLINE(int) vdIfIoFileGetFreeSpace(PVDINTERFACEIO pIfIo, const char *pszFilename,
566 int64_t *pcbFree)
567{
568 return pIfIo->pfnGetFreeSpace(pIfIo->Core.pvUser, pszFilename, pcbFree);
569}
570
571DECLINLINE(int) vdIfIoFileGetModificationTime(PVDINTERFACEIO pIfIo, const char *pcszFilename,
572 PRTTIMESPEC pModificationTime)
573{
574 return pIfIo->pfnGetModificationTime(pIfIo->Core.pvUser, pcszFilename,
575 pModificationTime);
576}
577
578DECLINLINE(int) vdIfIoFileGetSize(PVDINTERFACEIO pIfIo, void *pStorage,
579 uint64_t *pcbSize)
580{
581 return pIfIo->pfnGetSize(pIfIo->Core.pvUser, pStorage, pcbSize);
582}
583
584DECLINLINE(int) vdIfIoFileSetSize(PVDINTERFACEIO pIfIo, void *pStorage,
585 uint64_t cbSize)
586{
587 return pIfIo->pfnSetSize(pIfIo->Core.pvUser, pStorage, cbSize);
588}
589
590DECLINLINE(int) vdIfIoFileWriteSync(PVDINTERFACEIO pIfIo, void *pStorage,
591 uint64_t uOffset, const void *pvBuffer, size_t cbBuffer,
592 size_t *pcbWritten)
593{
594 return pIfIo->pfnWriteSync(pIfIo->Core.pvUser, pStorage, uOffset,
595 pvBuffer, cbBuffer, pcbWritten);
596}
597
598DECLINLINE(int) vdIfIoFileReadSync(PVDINTERFACEIO pIfIo, void *pStorage,
599 uint64_t uOffset, void *pvBuffer, size_t cbBuffer,
600 size_t *pcbRead)
601{
602 return pIfIo->pfnReadSync(pIfIo->Core.pvUser, pStorage, uOffset,
603 pvBuffer, cbBuffer, pcbRead);
604}
605
606DECLINLINE(int) vdIfIoFileFlushSync(PVDINTERFACEIO pIfIo, void *pStorage)
607{
608 return pIfIo->pfnFlushSync(pIfIo->Core.pvUser, pStorage);
609}
610
611/**
612 * Create a VFS stream handle around a VD I/O interface.
613 *
614 * The I/O interface will not be closed or free by the stream, the caller will
615 * do so after it is done with the stream and has released the instances of the
616 * I/O stream object returned by this API.
617 *
618 * @return VBox status code.
619 * @param pVDIfsIo Pointer to the VD I/O interface.
620 * @param pvStorage The storage argument to pass to the interface
621 * methods.
622 * @param fFlags RTFILE_O_XXX, access mask requied.
623 * @param phVfsIos Where to return the VFS I/O stream handle on
624 * success.
625 */
626VBOXDDU_DECL(int) VDIfCreateVfsStream(PVDINTERFACEIO pVDIfsIo, void *pvStorage, uint32_t fFlags, PRTVFSIOSTREAM phVfsIos);
627
628/**
629 * Create a VFS file handle around a VD I/O interface.
630 *
631 * The I/O interface will not be closed or free by the VFS file, the caller will
632 * do so after it is done with the VFS file and has released the instances of
633 * the VFS object returned by this API.
634 *
635 * @return VBox status code.
636 * @param pVDIfs Pointer to the VD I/O interface. If NULL, then @a
637 * pVDIfsInt must be specified.
638 * @param pVDIfsInt Pointer to the internal VD I/O interface. If NULL,
639 * then @ pVDIfs must be specified.
640 * @param pvStorage The storage argument to pass to the interface
641 * methods.
642 * @param fFlags RTFILE_O_XXX, access mask requied.
643 * @param phVfsFile Where to return the VFS file handle on success.
644 */
645VBOXDDU_DECL(int) VDIfCreateVfsFile(PVDINTERFACEIO pVDIfs, struct VDINTERFACEIOINT *pVDIfsInt, void *pvStorage, uint32_t fFlags, PRTVFSFILE phVfsFile);
646
647
648/**
649 * Callback which provides progress information about a currently running
650 * lengthy operation.
651 *
652 * @return VBox status code.
653 * @param pvUser The opaque user data associated with this interface.
654 * @param uPercent Completion percentage.
655 */
656typedef DECLCALLBACK(int) FNVDPROGRESS(void *pvUser, unsigned uPercentage);
657/** Pointer to FNVDPROGRESS() */
658typedef FNVDPROGRESS *PFNVDPROGRESS;
659
660/**
661 * Progress notification interface
662 *
663 * Per-operation. Optional.
664 */
665typedef struct VDINTERFACEPROGRESS
666{
667 /**
668 * Common interface header.
669 */
670 VDINTERFACE Core;
671
672 /**
673 * Progress notification callbacks.
674 */
675 PFNVDPROGRESS pfnProgress;
676
677} VDINTERFACEPROGRESS, *PVDINTERFACEPROGRESS;
678
679/**
680 * Get progress interface from interface list.
681 *
682 * @return Pointer to the first progress interface in the list.
683 * @param pVDIfs Pointer to the interface list.
684 */
685DECLINLINE(PVDINTERFACEPROGRESS) VDIfProgressGet(PVDINTERFACE pVDIfs)
686{
687 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_PROGRESS);
688
689 /* Check that the interface descriptor is a progress interface. */
690 AssertMsgReturn( !pIf
691 || ( (pIf->enmInterface == VDINTERFACETYPE_PROGRESS)
692 && (pIf->cbSize == sizeof(VDINTERFACEPROGRESS))),
693 ("Not a progress interface"), NULL);
694
695 return (PVDINTERFACEPROGRESS)pIf;
696}
697
698
699/**
700 * Configuration information interface
701 *
702 * Per-image. Optional for most backends, but mandatory for images which do
703 * not operate on files (including standard block or character devices).
704 */
705typedef struct VDINTERFACECONFIG
706{
707 /**
708 * Common interface header.
709 */
710 VDINTERFACE Core;
711
712 /**
713 * Validates that the keys are within a set of valid names.
714 *
715 * @return true if all key names are found in pszzAllowed.
716 * @return false if not.
717 * @param pvUser The opaque user data associated with this interface.
718 * @param pszzValid List of valid key names separated by '\\0' and ending with
719 * a double '\\0'.
720 */
721 DECLR3CALLBACKMEMBER(bool, pfnAreKeysValid, (void *pvUser, const char *pszzValid));
722
723 /**
724 * Retrieves the length of the string value associated with a key (including
725 * the terminator, for compatibility with CFGMR3QuerySize).
726 *
727 * @return VBox status code.
728 * VERR_CFGM_VALUE_NOT_FOUND means that the key is not known.
729 * @param pvUser The opaque user data associated with this interface.
730 * @param pszName Name of the key to query.
731 * @param pcbValue Where to store the value length. Non-NULL.
732 */
733 DECLR3CALLBACKMEMBER(int, pfnQuerySize, (void *pvUser, const char *pszName, size_t *pcbValue));
734
735 /**
736 * Query the string value associated with a key.
737 *
738 * @return VBox status code.
739 * VERR_CFGM_VALUE_NOT_FOUND means that the key is not known.
740 * VERR_CFGM_NOT_ENOUGH_SPACE means that the buffer is not big enough.
741 * @param pvUser The opaque user data associated with this interface.
742 * @param pszName Name of the key to query.
743 * @param pszValue Pointer to buffer where to store value.
744 * @param cchValue Length of value buffer.
745 */
746 DECLR3CALLBACKMEMBER(int, pfnQuery, (void *pvUser, const char *pszName, char *pszValue, size_t cchValue));
747
748 /**
749 * Query the bytes value associated with a key.
750 *
751 * @return VBox status code.
752 * VERR_CFGM_VALUE_NOT_FOUND means that the key is not known.
753 * VERR_CFGM_NOT_ENOUGH_SPACE means that the buffer is not big enough.
754 * @param pvUser The opaque user data associated with this interface.
755 * @param pszName Name of the key to query.
756 * @param ppvData Pointer to buffer where to store the data.
757 * @param cbData Length of data buffer.
758 */
759 DECLR3CALLBACKMEMBER(int, pfnQueryBytes, (void *pvUser, const char *pszName, void *ppvData, size_t cbData));
760
761} VDINTERFACECONFIG, *PVDINTERFACECONFIG;
762
763/**
764 * Get configuration information interface from interface list.
765 *
766 * @return Pointer to the first configuration information interface in the list.
767 * @param pVDIfs Pointer to the interface list.
768 */
769DECLINLINE(PVDINTERFACECONFIG) VDIfConfigGet(PVDINTERFACE pVDIfs)
770{
771 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_CONFIG);
772
773 /* Check that the interface descriptor is a progress interface. */
774 AssertMsgReturn( !pIf
775 || ( (pIf->enmInterface == VDINTERFACETYPE_CONFIG)
776 && (pIf->cbSize == sizeof(VDINTERFACECONFIG))),
777 ("Not a config interface"), NULL);
778
779 return (PVDINTERFACECONFIG)pIf;
780}
781
782/**
783 * Query configuration, validates that the keys are within a set of valid names.
784 *
785 * @return true if all key names are found in pszzAllowed.
786 * @return false if not.
787 * @param pCfgIf Pointer to configuration callback table.
788 * @param pszzValid List of valid names separated by '\\0' and ending with
789 * a double '\\0'.
790 */
791DECLINLINE(bool) VDCFGAreKeysValid(PVDINTERFACECONFIG pCfgIf, const char *pszzValid)
792{
793 return pCfgIf->pfnAreKeysValid(pCfgIf->Core.pvUser, pszzValid);
794}
795
796/**
797 * Checks whether a given key is existing.
798 *
799 * @return true if the key exists.
800 * @return false if the key does not exist.
801 * @param pCfgIf Pointer to configuration callback table.
802 * @param pszName Name of the key.
803 */
804DECLINLINE(bool) VDCFGIsKeyExisting(PVDINTERFACECONFIG pCfgIf, const char *pszName)
805{
806 size_t cb = 0;
807 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
808 return rc == VERR_CFGM_VALUE_NOT_FOUND ? false : true;
809}
810
811/**
812 * Query configuration, unsigned 64-bit integer value with default.
813 *
814 * @return VBox status code.
815 * @param pCfgIf Pointer to configuration callback table.
816 * @param pszName Name of an integer value
817 * @param pu64 Where to store the value. Set to default on failure.
818 * @param u64Def The default value.
819 */
820DECLINLINE(int) VDCFGQueryU64Def(PVDINTERFACECONFIG pCfgIf,
821 const char *pszName, uint64_t *pu64,
822 uint64_t u64Def)
823{
824 char aszBuf[32];
825 int rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, aszBuf, sizeof(aszBuf));
826 if (RT_SUCCESS(rc))
827 {
828 rc = RTStrToUInt64Full(aszBuf, 0, pu64);
829 }
830 else if (rc == VERR_CFGM_VALUE_NOT_FOUND)
831 {
832 rc = VINF_SUCCESS;
833 *pu64 = u64Def;
834 }
835 return rc;
836}
837
838/**
839 * Query configuration, unsigned 64-bit integer value.
840 *
841 * @return VBox status code.
842 * @param pCfgIf Pointer to configuration callback table.
843 * @param pszName Name of an integer value
844 * @param pu64 Where to store the value.
845 */
846DECLINLINE(int) VDCFGQueryU64(PVDINTERFACECONFIG pCfgIf, const char *pszName,
847 uint64_t *pu64)
848{
849 char aszBuf[32];
850 int rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, aszBuf, sizeof(aszBuf));
851 if (RT_SUCCESS(rc))
852 {
853 rc = RTStrToUInt64Full(aszBuf, 0, pu64);
854 }
855
856 return rc;
857}
858
859/**
860 * Query configuration, unsigned 32-bit integer value with default.
861 *
862 * @return VBox status code.
863 * @param pCfgIf Pointer to configuration callback table.
864 * @param pszName Name of an integer value
865 * @param pu32 Where to store the value. Set to default on failure.
866 * @param u32Def The default value.
867 */
868DECLINLINE(int) VDCFGQueryU32Def(PVDINTERFACECONFIG pCfgIf,
869 const char *pszName, uint32_t *pu32,
870 uint32_t u32Def)
871{
872 uint64_t u64;
873 int rc = VDCFGQueryU64Def(pCfgIf, pszName, &u64, u32Def);
874 if (RT_SUCCESS(rc))
875 {
876 if (!(u64 & UINT64_C(0xffffffff00000000)))
877 *pu32 = (uint32_t)u64;
878 else
879 rc = VERR_CFGM_INTEGER_TOO_BIG;
880 }
881 return rc;
882}
883
884/**
885 * Query configuration, bool value with default.
886 *
887 * @return VBox status code.
888 * @param pCfgIf Pointer to configuration callback table.
889 * @param pszName Name of an integer value
890 * @param pf Where to store the value. Set to default on failure.
891 * @param fDef The default value.
892 */
893DECLINLINE(int) VDCFGQueryBoolDef(PVDINTERFACECONFIG pCfgIf,
894 const char *pszName, bool *pf,
895 bool fDef)
896{
897 uint64_t u64;
898 int rc = VDCFGQueryU64Def(pCfgIf, pszName, &u64, fDef);
899 if (RT_SUCCESS(rc))
900 *pf = u64 ? true : false;
901 return rc;
902}
903
904/**
905 * Query configuration, bool value.
906 *
907 * @return VBox status code.
908 * @param pCfgIf Pointer to configuration callback table.
909 * @param pszName Name of an integer value
910 * @param pf Where to store the value.
911 */
912DECLINLINE(int) VDCFGQueryBool(PVDINTERFACECONFIG pCfgIf, const char *pszName,
913 bool *pf)
914{
915 uint64_t u64;
916 int rc = VDCFGQueryU64(pCfgIf, pszName, &u64);
917 if (RT_SUCCESS(rc))
918 *pf = u64 ? true : false;
919 return rc;
920}
921
922/**
923 * Query configuration, dynamically allocated (RTMemAlloc) zero terminated
924 * character value.
925 *
926 * @return VBox status code.
927 * @param pCfgIf Pointer to configuration callback table.
928 * @param pszName Name of an zero terminated character value
929 * @param ppszString Where to store the string pointer. Not set on failure.
930 * Free this using RTMemFree().
931 */
932DECLINLINE(int) VDCFGQueryStringAlloc(PVDINTERFACECONFIG pCfgIf,
933 const char *pszName, char **ppszString)
934{
935 size_t cb;
936 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
937 if (RT_SUCCESS(rc))
938 {
939 char *pszString = (char *)RTMemAlloc(cb);
940 if (pszString)
941 {
942 rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, pszString, cb);
943 if (RT_SUCCESS(rc))
944 *ppszString = pszString;
945 else
946 RTMemFree(pszString);
947 }
948 else
949 rc = VERR_NO_MEMORY;
950 }
951 return rc;
952}
953
954/**
955 * Query configuration, dynamically allocated (RTMemAlloc) zero terminated
956 * character value with default.
957 *
958 * @return VBox status code.
959 * @param pCfgIf Pointer to configuration callback table.
960 * @param pszName Name of an zero terminated character value
961 * @param ppszString Where to store the string pointer. Not set on failure.
962 * Free this using RTMemFree().
963 * @param pszDef The default value.
964 */
965DECLINLINE(int) VDCFGQueryStringAllocDef(PVDINTERFACECONFIG pCfgIf,
966 const char *pszName,
967 char **ppszString,
968 const char *pszDef)
969{
970 size_t cb;
971 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
972 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
973 {
974 cb = strlen(pszDef) + 1;
975 rc = VINF_SUCCESS;
976 }
977 if (RT_SUCCESS(rc))
978 {
979 char *pszString = (char *)RTMemAlloc(cb);
980 if (pszString)
981 {
982 rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, pszString, cb);
983 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
984 {
985 memcpy(pszString, pszDef, cb);
986 rc = VINF_SUCCESS;
987 }
988 if (RT_SUCCESS(rc))
989 *ppszString = pszString;
990 else
991 RTMemFree(pszString);
992 }
993 else
994 rc = VERR_NO_MEMORY;
995 }
996 return rc;
997}
998
999/**
1000 * Query configuration, dynamically allocated (RTMemAlloc) byte string value.
1001 *
1002 * @return VBox status code.
1003 * @param pCfgIf Pointer to configuration callback table.
1004 * @param pszName Name of an zero terminated character value
1005 * @param ppvData Where to store the byte string pointer. Not set on failure.
1006 * Free this using RTMemFree().
1007 * @param pcbData Where to store the byte string length.
1008 */
1009DECLINLINE(int) VDCFGQueryBytesAlloc(PVDINTERFACECONFIG pCfgIf,
1010 const char *pszName, void **ppvData, size_t *pcbData)
1011{
1012 size_t cb;
1013 int rc = pCfgIf->pfnQuerySize(pCfgIf->Core.pvUser, pszName, &cb);
1014 if (RT_SUCCESS(rc))
1015 {
1016 char *pbData;
1017 Assert(cb);
1018
1019 pbData = (char *)RTMemAlloc(cb);
1020 if (pbData)
1021 {
1022 if(pCfgIf->pfnQueryBytes)
1023 rc = pCfgIf->pfnQueryBytes(pCfgIf->Core.pvUser, pszName, pbData, cb);
1024 else
1025 rc = pCfgIf->pfnQuery(pCfgIf->Core.pvUser, pszName, pbData, cb);
1026
1027 if (RT_SUCCESS(rc))
1028 {
1029 *ppvData = pbData;
1030 /* Exclude terminator if the byte data was obtained using the string query callback. */
1031 *pcbData = cb;
1032 if (!pCfgIf->pfnQueryBytes)
1033 (*pcbData)--;
1034 }
1035 else
1036 RTMemFree(pbData);
1037 }
1038 else
1039 rc = VERR_NO_MEMORY;
1040 }
1041 return rc;
1042}
1043
1044/** Forward declaration of a VD socket. */
1045typedef struct VDSOCKETINT *VDSOCKET;
1046/** Pointer to a VD socket. */
1047typedef VDSOCKET *PVDSOCKET;
1048/** Nil socket handle. */
1049#define NIL_VDSOCKET ((VDSOCKET)0)
1050
1051/** Connect flag to indicate that the backend wants to use the extended
1052 * socket I/O multiplexing call. This might not be supported on all configurations
1053 * (internal networking and iSCSI)
1054 * and the backend needs to take appropriate action.
1055 */
1056#define VD_INTERFACETCPNET_CONNECT_EXTENDED_SELECT RT_BIT_32(0)
1057
1058/** @name Select events
1059 * @{ */
1060/** Readable without blocking. */
1061#define VD_INTERFACETCPNET_EVT_READ RT_BIT_32(0)
1062/** Writable without blocking. */
1063#define VD_INTERFACETCPNET_EVT_WRITE RT_BIT_32(1)
1064/** Error condition, hangup, exception or similar. */
1065#define VD_INTERFACETCPNET_EVT_ERROR RT_BIT_32(2)
1066/** Hint for the select that getting interrupted while waiting is more likely.
1067 * The interface implementation can optimize the waiting strategy based on this.
1068 * It is assumed that it is more likely to get one of the above socket events
1069 * instead of being interrupted if the flag is not set. */
1070#define VD_INTERFACETCPNET_HINT_INTERRUPT RT_BIT_32(3)
1071/** Mask of the valid bits. */
1072#define VD_INTERFACETCPNET_EVT_VALID_MASK UINT32_C(0x0000000f)
1073/** @} */
1074
1075/**
1076 * TCP network stack interface
1077 *
1078 * Per-image. Mandatory for backends which have the VD_CAP_TCPNET bit set.
1079 */
1080typedef struct VDINTERFACETCPNET
1081{
1082 /**
1083 * Common interface header.
1084 */
1085 VDINTERFACE Core;
1086
1087 /**
1088 * Creates a socket. The socket is not connected if this succeeds.
1089 *
1090 * @return iprt status code.
1091 * @retval VERR_NOT_SUPPORTED if the combination of flags is not supported.
1092 * @param fFlags Combination of the VD_INTERFACETCPNET_CONNECT_* #defines.
1093 * @param pSock Where to store the handle.
1094 */
1095 DECLR3CALLBACKMEMBER(int, pfnSocketCreate, (uint32_t fFlags, PVDSOCKET pSock));
1096
1097 /**
1098 * Destroys the socket.
1099 *
1100 * @return iprt status code.
1101 * @param Sock Socket descriptor.
1102 */
1103 DECLR3CALLBACKMEMBER(int, pfnSocketDestroy, (VDSOCKET Sock));
1104
1105 /**
1106 * Connect as a client to a TCP port.
1107 *
1108 * @return iprt status code.
1109 * @param Sock Socket descriptor.
1110 * @param pszAddress The address to connect to.
1111 * @param uPort The port to connect to.
1112 * @param cMillies Number of milliseconds to wait for the connect attempt to complete.
1113 * Use RT_INDEFINITE_WAIT to wait for ever.
1114 * Use RT_SOCKETCONNECT_DEFAULT_WAIT to wait for the default time
1115 * configured on the running system.
1116 */
1117 DECLR3CALLBACKMEMBER(int, pfnClientConnect, (VDSOCKET Sock, const char *pszAddress, uint32_t uPort,
1118 RTMSINTERVAL cMillies));
1119
1120 /**
1121 * Close a TCP connection.
1122 *
1123 * @return iprt status code.
1124 * @param Sock Socket descriptor.
1125 */
1126 DECLR3CALLBACKMEMBER(int, pfnClientClose, (VDSOCKET Sock));
1127
1128 /**
1129 * Returns whether the socket is currently connected to the client.
1130 *
1131 * @returns true if the socket is connected.
1132 * false otherwise.
1133 * @param Sock Socket descriptor.
1134 */
1135 DECLR3CALLBACKMEMBER(bool, pfnIsClientConnected, (VDSOCKET Sock));
1136
1137 /**
1138 * Socket I/O multiplexing.
1139 * Checks if the socket is ready for reading.
1140 *
1141 * @return iprt status code.
1142 * @param Sock Socket descriptor.
1143 * @param cMillies Number of milliseconds to wait for the socket.
1144 * Use RT_INDEFINITE_WAIT to wait for ever.
1145 */
1146 DECLR3CALLBACKMEMBER(int, pfnSelectOne, (VDSOCKET Sock, RTMSINTERVAL cMillies));
1147
1148 /**
1149 * Receive data from a socket.
1150 *
1151 * @return iprt status code.
1152 * @param Sock Socket descriptor.
1153 * @param pvBuffer Where to put the data we read.
1154 * @param cbBuffer Read buffer size.
1155 * @param pcbRead Number of bytes read.
1156 * If NULL the entire buffer will be filled upon successful return.
1157 * If not NULL a partial read can be done successfully.
1158 */
1159 DECLR3CALLBACKMEMBER(int, pfnRead, (VDSOCKET Sock, void *pvBuffer, size_t cbBuffer, size_t *pcbRead));
1160
1161 /**
1162 * Send data to a socket.
1163 *
1164 * @return iprt status code.
1165 * @param Sock Socket descriptor.
1166 * @param pvBuffer Buffer to write data to socket.
1167 * @param cbBuffer How much to write.
1168 */
1169 DECLR3CALLBACKMEMBER(int, pfnWrite, (VDSOCKET Sock, const void *pvBuffer, size_t cbBuffer));
1170
1171 /**
1172 * Send data from scatter/gather buffer to a socket.
1173 *
1174 * @return iprt status code.
1175 * @param Sock Socket descriptor.
1176 * @param pSgBuffer Scatter/gather buffer to write data to socket.
1177 */
1178 DECLR3CALLBACKMEMBER(int, pfnSgWrite, (VDSOCKET Sock, PCRTSGBUF pSgBuffer));
1179
1180 /**
1181 * Receive data from a socket - not blocking.
1182 *
1183 * @return iprt status code.
1184 * @param Sock Socket descriptor.
1185 * @param pvBuffer Where to put the data we read.
1186 * @param cbBuffer Read buffer size.
1187 * @param pcbRead Number of bytes read.
1188 */
1189 DECLR3CALLBACKMEMBER(int, pfnReadNB, (VDSOCKET Sock, void *pvBuffer, size_t cbBuffer, size_t *pcbRead));
1190
1191 /**
1192 * Send data to a socket - not blocking.
1193 *
1194 * @return iprt status code.
1195 * @param Sock Socket descriptor.
1196 * @param pvBuffer Buffer to write data to socket.
1197 * @param cbBuffer How much to write.
1198 * @param pcbWritten Number of bytes written.
1199 */
1200 DECLR3CALLBACKMEMBER(int, pfnWriteNB, (VDSOCKET Sock, const void *pvBuffer, size_t cbBuffer, size_t *pcbWritten));
1201
1202 /**
1203 * Send data from scatter/gather buffer to a socket - not blocking.
1204 *
1205 * @return iprt status code.
1206 * @param Sock Socket descriptor.
1207 * @param pSgBuffer Scatter/gather buffer to write data to socket.
1208 * @param pcbWritten Number of bytes written.
1209 */
1210 DECLR3CALLBACKMEMBER(int, pfnSgWriteNB, (VDSOCKET Sock, PRTSGBUF pSgBuffer, size_t *pcbWritten));
1211
1212 /**
1213 * Flush socket write buffers.
1214 *
1215 * @return iprt status code.
1216 * @param Sock Socket descriptor.
1217 */
1218 DECLR3CALLBACKMEMBER(int, pfnFlush, (VDSOCKET Sock));
1219
1220 /**
1221 * Enables or disables delaying sends to coalesce packets.
1222 *
1223 * @return iprt status code.
1224 * @param Sock Socket descriptor.
1225 * @param fEnable When set to true enables coalescing.
1226 */
1227 DECLR3CALLBACKMEMBER(int, pfnSetSendCoalescing, (VDSOCKET Sock, bool fEnable));
1228
1229 /**
1230 * Gets the address of the local side.
1231 *
1232 * @return iprt status code.
1233 * @param Sock Socket descriptor.
1234 * @param pAddr Where to store the local address on success.
1235 */
1236 DECLR3CALLBACKMEMBER(int, pfnGetLocalAddress, (VDSOCKET Sock, PRTNETADDR pAddr));
1237
1238 /**
1239 * Gets the address of the other party.
1240 *
1241 * @return iprt status code.
1242 * @param Sock Socket descriptor.
1243 * @param pAddr Where to store the peer address on success.
1244 */
1245 DECLR3CALLBACKMEMBER(int, pfnGetPeerAddress, (VDSOCKET Sock, PRTNETADDR pAddr));
1246
1247 /**
1248 * Socket I/O multiplexing - extended version which can be woken up.
1249 * Checks if the socket is ready for reading or writing.
1250 *
1251 * @return iprt status code.
1252 * @retval VERR_INTERRUPTED if the thread was woken up by a pfnPoke call.
1253 * @param Sock Socket descriptor.
1254 * @param fEvents Mask of events to wait for.
1255 * @param pfEvents Where to store the received events.
1256 * @param cMillies Number of milliseconds to wait for the socket.
1257 * Use RT_INDEFINITE_WAIT to wait for ever.
1258 */
1259 DECLR3CALLBACKMEMBER(int, pfnSelectOneEx, (VDSOCKET Sock, uint32_t fEvents,
1260 uint32_t *pfEvents, RTMSINTERVAL cMillies));
1261
1262 /**
1263 * Wakes up the thread waiting in pfnSelectOneEx.
1264 *
1265 * @return iprt status code.
1266 * @param Sock Socket descriptor.
1267 */
1268 DECLR3CALLBACKMEMBER(int, pfnPoke, (VDSOCKET Sock));
1269
1270} VDINTERFACETCPNET, *PVDINTERFACETCPNET;
1271
1272/**
1273 * Get TCP network stack interface from interface list.
1274 *
1275 * @return Pointer to the first TCP network stack interface in the list.
1276 * @param pVDIfs Pointer to the interface list.
1277 */
1278DECLINLINE(PVDINTERFACETCPNET) VDIfTcpNetGet(PVDINTERFACE pVDIfs)
1279{
1280 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_TCPNET);
1281
1282 /* Check that the interface descriptor is a progress interface. */
1283 AssertMsgReturn( !pIf
1284 || ( (pIf->enmInterface == VDINTERFACETYPE_TCPNET)
1285 && (pIf->cbSize == sizeof(VDINTERFACETCPNET))),
1286 ("Not a TCP net interface"), NULL);
1287
1288 return (PVDINTERFACETCPNET)pIf;
1289}
1290
1291
1292/**
1293 * Interface to synchronize concurrent accesses by several threads.
1294 *
1295 * @note The scope of this interface is to manage concurrent accesses after
1296 * the HDD container has been created, and they must stop before destroying the
1297 * container. Opening or closing images is covered by the synchronization, but
1298 * that does not mean it is safe to close images while a thread executes
1299 * <link to="VDMerge"/> or <link to="VDCopy"/> operating on these images.
1300 * Making them safe would require the lock to be held during the entire
1301 * operation, which prevents other concurrent acitivities.
1302 *
1303 * @note Right now this is kept as simple as possible, and does not even
1304 * attempt to provide enough information to allow e.g. concurrent write
1305 * accesses to different areas of the disk. The reason is that it is very
1306 * difficult to predict which area of a disk is affected by a write,
1307 * especially when different image formats are mixed. Maybe later a more
1308 * sophisticated interface will be provided which has the necessary information
1309 * about worst case affected areas.
1310 *
1311 * Per-disk interface. Optional, needed if the disk is accessed concurrently
1312 * by several threads, e.g. when merging diff images while a VM is running.
1313 */
1314typedef struct VDINTERFACETHREADSYNC
1315{
1316 /**
1317 * Common interface header.
1318 */
1319 VDINTERFACE Core;
1320
1321 /**
1322 * Start a read operation.
1323 */
1324 DECLR3CALLBACKMEMBER(int, pfnStartRead, (void *pvUser));
1325
1326 /**
1327 * Finish a read operation.
1328 */
1329 DECLR3CALLBACKMEMBER(int, pfnFinishRead, (void *pvUser));
1330
1331 /**
1332 * Start a write operation.
1333 */
1334 DECLR3CALLBACKMEMBER(int, pfnStartWrite, (void *pvUser));
1335
1336 /**
1337 * Finish a write operation.
1338 */
1339 DECLR3CALLBACKMEMBER(int, pfnFinishWrite, (void *pvUser));
1340
1341} VDINTERFACETHREADSYNC, *PVDINTERFACETHREADSYNC;
1342
1343/**
1344 * Get thread synchronization interface from interface list.
1345 *
1346 * @return Pointer to the first thread synchronization interface in the list.
1347 * @param pVDIfs Pointer to the interface list.
1348 */
1349DECLINLINE(PVDINTERFACETHREADSYNC) VDIfThreadSyncGet(PVDINTERFACE pVDIfs)
1350{
1351 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_THREADSYNC);
1352
1353 /* Check that the interface descriptor is a progress interface. */
1354 AssertMsgReturn( !pIf
1355 || ( (pIf->enmInterface == VDINTERFACETYPE_THREADSYNC)
1356 && (pIf->cbSize == sizeof(VDINTERFACETHREADSYNC))),
1357 ("Not a thread synchronization interface"), NULL);
1358
1359 return (PVDINTERFACETHREADSYNC)pIf;
1360}
1361
1362/**
1363 * Interface to query usage of disk ranges.
1364 *
1365 * Per-operation interface. Optional.
1366 */
1367typedef struct VDINTERFACEQUERYRANGEUSE
1368{
1369 /**
1370 * Common interface header.
1371 */
1372 VDINTERFACE Core;
1373
1374 /**
1375 * Query use of a disk range.
1376 */
1377 DECLR3CALLBACKMEMBER(int, pfnQueryRangeUse, (void *pvUser, uint64_t off, uint64_t cb,
1378 bool *pfUsed));
1379
1380} VDINTERFACEQUERYRANGEUSE, *PVDINTERFACEQUERYRANGEUSE;
1381
1382/**
1383 * Get query range use interface from interface list.
1384 *
1385 * @return Pointer to the first thread synchronization interface in the list.
1386 * @param pVDIfs Pointer to the interface list.
1387 */
1388DECLINLINE(PVDINTERFACEQUERYRANGEUSE) VDIfQueryRangeUseGet(PVDINTERFACE pVDIfs)
1389{
1390 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_QUERYRANGEUSE);
1391
1392 /* Check that the interface descriptor is a progress interface. */
1393 AssertMsgReturn( !pIf
1394 || ( (pIf->enmInterface == VDINTERFACETYPE_QUERYRANGEUSE)
1395 && (pIf->cbSize == sizeof(VDINTERFACEQUERYRANGEUSE))),
1396 ("Not a query range use interface"), NULL);
1397
1398 return (PVDINTERFACEQUERYRANGEUSE)pIf;
1399}
1400
1401DECLINLINE(int) vdIfQueryRangeUse(PVDINTERFACEQUERYRANGEUSE pIfQueryRangeUse, uint64_t off, uint64_t cb,
1402 bool *pfUsed)
1403{
1404 return pIfQueryRangeUse->pfnQueryRangeUse(pIfQueryRangeUse->Core.pvUser, off, cb, pfUsed);
1405}
1406
1407
1408/**
1409 * Interface used to retrieve keys for cryptographic operations.
1410 *
1411 * Per-module interface. Optional but cryptographic modules might fail and
1412 * return an error if this is not present.
1413 */
1414typedef struct VDINTERFACECRYPTO
1415{
1416 /**
1417 * Common interface header.
1418 */
1419 VDINTERFACE Core;
1420
1421 /**
1422 * Retains a key identified by the ID. The caller will only hold a reference
1423 * to the key and must not modify the key buffer in any way.
1424 *
1425 * @returns VBox status code.
1426 * @param pvUser The opaque user data associated with this interface.
1427 * @param pszId The alias/id for the key to retrieve.
1428 * @param ppbKey Where to store the pointer to the key buffer on success.
1429 * @param pcbKey Where to store the size of the key in bytes on success.
1430 */
1431 DECLR3CALLBACKMEMBER(int, pfnKeyRetain, (void *pvUser, const char *pszId, const uint8_t **ppbKey, size_t *pcbKey));
1432
1433 /**
1434 * Releases one reference of the key identified by the given identifier.
1435 * The caller must not access the key buffer after calling this operation.
1436 *
1437 * @returns VBox status code.
1438 * @param pvUser The opaque user data associated with this interface.
1439 * @param pszId The alias/id for the key to release.
1440 *
1441 * @note: It is advised to release the key whenever it is not used anymore so the entity
1442 * storing the key can do anything to make retrieving the key from memory more
1443 * difficult like scrambling the memory buffer for instance.
1444 */
1445 DECLR3CALLBACKMEMBER(int, pfnKeyRelease, (void *pvUser, const char *pszId));
1446
1447 /**
1448 * Gets a reference to the password identified by the given ID to open a key store supplied through the config interface.
1449 *
1450 * @returns VBox status code.
1451 * @param pvUser The opaque user data associated with this interface.
1452 * @param pszId The alias/id for the password to retain.
1453 * @param ppszPassword Where to store the password to unlock the key store on success.
1454 */
1455 DECLR3CALLBACKMEMBER(int, pfnKeyStorePasswordRetain, (void *pvUser, const char *pszId, const char **ppszPassword));
1456
1457 /**
1458 * Releases a reference of the password previously acquired with VDINTERFACECRYPTO::pfnKeyStorePasswordRetain()
1459 * identified by the given ID.
1460 *
1461 * @returns VBox status code.
1462 * @param pvUser The opaque user data associated with this interface.
1463 * @param pszId The alias/id for the password to release.
1464 */
1465 DECLR3CALLBACKMEMBER(int, pfnKeyStorePasswordRelease, (void *pvUser, const char *pszId));
1466
1467 /**
1468 * Saves a key store.
1469 *
1470 * @returns VBox status code.
1471 * @param pvUser The opaque user data associated with this interface.
1472 * @param pvKeyStore The key store to save.
1473 * @param cbKeyStore Size of the key store in bytes.
1474 *
1475 * @note The format is filter specific and should be treated as binary data.
1476 */
1477 DECLR3CALLBACKMEMBER(int, pfnKeyStoreSave, (void *pvUser, const void *pvKeyStore, size_t cbKeyStore));
1478
1479 /**
1480 * Returns the parameters after the key store was loaded successfully.
1481 *
1482 * @returns VBox status code.
1483 * @param pvUser The opaque user data associated with this interface.
1484 * @param pszCipher The cipher identifier the DEK is used for.
1485 * @param pbDek The raw DEK which was contained in the key store loaded by
1486 * VDINTERFACECRYPTO::pfnKeyStoreLoad().
1487 * @param cbDek The size of the DEK.
1488 *
1489 * @note The provided pointer to the DEK is only valid until this call returns.
1490 * The content might change afterwards with out notice (when scrambling the key
1491 * for further protection for example) or might be even freed.
1492 *
1493 * @note This method is optional and can be NULL if the caller does not require the
1494 * parameters.
1495 */
1496 DECLR3CALLBACKMEMBER(int, pfnKeyStoreReturnParameters, (void *pvUser, const char *pszCipher,
1497 const uint8_t *pbDek, size_t cbDek));
1498
1499} VDINTERFACECRYPTO, *PVDINTERFACECRYPTO;
1500
1501
1502/**
1503 * Get error interface from interface list.
1504 *
1505 * @return Pointer to the first error interface in the list.
1506 * @param pVDIfs Pointer to the interface list.
1507 */
1508DECLINLINE(PVDINTERFACECRYPTO) VDIfCryptoGet(PVDINTERFACE pVDIfs)
1509{
1510 PVDINTERFACE pIf = VDInterfaceGet(pVDIfs, VDINTERFACETYPE_CRYPTO);
1511
1512 /* Check that the interface descriptor is a crypto interface. */
1513 AssertMsgReturn( !pIf
1514 || ( (pIf->enmInterface == VDINTERFACETYPE_CRYPTO)
1515 && (pIf->cbSize == sizeof(VDINTERFACECRYPTO))),
1516 ("Not an crypto interface\n"), NULL);
1517
1518 return (PVDINTERFACECRYPTO)pIf;
1519}
1520
1521/**
1522 * @copydoc VDINTERFACECRYPTO::pfnKeyRetain
1523 */
1524DECLINLINE(int) vdIfCryptoKeyRetain(PVDINTERFACECRYPTO pIfCrypto, const char *pszId, const uint8_t **ppbKey, size_t *pcbKey)
1525{
1526 return pIfCrypto->pfnKeyRetain(pIfCrypto->Core.pvUser, pszId, ppbKey, pcbKey);
1527}
1528
1529/**
1530 * @copydoc VDINTERFACECRYPTO::pfnKeyRelease
1531 */
1532DECLINLINE(int) vdIfCryptoKeyRelease(PVDINTERFACECRYPTO pIfCrypto, const char *pszId)
1533{
1534 return pIfCrypto->pfnKeyRelease(pIfCrypto->Core.pvUser, pszId);
1535}
1536
1537/**
1538 * @copydoc VDINTERFACECRYPTO::pfnKeyStorePasswordRetain
1539 */
1540DECLINLINE(int) vdIfCryptoKeyStorePasswordRetain(PVDINTERFACECRYPTO pIfCrypto, const char *pszId, const char **ppszPassword)
1541{
1542 return pIfCrypto->pfnKeyStorePasswordRetain(pIfCrypto->Core.pvUser, pszId, ppszPassword);
1543}
1544
1545/**
1546 * @copydoc VDINTERFACECRYPTO::pfnKeyStorePasswordRelease
1547 */
1548DECLINLINE(int) vdIfCryptoKeyStorePasswordRelease(PVDINTERFACECRYPTO pIfCrypto, const char *pszId)
1549{
1550 return pIfCrypto->pfnKeyStorePasswordRelease(pIfCrypto->Core.pvUser, pszId);
1551}
1552
1553/**
1554 * @copydoc VDINTERFACECRYPTO::pfnKeyStoreSave
1555 */
1556DECLINLINE(int) vdIfCryptoKeyStoreSave(PVDINTERFACECRYPTO pIfCrypto, const void *pvKeyStore, size_t cbKeyStore)
1557{
1558 return pIfCrypto->pfnKeyStoreSave(pIfCrypto->Core.pvUser, pvKeyStore, cbKeyStore);
1559}
1560
1561/**
1562 * @copydoc VDINTERFACECRYPTO::pfnKeyStoreReturnParameters
1563 */
1564DECLINLINE(int) vdIfCryptoKeyStoreReturnParameters(PVDINTERFACECRYPTO pIfCrypto, const char *pszCipher,
1565 const uint8_t *pbDek, size_t cbDek)
1566{
1567 if (pIfCrypto->pfnKeyStoreReturnParameters)
1568 return pIfCrypto->pfnKeyStoreReturnParameters(pIfCrypto->Core.pvUser, pszCipher, pbDek, cbDek);
1569
1570 return VINF_SUCCESS;
1571}
1572
1573
1574RT_C_DECLS_END
1575
1576/** @} */
1577
1578#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