VirtualBox

source: vbox/trunk/include/VBox/vmm/pdmstorageifs.h@ 69759

Last change on this file since 69759 was 69475, checked in by vboxsync, 7 years ago

*: scm updates - header files should have 'svn:keywords=Id Revision' too (doesn't mean they have to use them).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id Revision
File size: 44.5 KB
Line 
1/** @file
2 * PDM - Pluggable Device Manager, Storage related interfaces.
3 */
4
5/*
6 * Copyright (C) 2006-2017 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_vmm_pdmstorageifs_h
27#define ___VBox_vmm_pdmstorageifs_h
28
29#include <iprt/sg.h>
30#include <VBox/types.h>
31#include <VBox/vdmedia.h>
32
33RT_C_DECLS_BEGIN
34
35struct PDMISECKEY;
36struct PDMISECKEYHLP;
37
38
39/** @defgroup grp_pdm_ifs_storage PDM Storage Interfaces
40 * @ingroup grp_pdm_interfaces
41 * @{
42 */
43
44
45/** Pointer to a mount interface. */
46typedef struct PDMIMOUNTNOTIFY *PPDMIMOUNTNOTIFY;
47/**
48 * Block interface (up).
49 * Pair with PDMIMOUNT.
50 */
51typedef struct PDMIMOUNTNOTIFY
52{
53 /**
54 * Called when a media is mounted.
55 *
56 * @param pInterface Pointer to the interface structure containing the called function pointer.
57 * @thread The emulation thread.
58 */
59 DECLR3CALLBACKMEMBER(void, pfnMountNotify,(PPDMIMOUNTNOTIFY pInterface));
60
61 /**
62 * Called when a media is unmounted
63 * @param pInterface Pointer to the interface structure containing the called function pointer.
64 * @thread The emulation thread.
65 */
66 DECLR3CALLBACKMEMBER(void, pfnUnmountNotify,(PPDMIMOUNTNOTIFY pInterface));
67} PDMIMOUNTNOTIFY;
68/** PDMIMOUNTNOTIFY interface ID. */
69#define PDMIMOUNTNOTIFY_IID "fa143ac9-9fc6-498e-997f-945380a558f9"
70
71
72/** Pointer to mount interface. */
73typedef struct PDMIMOUNT *PPDMIMOUNT;
74/**
75 * Mount interface (down).
76 * Pair with PDMIMOUNTNOTIFY.
77 */
78typedef struct PDMIMOUNT
79{
80 /**
81 * Unmount the media.
82 *
83 * The driver will validate and pass it on. On the rebounce it will decide whether or not to detach it self.
84 *
85 * @returns VBox status code.
86 * @param pInterface Pointer to the interface structure containing the called function pointer.
87 * @thread The emulation thread.
88 * @param fForce Force the unmount, even for locked media.
89 * @param fEject Eject the medium. Only relevant for host drives.
90 * @thread The emulation thread.
91 */
92 DECLR3CALLBACKMEMBER(int, pfnUnmount,(PPDMIMOUNT pInterface, bool fForce, bool fEject));
93
94 /**
95 * Checks if a media is mounted.
96 *
97 * @returns true if mounted.
98 * @returns false if not mounted.
99 * @param pInterface Pointer to the interface structure containing the called function pointer.
100 * @thread Any thread.
101 */
102 DECLR3CALLBACKMEMBER(bool, pfnIsMounted,(PPDMIMOUNT pInterface));
103
104 /**
105 * Locks the media, preventing any unmounting of it.
106 *
107 * @returns VBox status code.
108 * @param pInterface Pointer to the interface structure containing the called function pointer.
109 * @thread The emulation thread.
110 */
111 DECLR3CALLBACKMEMBER(int, pfnLock,(PPDMIMOUNT pInterface));
112
113 /**
114 * Unlocks the media, canceling previous calls to pfnLock().
115 *
116 * @returns VBox status code.
117 * @param pInterface Pointer to the interface structure containing the called function pointer.
118 * @thread The emulation thread.
119 */
120 DECLR3CALLBACKMEMBER(int, pfnUnlock,(PPDMIMOUNT pInterface));
121
122 /**
123 * Checks if a media is locked.
124 *
125 * @returns true if locked.
126 * @returns false if not locked.
127 * @param pInterface Pointer to the interface structure containing the called function pointer.
128 * @thread Any thread.
129 */
130 DECLR3CALLBACKMEMBER(bool, pfnIsLocked,(PPDMIMOUNT pInterface));
131} PDMIMOUNT;
132/** PDMIMOUNT interface ID. */
133#define PDMIMOUNT_IID "34fc7a4c-623a-4806-a6bf-5be1be33c99f"
134
135
136/**
137 * Callback which provides progress information.
138 *
139 * @return VBox status code.
140 * @param pvUser Opaque user data.
141 * @param uPercent Completion percentage.
142 */
143typedef DECLCALLBACK(int) FNSIMPLEPROGRESS(void *pvUser, unsigned uPercentage);
144/** Pointer to FNSIMPLEPROGRESS() */
145typedef FNSIMPLEPROGRESS *PFNSIMPLEPROGRESS;
146
147
148/**
149 * Media type.
150 */
151typedef enum PDMMEDIATYPE
152{
153 /** Error (for the query function). */
154 PDMMEDIATYPE_ERROR = 1,
155 /** 360KB 5 1/4" floppy drive. */
156 PDMMEDIATYPE_FLOPPY_360,
157 /** 720KB 3 1/2" floppy drive. */
158 PDMMEDIATYPE_FLOPPY_720,
159 /** 1.2MB 5 1/4" floppy drive. */
160 PDMMEDIATYPE_FLOPPY_1_20,
161 /** 1.44MB 3 1/2" floppy drive. */
162 PDMMEDIATYPE_FLOPPY_1_44,
163 /** 2.88MB 3 1/2" floppy drive. */
164 PDMMEDIATYPE_FLOPPY_2_88,
165 /** Fake drive that can take up to 15.6 MB images.
166 * C=255, H=2, S=63. */
167 PDMMEDIATYPE_FLOPPY_FAKE_15_6,
168 /** Fake drive that can take up to 63.5 MB images.
169 * C=255, H=2, S=255. */
170 PDMMEDIATYPE_FLOPPY_FAKE_63_5,
171 /** CDROM drive. */
172 PDMMEDIATYPE_CDROM,
173 /** DVD drive. */
174 PDMMEDIATYPE_DVD,
175 /** Hard disk drive. */
176 PDMMEDIATYPE_HARD_DISK
177} PDMMEDIATYPE;
178
179/** Check if the given block type is a floppy. */
180#define PDMMEDIATYPE_IS_FLOPPY(a_enmType) ( (a_enmType) >= PDMMEDIATYPE_FLOPPY_360 && (a_enmType) <= PDMMEDIATYPE_FLOPPY_2_88 )
181
182/**
183 * Raw command data transfer direction.
184 */
185typedef enum PDMMEDIATXDIR
186{
187 PDMMEDIATXDIR_NONE = 0,
188 PDMMEDIATXDIR_FROM_DEVICE,
189 PDMMEDIATXDIR_TO_DEVICE
190} PDMMEDIATXDIR;
191
192/**
193 * Media geometry structure.
194 */
195typedef struct PDMMEDIAGEOMETRY
196{
197 /** Number of cylinders. */
198 uint32_t cCylinders;
199 /** Number of heads. */
200 uint32_t cHeads;
201 /** Number of sectors. */
202 uint32_t cSectors;
203} PDMMEDIAGEOMETRY;
204
205/** Pointer to media geometry structure. */
206typedef PDMMEDIAGEOMETRY *PPDMMEDIAGEOMETRY;
207/** Pointer to constant media geometry structure. */
208typedef const PDMMEDIAGEOMETRY *PCPDMMEDIAGEOMETRY;
209
210/** Pointer to a media port interface. */
211typedef struct PDMIMEDIAPORT *PPDMIMEDIAPORT;
212/**
213 * Media port interface (down).
214 */
215typedef struct PDMIMEDIAPORT
216{
217 /**
218 * Returns the storage controller name, instance and LUN of the attached medium.
219 *
220 * @returns VBox status.
221 * @param pInterface Pointer to this interface.
222 * @param ppcszController Where to store the name of the storage controller.
223 * @param piInstance Where to store the instance number of the controller.
224 * @param piLUN Where to store the LUN of the attached device.
225 */
226 DECLR3CALLBACKMEMBER(int, pfnQueryDeviceLocation, (PPDMIMEDIAPORT pInterface, const char **ppcszController,
227 uint32_t *piInstance, uint32_t *piLUN));
228
229} PDMIMEDIAPORT;
230/** PDMIMEDIAPORT interface ID. */
231#define PDMIMEDIAPORT_IID "9f7e8c9e-6d35-4453-bbef-1f78033174d6"
232
233/** Pointer to a media interface. */
234typedef struct PDMIMEDIA *PPDMIMEDIA;
235/**
236 * Media interface (up).
237 * Pairs with PDMIMEDIAPORT.
238 */
239typedef struct PDMIMEDIA
240{
241 /**
242 * Read bits.
243 *
244 * @returns VBox status code.
245 * @param pInterface Pointer to the interface structure containing the called function pointer.
246 * @param off Offset to start reading from. The offset must be aligned to a sector boundary.
247 * @param pvBuf Where to store the read bits.
248 * @param cbRead Number of bytes to read. Must be aligned to a sector boundary.
249 * @thread Any thread.
250 */
251 DECLR3CALLBACKMEMBER(int, pfnRead,(PPDMIMEDIA pInterface, uint64_t off, void *pvBuf, size_t cbRead));
252
253 /**
254 * Read bits - version for DevPcBios.
255 *
256 * @returns VBox status code.
257 * @param pInterface Pointer to the interface structure containing the called function pointer.
258 * @param off Offset to start reading from. The offset must be aligned to a sector boundary.
259 * @param pvBuf Where to store the read bits.
260 * @param cbRead Number of bytes to read. Must be aligned to a sector boundary.
261 * @thread Any thread.
262 *
263 * @note: Special version of pfnRead which doesn't try to suspend the VM when the DEKs for encrypted disks
264 * are missing but just returns an error.
265 */
266 DECLR3CALLBACKMEMBER(int, pfnReadPcBios,(PPDMIMEDIA pInterface, uint64_t off, void *pvBuf, size_t cbRead));
267
268 /**
269 * Write bits.
270 *
271 * @returns VBox status code.
272 * @param pInterface Pointer to the interface structure containing the called function pointer.
273 * @param off Offset to start writing at. The offset must be aligned to a sector boundary.
274 * @param pvBuf Where to store the write bits.
275 * @param cbWrite Number of bytes to write. Must be aligned to a sector boundary.
276 * @thread Any thread.
277 */
278 DECLR3CALLBACKMEMBER(int, pfnWrite,(PPDMIMEDIA pInterface, uint64_t off, const void *pvBuf, size_t cbWrite));
279
280 /**
281 * Make sure that the bits written are actually on the storage medium.
282 *
283 * @returns VBox status code.
284 * @param pInterface Pointer to the interface structure containing the called function pointer.
285 * @thread Any thread.
286 */
287 DECLR3CALLBACKMEMBER(int, pfnFlush,(PPDMIMEDIA pInterface));
288
289 /**
290 * Send a raw command to the underlying device (CDROM).
291 * This method is optional (i.e. the function pointer may be NULL).
292 *
293 * @returns VBox status code.
294 * @param pInterface Pointer to the interface structure containing the called function pointer.
295 * @param pbCdb The command to process.
296 * @param cbCdb The length of the command in bytes.
297 * @param enmTxDir Direction of transfer.
298 * @param pvBuf Pointer tp the transfer buffer.
299 * @param pcbBuf Size of the transfer buffer.
300 * @param pabSense Status of the command (when return value is VERR_DEV_IO_ERROR).
301 * @param cbSense Size of the sense buffer in bytes.
302 * @param cTimeoutMillies Command timeout in milliseconds.
303 * @thread Any thread.
304 */
305 DECLR3CALLBACKMEMBER(int, pfnSendCmd,(PPDMIMEDIA pInterface, const uint8_t *pbCdb, size_t cbCdb,
306 PDMMEDIATXDIR enmTxDir, void *pvBuf, uint32_t *pcbBuf,
307 uint8_t *pabSense, size_t cbSense, uint32_t cTimeoutMillies));
308
309 /**
310 * Merge medium contents during a live snapshot deletion. All details
311 * must have been configured through CFGM or this will fail.
312 * This method is optional (i.e. the function pointer may be NULL).
313 *
314 * @returns VBox status code.
315 * @param pInterface Pointer to the interface structure containing the called function pointer.
316 * @param pfnProgress Function pointer for progress notification.
317 * @param pvUser Opaque user data for progress notification.
318 * @thread Any thread.
319 */
320 DECLR3CALLBACKMEMBER(int, pfnMerge,(PPDMIMEDIA pInterface, PFNSIMPLEPROGRESS pfnProgress, void *pvUser));
321
322 /**
323 * Sets the secret key retrieval interface to use to get secret keys.
324 *
325 * @returns VBox status code.
326 * @param pInterface Pointer to the interface structure containing the called function pointer.
327 * @param pIfSecKey The secret key interface to use.
328 * Use NULL to clear the currently set interface and clear all secret
329 * keys from the user.
330 * @param pIfSecKeyHlp The secret key helper interface to use.
331 * @thread Any thread.
332 */
333 DECLR3CALLBACKMEMBER(int, pfnSetSecKeyIf,(PPDMIMEDIA pInterface, struct PDMISECKEY *pIfSecKey,
334 struct PDMISECKEYHLP *pIfSecKeyHlp));
335
336 /**
337 * Get the media size in bytes.
338 *
339 * @returns Media size in bytes.
340 * @param pInterface Pointer to the interface structure containing the called function pointer.
341 * @thread Any thread.
342 */
343 DECLR3CALLBACKMEMBER(uint64_t, pfnGetSize,(PPDMIMEDIA pInterface));
344
345 /**
346 * Gets the media sector size in bytes.
347 *
348 * @returns Media sector size in bytes.
349 * @param pInterface Pointer to the interface structure containing the called function pointer.
350 * @thread Any thread.
351 */
352 DECLR3CALLBACKMEMBER(uint32_t, pfnGetSectorSize,(PPDMIMEDIA pInterface));
353
354 /**
355 * Check if the media is readonly or not.
356 *
357 * @returns true if readonly.
358 * @returns false if read/write.
359 * @param pInterface Pointer to the interface structure containing the called function pointer.
360 * @thread Any thread.
361 */
362 DECLR3CALLBACKMEMBER(bool, pfnIsReadOnly,(PPDMIMEDIA pInterface));
363
364 /**
365 * Returns whether the medium should be marked as rotational or not.
366 *
367 * @returns true if non rotating medium.
368 * @returns false if rotating medium.
369 * @param pInterface Pointer to the interface structure containing the called function pointer.
370 * @thread Any thread.
371 */
372 DECLR3CALLBACKMEMBER(bool, pfnIsNonRotational,(PPDMIMEDIA pInterface));
373
374 /**
375 * Get stored media geometry (physical CHS, PCHS) - BIOS property.
376 * This is an optional feature of a media.
377 *
378 * @returns VBox status code.
379 * @returns VERR_NOT_IMPLEMENTED if the media doesn't support storing the geometry.
380 * @returns VERR_PDM_GEOMETRY_NOT_SET if the geometry hasn't been set using pfnBiosSetPCHSGeometry() yet.
381 * @param pInterface Pointer to the interface structure containing the called function pointer.
382 * @param pPCHSGeometry Pointer to PCHS geometry (cylinders/heads/sectors).
383 * @remark This has no influence on the read/write operations.
384 * @thread Any thread.
385 */
386 DECLR3CALLBACKMEMBER(int, pfnBiosGetPCHSGeometry,(PPDMIMEDIA pInterface, PPDMMEDIAGEOMETRY pPCHSGeometry));
387
388 /**
389 * Store the media geometry (physical CHS, PCHS) - BIOS property.
390 * This is an optional feature of a media.
391 *
392 * @returns VBox status code.
393 * @returns VERR_NOT_IMPLEMENTED if the media doesn't support storing the geometry.
394 * @param pInterface Pointer to the interface structure containing the called function pointer.
395 * @param pPCHSGeometry Pointer to PCHS geometry (cylinders/heads/sectors).
396 * @remark This has no influence on the read/write operations.
397 * @thread The emulation thread.
398 */
399 DECLR3CALLBACKMEMBER(int, pfnBiosSetPCHSGeometry,(PPDMIMEDIA pInterface, PCPDMMEDIAGEOMETRY pPCHSGeometry));
400
401 /**
402 * Get stored media geometry (logical CHS, LCHS) - BIOS property.
403 * This is an optional feature of a media.
404 *
405 * @returns VBox status code.
406 * @returns VERR_NOT_IMPLEMENTED if the media doesn't support storing the geometry.
407 * @returns VERR_PDM_GEOMETRY_NOT_SET if the geometry hasn't been set using pfnBiosSetLCHSGeometry() yet.
408 * @param pInterface Pointer to the interface structure containing the called function pointer.
409 * @param pLCHSGeometry Pointer to LCHS geometry (cylinders/heads/sectors).
410 * @remark This has no influence on the read/write operations.
411 * @thread Any thread.
412 */
413 DECLR3CALLBACKMEMBER(int, pfnBiosGetLCHSGeometry,(PPDMIMEDIA pInterface, PPDMMEDIAGEOMETRY pLCHSGeometry));
414
415 /**
416 * Store the media geometry (logical CHS, LCHS) - BIOS property.
417 * This is an optional feature of a media.
418 *
419 * @returns VBox status code.
420 * @returns VERR_NOT_IMPLEMENTED if the media doesn't support storing the geometry.
421 * @param pInterface Pointer to the interface structure containing the called function pointer.
422 * @param pLCHSGeometry Pointer to LCHS geometry (cylinders/heads/sectors).
423 * @remark This has no influence on the read/write operations.
424 * @thread The emulation thread.
425 */
426 DECLR3CALLBACKMEMBER(int, pfnBiosSetLCHSGeometry,(PPDMIMEDIA pInterface, PCPDMMEDIAGEOMETRY pLCHSGeometry));
427
428 /**
429 * Checks if the device should be visible to the BIOS or not.
430 *
431 * @returns true if the device is visible to the BIOS.
432 * @returns false if the device is not visible to the BIOS.
433 * @param pInterface Pointer to the interface structure containing the called function pointer.
434 * @thread Any thread.
435 */
436 DECLR3CALLBACKMEMBER(bool, pfnBiosIsVisible,(PPDMIMEDIA pInterface));
437
438 /**
439 * Gets the media type.
440 *
441 * @returns media type.
442 * @param pInterface Pointer to the interface structure containing the called function pointer.
443 * @thread Any thread.
444 */
445 DECLR3CALLBACKMEMBER(PDMMEDIATYPE, pfnGetType,(PPDMIMEDIA pInterface));
446
447 /**
448 * Gets the UUID of the media drive.
449 *
450 * @returns VBox status code.
451 * @param pInterface Pointer to the interface structure containing the called function pointer.
452 * @param pUuid Where to store the UUID on success.
453 * @thread Any thread.
454 */
455 DECLR3CALLBACKMEMBER(int, pfnGetUuid,(PPDMIMEDIA pInterface, PRTUUID pUuid));
456
457 /**
458 * Discards the given range.
459 *
460 * @returns VBox status code.
461 * @param pInterface Pointer to the interface structure containing the called function pointer.
462 * @param paRanges Array of ranges to discard.
463 * @param cRanges Number of entries in the array.
464 * @thread Any thread.
465 */
466 DECLR3CALLBACKMEMBER(int, pfnDiscard,(PPDMIMEDIA pInterface, PCRTRANGE paRanges, unsigned cRanges));
467
468 /**
469 * Returns the number of regions for the medium.
470 *
471 * @returns Number of regions.
472 * @param pInterface Pointer to the interface structure containing the called function pointer.
473 */
474 DECLR3CALLBACKMEMBER(uint32_t, pfnGetRegionCount,(PPDMIMEDIA pInterface));
475
476 /**
477 * Queries the properties for the given region.
478 *
479 * @returns VBox status code.
480 * @retval VERR_NOT_FOUND if the region index is not known.
481 * @param pInterface Pointer to the interface structure containing the called function pointer.
482 * @param uRegion The region index to query the properties of.
483 * @param pu64LbaStart Where to store the starting LBA for the region on success.
484 * @param pcBlocks Where to store the number of blocks for the region on success.
485 * @param pcbBlock Where to store the size of one block in bytes on success.
486 * @param penmDataForm WHere to store the data form for the region on success.
487 */
488 DECLR3CALLBACKMEMBER(int, pfnQueryRegionProperties,(PPDMIMEDIA pInterface, uint32_t uRegion, uint64_t *pu64LbaStart,
489 uint64_t *pcBlocks, uint64_t *pcbBlock,
490 PVDREGIONDATAFORM penmDataForm));
491
492 /**
493 * Queries the properties for the region covering the given LBA.
494 *
495 * @returns VBox status code.
496 * @retval VERR_NOT_FOUND if the region index is not known.
497 * @param pInterface Pointer to the interface structure containing the called function pointer.
498 * @param u64LbaStart Where to store the starting LBA for the region on success.
499 * @param puRegion Where to store the region number on success.
500 * @param pcBlocks Where to store the number of blocks left in this region starting from the given LBA.
501 * @param pcbBlock Where to store the size of one block in bytes on success.
502 * @param penmDataForm WHere to store the data form for the region on success.
503 */
504 DECLR3CALLBACKMEMBER(int, pfnQueryRegionPropertiesForLba,(PPDMIMEDIA pInterface, uint64_t u64LbaStart,
505 uint32_t *puRegion, uint64_t *pcBlocks,
506 uint64_t *pcbBlock, PVDREGIONDATAFORM penmDataForm));
507
508} PDMIMEDIA;
509/** PDMIMEDIA interface ID. */
510#define PDMIMEDIA_IID "8ec68c48-dd20-4430-8386-f0d628a5aca6"
511
512
513/**
514 * Opaque I/O request handle.
515 *
516 * The specific content depends on the driver implementing this interface.
517 */
518typedef struct PDMMEDIAEXIOREQINT *PDMMEDIAEXIOREQ;
519/** Pointer to an I/O request handle. */
520typedef PDMMEDIAEXIOREQ *PPDMMEDIAEXIOREQ;
521
522/** A I/O request ID. */
523typedef uint64_t PDMMEDIAEXIOREQID;
524
525/**
526 * I/O Request Type.
527 */
528typedef enum PDMMEDIAEXIOREQTYPE
529{
530 /** Invalid tpe. */
531 PDMMEDIAEXIOREQTYPE_INVALID = 0,
532 /** Flush request. */
533 PDMMEDIAEXIOREQTYPE_FLUSH,
534 /** Write request. */
535 PDMMEDIAEXIOREQTYPE_WRITE,
536 /** Read request. */
537 PDMMEDIAEXIOREQTYPE_READ,
538 /** Discard request. */
539 PDMMEDIAEXIOREQTYPE_DISCARD,
540 /** SCSI command. */
541 PDMMEDIAEXIOREQTYPE_SCSI
542} PDMMEDIAEXIOREQTYPE;
543/** Pointer to a I/O request type. */
544typedef PDMMEDIAEXIOREQTYPE *PPDMMEDIAEXIOREQTYPE;
545
546/**
547 * Data direction for raw SCSI commands.
548 */
549typedef enum PDMMEDIAEXIOREQSCSITXDIR
550{
551 /** Invalid data direction. */
552 PDMMEDIAEXIOREQSCSITXDIR_INVALID = 0,
553 /** Direction is unknown. */
554 PDMMEDIAEXIOREQSCSITXDIR_UNKNOWN,
555 /** Direction is from device to host. */
556 PDMMEDIAEXIOREQSCSITXDIR_FROM_DEVICE,
557 /** Direction is from host to device. */
558 PDMMEDIAEXIOREQSCSITXDIR_TO_DEVICE,
559 /** No data transfer associated with this request. */
560 PDMMEDIAEXIOREQSCSITXDIR_NONE,
561 /** 32bit hack. */
562 PDMMEDIAEXIOREQSCSITXDIR_32BIT_HACK = 0x7fffffff
563} PDMMEDIAEXIOREQSCSITXDIR;
564
565/**
566 * I/O request state.
567 */
568typedef enum PDMMEDIAEXIOREQSTATE
569{
570 /** Invalid state. */
571 PDMMEDIAEXIOREQSTATE_INVALID = 0,
572 /** The request is active and being processed. */
573 PDMMEDIAEXIOREQSTATE_ACTIVE,
574 /** The request is suspended due to an error and no processing will take place. */
575 PDMMEDIAEXIOREQSTATE_SUSPENDED,
576 /** 32bit hack. */
577 PDMMEDIAEXIOREQSTATE_32BIT_HACK = 0x7fffffff
578} PDMMEDIAEXIOREQSTATE;
579/** Pointer to a I/O request state. */
580typedef PDMMEDIAEXIOREQSTATE *PPDMMEDIAEXIOREQSTATE;
581
582/** @name Supported feature flags
583 * @{ */
584/** I/O requests will execute asynchronously by default. */
585#define PDMIMEDIAEX_FEATURE_F_ASYNC RT_BIT_32(0)
586/** The discard request is supported. */
587#define PDMIMEDIAEX_FEATURE_F_DISCARD RT_BIT_32(1)
588/** The send raw SCSI command request is supported. */
589#define PDMIMEDIAEX_FEATURE_F_RAWSCSICMD RT_BIT_32(2)
590/** Mask of valid flags. */
591#define PDMIMEDIAEX_FEATURE_F_VALID (PDMIMEDIAEX_FEATURE_F_ASYNC | PDMIMEDIAEX_FEATURE_F_DISCARD | PDMIMEDIAEX_FEATURE_F_RAWSCSICMD)
592/** @} */
593
594/** @name I/O request specific flags
595 * @{ */
596/** Default behavior (async I/O).*/
597#define PDMIMEDIAEX_F_DEFAULT (0)
598/** The I/O request will be executed synchronously. */
599#define PDMIMEDIAEX_F_SYNC RT_BIT_32(0)
600/** Whether to suspend the VM on a recoverable error with
601 * an appropriate error message (disk full, etc.).
602 * The request will be retried by the driver implementing the interface
603 * when the VM resumes the next time. However before suspending the request
604 * the owner of the request will be notified using the PDMMEDIAEXPORT::pfnIoReqStateChanged.
605 * The same goes for resuming the request after the VM was resumed.
606 */
607#define PDMIMEDIAEX_F_SUSPEND_ON_RECOVERABLE_ERR RT_BIT_32(1)
608 /** Mask of valid flags. */
609#define PDMIMEDIAEX_F_VALID (PDMIMEDIAEX_F_SYNC | PDMIMEDIAEX_F_SUSPEND_ON_RECOVERABLE_ERR)
610/** @} */
611
612/** Pointer to an extended media notification interface. */
613typedef struct PDMIMEDIAEXPORT *PPDMIMEDIAEXPORT;
614
615/**
616 * Asynchronous version of the media interface (up).
617 * Pair with PDMIMEDIAEXPORT.
618 */
619typedef struct PDMIMEDIAEXPORT
620{
621 /**
622 * Notify completion of a I/O request.
623 *
624 * @returns VBox status code.
625 * @param pInterface Pointer to the interface structure containing the called function pointer.
626 * @param hIoReq The I/O request handle.
627 * @param pvIoReqAlloc The allocator specific memory for this request.
628 * @param rcReq IPRT Status code of the completed request.
629 * VERR_PDM_MEDIAEX_IOREQ_CANCELED if the request was canceled by a call to
630 * PDMIMEDIAEX::pfnIoReqCancel.
631 * @thread Any thread.
632 */
633 DECLR3CALLBACKMEMBER(int, pfnIoReqCompleteNotify, (PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
634 void *pvIoReqAlloc, int rcReq));
635
636 /**
637 * Copy data from the memory buffer of the caller to the callees memory buffer for the given request.
638 *
639 * @returns VBox status code.
640 * @retval VERR_PDM_MEDIAEX_IOBUF_OVERFLOW if there is not enough room to store the data.
641 * @param pInterface Pointer to the interface structure containing the called function pointer.
642 * @param hIoReq The I/O request handle.
643 * @param pvIoReqAlloc The allocator specific memory for this request.
644 * @param offDst The destination offset from the start to write the data to.
645 * @param pSgBuf The S/G buffer to read the data from.
646 * @param cbCopy How many bytes to copy.
647 */
648 DECLR3CALLBACKMEMBER(int, pfnIoReqCopyFromBuf, (PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
649 void *pvIoReqAlloc, uint32_t offDst, PRTSGBUF pSgBuf,
650 size_t cbCopy));
651
652 /**
653 * Copy data to the memory buffer of the caller from the callees memory buffer for the given request.
654 *
655 * @returns VBox status code.
656 * @retval VERR_PDM_MEDIAEX_IOBUF_UNDERRUN if there is not enough data to copy from the buffer.
657 * @param pInterface Pointer to the interface structure containing the called function pointer.
658 * @param hIoReq The I/O request handle.
659 * @param pvIoReqAlloc The allocator specific memory for this request.
660 * @param offSrc The offset from the start of the buffer to read the data from.
661 * @param pSgBuf The S/G buffer to write the data to.
662 * @param cbCopy How many bytes to copy.
663 */
664 DECLR3CALLBACKMEMBER(int, pfnIoReqCopyToBuf, (PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
665 void *pvIoReqAlloc, uint32_t offSrc, PRTSGBUF pSgBuf,
666 size_t cbCopy));
667
668 /**
669 * Queries a pointer to the memory buffer for the request from the drive/device above.
670 *
671 * @returns VBox status code.
672 * @retval VERR_NOT_SUPPORTED if this is not supported for this request.
673 * @param pInterface Pointer to the interface structure containing the called function pointer.
674 * @param hIoReq The I/O request handle.
675 * @param pvIoReqAlloc The allocator specific memory for this request.
676 * @param ppvBuf Where to store the pointer to the guest buffer on success.
677 * @param pcbBuf Where to store the size of the buffer on success.
678 *
679 * @note This is an optional feature of the entity implementing this interface to avoid overhead
680 * by copying the data between buffers. If NULL it is not supported at all and the caller
681 * has to resort to PDMIMEDIAEXPORT::pfnIoReqCopyToBuf and PDMIMEDIAEXPORT::pfnIoReqCopyFromBuf.
682 * The same holds when VERR_NOT_SUPPORTED is returned.
683 *
684 * On the upside the caller of this interface might not call this method at all and just
685 * use the before mentioned methods to copy the data between the buffers.
686 */
687 DECLR3CALLBACKMEMBER(int, pfnIoReqQueryBuf, (PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
688 void *pvIoReqAlloc, void **ppvBuf, size_t *pcbBuf));
689
690 /**
691 * Queries the specified amount of ranges to discard from the callee for the given I/O request.
692 *
693 * @returns VBox status code.
694 * @param pInterface Pointer to the interface structure containing the called function pointer.
695 * @param hIoReq The I/O request handle.
696 * @param pvIoReqAlloc The allocator specific memory for this request.
697 * @param idxRangeStart The range index to start with.
698 * @param cRanges How man ranges can be stored in the provided array.
699 * @param paRanges Where to store the ranges on success.
700 * @param *pcRanges Where to store the number of ranges copied over on success.
701 */
702 DECLR3CALLBACKMEMBER(int, pfnIoReqQueryDiscardRanges, (PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
703 void *pvIoReqAlloc, uint32_t idxRangeStart,
704 uint32_t cRanges, PRTRANGE paRanges,
705 uint32_t *pcRanges));
706
707 /**
708 * Notify the request owner about a state change for the request.
709 *
710 * @returns nothing.
711 * @param pInterface Pointer to the interface structure containing the called function pointer.
712 * @param hIoReq The I/O request handle.
713 * @param pvIoReqAlloc The allocator specific memory for this request.
714 * @param enmState The new state of the request.
715 */
716 DECLR3CALLBACKMEMBER(void, pfnIoReqStateChanged, (PPDMIMEDIAEXPORT pInterface, PDMMEDIAEXIOREQ hIoReq,
717 void *pvIoReqAlloc, PDMMEDIAEXIOREQSTATE enmState));
718
719 /**
720 * Informs the device that the underlying medium was ejected.
721 *
722 * @returns nothing.
723 * @param pInterface Pointer to the interface structure containing the called function pointer.
724 */
725 DECLR3CALLBACKMEMBER(void, pfnMediumEjected, (PPDMIMEDIAEXPORT pInterface));
726
727} PDMIMEDIAEXPORT;
728
729/** PDMIMEDIAAEXPORT interface ID. */
730#define PDMIMEDIAEXPORT_IID "0ae2e534-6c28-41d6-9a88-7f88f2cb2ff8"
731
732
733/** Pointer to an extended media interface. */
734typedef struct PDMIMEDIAEX *PPDMIMEDIAEX;
735
736/**
737 * Extended version of PDMIMEDIA (down).
738 * Pair with PDMIMEDIAEXPORT.
739 */
740typedef struct PDMIMEDIAEX
741{
742 /**
743 * Queries the features supported by the entity implementing this interface.
744 *
745 * @returns VBox status code.
746 * @param pInterface Pointer to the interface structure containing the called function pointer.
747 * @param pfFeatures Where to store the supported feature flags on success.
748 */
749 DECLR3CALLBACKMEMBER(int, pfnQueryFeatures, (PPDMIMEDIAEX pInterface, uint32_t *pfFeatures));
750
751 /**
752 * Sets the size of the allocator specific memory for a I/O request.
753 *
754 * @returns VBox status code.
755 * @param pInterface Pointer to the interface structure containing the called function pointer.
756 * @param cbIoReqAlloc The size of the allocator specific memory in bytes.
757 * @thread EMT.
758 */
759 DECLR3CALLBACKMEMBER(int, pfnIoReqAllocSizeSet, (PPDMIMEDIAEX pInterface, size_t cbIoReqAlloc));
760
761 /**
762 * Allocates a new I/O request.
763 *
764 * @returns VBox status code.
765 * @retval VERR_PDM_MEDIAEX_IOREQID_CONFLICT if the ID belongs to a still active request.
766 * @param pInterface Pointer to the interface structure containing the called function pointer.
767 * @param phIoReq Where to store the handle to the new I/O request on success.
768 * @param ppvIoReqAlloc Where to store the pointer to the allocator specific memory on success.
769 * NULL if the memory size was not set or set to 0.
770 * @param uIoReqId A custom request ID which can be used to cancel the request.
771 * @param fFlags A combination of PDMIMEDIAEX_F_* flags.
772 * @thread Any thread.
773 */
774 DECLR3CALLBACKMEMBER(int, pfnIoReqAlloc, (PPDMIMEDIAEX pInterface, PPDMMEDIAEXIOREQ phIoReq, void **ppvIoReqAlloc,
775 PDMMEDIAEXIOREQID uIoReqId, uint32_t fFlags));
776
777 /**
778 * Frees a given I/O request.
779 *
780 * @returns VBox status code.
781 * @retval VERR_PDM_MEDIAEX_IOREQ_INVALID_STATE if the given request is still active.
782 * @param pInterface Pointer to the interface structure containing the called function pointer.
783 * @param hIoReq The I/O request to free.
784 * @thread Any thread.
785 */
786 DECLR3CALLBACKMEMBER(int, pfnIoReqFree, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq));
787
788 /**
789 * Queries the residual amount of data not transfered when the request completed.
790 *
791 * @returns VBox status code.
792 * @retval VERR_PDM_MEDIAEX_IOREQ_INVALID_STATE has not completed yet.
793 * @param pInterface Pointer to the interface structure containing the called function pointer.
794 * @param hIoReq The I/O request.
795 * @param pcbResidual Where to store the amount of resdiual data in bytes.
796 * @thread Any thread.
797 */
798 DECLR3CALLBACKMEMBER(int, pfnIoReqQueryResidual, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, size_t *pcbResidual));
799
800 /**
801 * Queries the residual amount of data not transfered when the request completed.
802 *
803 * @returns VBox status code.
804 * @retval VERR_PDM_MEDIAEX_IOREQ_INVALID_STATE has not completed yet.
805 * @param pInterface Pointer to the interface structure containing the called function pointer.
806 * @param hIoReq The I/O request.
807 * @param pcbXfer Where to store the amount of resdiual data in bytes.
808 * @thread Any thread.
809 *
810 * @note For simple read/write requests this returns the amount to read/write as given to the
811 * PDMIMEDIAEX::pfnIoReqRead or PDMIMEDIAEX::pfnIoReqWrite call.
812 * For SCSI commands this returns the transfer size as given in the provided CDB.
813 */
814 DECLR3CALLBACKMEMBER(int, pfnIoReqQueryXferSize, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, size_t *pcbXfer));
815
816 /**
817 * Cancels all I/O active requests.
818 *
819 * @returns VBox status code.
820 * @param pInterface Pointer to the interface structure containing the called function pointer.
821 * @thread Any thread.
822 */
823 DECLR3CALLBACKMEMBER(int, pfnIoReqCancelAll, (PPDMIMEDIAEX pInterface));
824
825 /**
826 * Cancels a I/O request identified by the ID.
827 *
828 * @returns VBox status code.
829 * @retval VERR_PDM_MEDIAEX_IOREQID_NOT_FOUND if the given ID could not be found in the active request list.
830 * (The request has either completed already or an invalid ID was given).
831 * @param pInterface Pointer to the interface structure containing the called function pointer.
832 * @param uIoReqId The I/O request ID
833 * @thread Any thread.
834 */
835 DECLR3CALLBACKMEMBER(int, pfnIoReqCancel, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQID uIoReqId));
836
837 /**
838 * Start a reading request.
839 *
840 * @returns VBox status code.
841 * @retval VERR_PDM_MEDIAEX_IOREQ_CANCELED if the request was canceled by a call to
842 * PDMIMEDIAEX::pfnIoReqCancel.
843 * @retval VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS if the request was successfully submitted but is still in progress.
844 * Completion will be notified through PDMIMEDIAEXPORT::pfnIoReqCompleteNotify with the appropriate status code.
845 * @retval VINF_SUCCESS if the request completed successfully.
846 * @param pInterface Pointer to the interface structure containing the called function pointer.
847 * @param hIoReq The I/O request to associate the read with.
848 * @param off Offset to start reading from. Must be aligned to a sector boundary.
849 * @param cbRead Number of bytes to read. Must be aligned to a sector boundary.
850 * @thread Any thread.
851 */
852 DECLR3CALLBACKMEMBER(int, pfnIoReqRead, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, uint64_t off, size_t cbRead));
853
854 /**
855 * Start a writing request.
856 *
857 * @returns VBox status code.
858 * @retval VERR_PDM_MEDIAEX_IOREQ_CANCELED if the request was canceled by a call to
859 * PDMIMEDIAEX::pfnIoReqCancel.
860 * @retval VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS if the request was successfully submitted but is still in progress.
861 * Completion will be notified through PDMIMEDIAEXPORT::pfnIoReqCompleteNotify with the appropriate status code.
862 * @retval VINF_SUCCESS if the request completed successfully.
863 * @param pInterface Pointer to the interface structure containing the called function pointer.
864 * @param hIoReq The I/O request to associate the write with.
865 * @param off Offset to start reading from. Must be aligned to a sector boundary.
866 * @param cbWrite Number of bytes to write. Must be aligned to a sector boundary.
867 * @thread Any thread.
868 */
869 DECLR3CALLBACKMEMBER(int, pfnIoReqWrite, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, uint64_t off, size_t cbWrite));
870
871 /**
872 * Flush everything to disk.
873 *
874 * @returns VBox status code.
875 * @retval VERR_PDM_MEDIAEX_IOREQ_CANCELED if the request was canceled by a call to
876 * PDMIMEDIAEX::pfnIoReqCancel.
877 * @retval VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS if the request was successfully submitted but is still in progress.
878 * Completion will be notified through PDMIMEDIAEXPORT::pfnIoReqCompleteNotify with the appropriate status code.
879 * @retval VINF_SUCCESS if the request completed successfully.
880 * @param pInterface Pointer to the interface structure containing the called function pointer.
881 * @param hIoReq The I/O request to associate the flush with.
882 * @thread Any thread.
883 */
884 DECLR3CALLBACKMEMBER(int, pfnIoReqFlush, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq));
885
886 /**
887 * Discards the given range.
888 *
889 * @returns VBox status code.
890 * @retval VERR_PDM_MEDIAEX_IOREQ_CANCELED if the request was canceled by a call to
891 * PDMIMEDIAEX::pfnIoReqCancel.
892 * @retval VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS if the request was successfully submitted but is still in progress.
893 * Completion will be notified through PDMIMEDIAEXPORT::pfnIoReqCompleteNotify with the appropriate status code.
894 * @retval VINF_SUCCESS if the request completed successfully.
895 * @param pInterface Pointer to the interface structure containing the called function pointer.
896 * @param hIoReq The I/O request to associate the discard with.
897 * @param cRangesMax The maximum number of ranges this request has associated, this must not be accurate
898 * but can actually be bigger than the amount of ranges actually available.
899 * @thread Any thread.
900 */
901 DECLR3CALLBACKMEMBER(int, pfnIoReqDiscard, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, unsigned cRangesMax));
902
903 /**
904 * Send a raw command to the underlying device (CDROM).
905 *
906 * @returns VBox status code.
907 * @retval VERR_PDM_MEDIAEX_IOREQ_CANCELED if the request was canceled by a call to
908 * PDMIMEDIAEX::pfnIoReqCancel.
909 * @retval VINF_PDM_MEDIAEX_IOREQ_IN_PROGRESS if the request was successfully submitted but is still in progress.
910 * Completion will be notified through PDMIMEDIAEXPORT::pfnIoReqCompleteNotify with the appropriate status code.
911 * @param pInterface Pointer to the interface structure containing the called function pointer.
912 * @param hIoReq The I/O request to associate the command with.
913 * @param uLun The LUN the command is for.
914 * @param pbCdb The SCSI CDB containing the command.
915 * @param cbCdb Size of the CDB in bytes.
916 * @param enmTxDir Direction of transfer.
917 * @param cbBuf Size of the transfer buffer.
918 * @param pabSense Where to store the optional sense key.
919 * @param cbSense Size of the sense key buffer.
920 * @param pu8ScsiSts Where to store the SCSI status on success.
921 * @param cTimeoutMillies Command timeout in milliseconds.
922 * @thread Any thread.
923 */
924 DECLR3CALLBACKMEMBER(int, pfnIoReqSendScsiCmd,(PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq, uint32_t uLun,
925 const uint8_t *pbCdb, size_t cbCdb, PDMMEDIAEXIOREQSCSITXDIR enmTxDir,
926 size_t cbBuf, uint8_t *pabSense, size_t cbSense, uint8_t *pu8ScsiSts,
927 uint32_t cTimeoutMillies));
928
929 /**
930 * Returns the number of active I/O requests.
931 *
932 * @returns Number of active I/O requests.
933 * @param pInterface Pointer to the interface structure containing the called function pointer.
934 * @thread Any thread.
935 */
936 DECLR3CALLBACKMEMBER(uint32_t, pfnIoReqGetActiveCount, (PPDMIMEDIAEX pInterface));
937
938 /**
939 * Returns the number of suspended requests.
940 *
941 * @returns Number of suspended I/O requests.
942 * @param pInterface Pointer to the interface structure containing the called function pointer.
943 * @thread Any thread.
944 */
945 DECLR3CALLBACKMEMBER(uint32_t, pfnIoReqGetSuspendedCount, (PPDMIMEDIAEX pInterface));
946
947 /**
948 * Gets the first suspended request handle.
949 *
950 * @returns VBox status code.
951 * @retval VERR_NOT_FOUND if there is no suspended request waiting.
952 * @param pInterface Pointer to the interface structure containing the called function pointer.
953 * @param phIoReq Where to store the request handle on success.
954 * @param ppvIoReqAlloc Where to store the pointer to the allocator specific memory on success.
955 * @thread Any thread.
956 *
957 * @note This should only be called when the VM is suspended to make sure the request doesn't suddenly
958 * changes into the active state again. The only purpose for this method for now is to make saving the state
959 * possible without breaking saved state versions.
960 */
961 DECLR3CALLBACKMEMBER(int, pfnIoReqQuerySuspendedStart, (PPDMIMEDIAEX pInterface, PPDMMEDIAEXIOREQ phIoReq, void **ppvIoReqAlloc));
962
963 /**
964 * Gets the next suspended request handle.
965 *
966 * @returns VBox status code.
967 * @retval VERR_NOT_FOUND if there is no suspended request waiting.
968 * @param pInterface Pointer to the interface structure containing the called function pointer.
969 * @param hIoReq The current request handle.
970 * @param phIoReqNext Where to store the request handle on success.
971 * @param ppvIoReqAllocNext Where to store the pointer to the allocator specific memory on success.
972 * @thread Any thread.
973 *
974 * @note This should only be called when the VM is suspended to make sure the request doesn't suddenly
975 * changes into the active state again. The only purpose for this method for now is to make saving the state
976 * possible without breaking saved state versions.
977 */
978 DECLR3CALLBACKMEMBER(int, pfnIoReqQuerySuspendedNext, (PPDMIMEDIAEX pInterface, PDMMEDIAEXIOREQ hIoReq,
979 PPDMMEDIAEXIOREQ phIoReqNext, void **ppvIoReqAllocNext));
980
981 /**
982 * Saves the given I/O request state in the provided saved state unit.
983 *
984 * @returns VBox status code.
985 * @param pInterface Pointer to the interface structure containing the called function pointer.
986 * @param pSSM The SSM handle.
987 * @param hIoReq The request handle to save.
988 */
989 DECLR3CALLBACKMEMBER(int, pfnIoReqSuspendedSave, (PPDMIMEDIAEX pInterface, PSSMHANDLE pSSM, PDMMEDIAEXIOREQ hIoReq));
990
991 /**
992 * Load a suspended request state from the given saved state unit and link it into the suspended list.
993 *
994 * @returns VBox status code.
995 * @param pInterface Pointer to the interface structure containing the called function pointer.
996 * @param pSSM The SSM handle to read the state from.
997 * @param hIoReq The request handle to load the state into.
998 */
999 DECLR3CALLBACKMEMBER(int, pfnIoReqSuspendedLoad, (PPDMIMEDIAEX pInterface, PSSMHANDLE pSSM, PDMMEDIAEXIOREQ hIoReq));
1000
1001} PDMIMEDIAEX;
1002/** PDMIMEDIAEX interface ID. */
1003#define PDMIMEDIAEX_IID "1f82b709-a9f7-4928-ad50-e879c9bbeba1"
1004
1005/** @} */
1006
1007RT_C_DECLS_END
1008
1009#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