VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/VSCSI/VSCSIInternal.h@ 38680

Last change on this file since 38680 was 38680, checked in by vboxsync, 13 years ago

VSCSI+DrvSCSI: Add support for the UNMAP command if discarding is enabled

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 12.5 KB
Line 
1/* $Id: VSCSIInternal.h 38680 2011-09-08 07:52:08Z vboxsync $ */
2/** @file
3 * Virtual SCSI driver: Internal defines
4 */
5
6/*
7 * Copyright (C) 2006-2011 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17#ifndef ___VSCSIInternal_h
18#define ___VSCSIInternal_h
19
20#include <VBox/vscsi.h>
21#include <VBox/scsi.h>
22#include <iprt/memcache.h>
23#include <iprt/sg.h>
24#include <iprt/list.h>
25
26#include "VSCSIInline.h"
27#include "VSCSIVpdPages.h"
28
29/** Pointer to an internal virtual SCSI device. */
30typedef VSCSIDEVICEINT *PVSCSIDEVICEINT;
31/** Pointer to an internal virtual SCSI device LUN. */
32typedef VSCSILUNINT *PVSCSILUNINT;
33/** Pointer to an internal virtual SCSI device LUN pointer. */
34typedef PVSCSILUNINT *PPVSCSILUNINT;
35/** Pointer to a virtual SCSI LUN descriptor. */
36typedef struct VSCSILUNDESC *PVSCSILUNDESC;
37/** Pointer to a virtual SCSI request. */
38typedef VSCSIREQINT *PVSCSIREQINT;
39/** Pointer to a virtual SCSI I/O request. */
40typedef VSCSIIOREQINT *PVSCSIIOREQINT;
41/** Pointer to virtual SCSI sense data state. */
42typedef struct VSCSISENSE *PVSCSISENSE;
43
44/**
45 * Virtual SCSI sense data handling.
46 */
47typedef struct VSCSISENSE
48{
49 /** Buffer holding the sense data. */
50 uint8_t abSenseBuf[32];
51} VSCSISENSE;
52
53/**
54 * Virtual SCSI device.
55 */
56typedef struct VSCSIDEVICEINT
57{
58 /** Request completion callback */
59 PFNVSCSIREQCOMPLETED pfnVScsiReqCompleted;
60 /** Opaque user data. */
61 void *pvVScsiDeviceUser;
62 /** Number of LUNs currently attached. */
63 uint32_t cLunsAttached;
64 /** How many LUNs are fitting in the array. */
65 uint32_t cLunsMax;
66 /** Request cache */
67 RTMEMCACHE hCacheReq;
68 /** Sense data handling. */
69 VSCSISENSE VScsiSense;
70 /** Pointer to the array of LUN handles.
71 * The index is the LUN id. */
72 PPVSCSILUNINT papVScsiLun;
73} VSCSIDEVICEINT;
74
75/**
76 * Virtual SCSI device LUN.
77 */
78typedef struct VSCSILUNINT
79{
80 /** Pointer to the parent SCSI device. */
81 PVSCSIDEVICEINT pVScsiDevice;
82 /** Opaque user data */
83 void *pvVScsiLunUser;
84 /** I/O callback table */
85 PVSCSILUNIOCALLBACKS pVScsiLunIoCallbacks;
86 /** Pointer to the LUN type descriptor. */
87 PVSCSILUNDESC pVScsiLunDesc;
88 /** Flags of supported features. */
89 uint64_t fFeatures;
90 /** I/O request processing data */
91 struct
92 {
93 /** Number of outstanding tasks on this LUN. */
94 volatile uint32_t cReqOutstanding;
95 } IoReq;
96} VSCSILUNINT;
97
98/**
99 * Virtual SCSI request.
100 */
101typedef struct VSCSIREQINT
102{
103 /** The LUN the request is for. */
104 uint32_t iLun;
105 /** The CDB */
106 uint8_t *pbCDB;
107 /** Size of the CDB */
108 size_t cbCDB;
109 /** S/G buffer. */
110 RTSGBUF SgBuf;
111 /** Pointer to the sense buffer. */
112 uint8_t *pbSense;
113 /** Size of the sense buffer */
114 size_t cbSense;
115 /** Opaque user data associated with this request */
116 void *pvVScsiReqUser;
117} VSCSIREQINT;
118
119/**
120 * Virtual SCSI I/O request.
121 */
122typedef struct VSCSIIOREQINT
123{
124 /** The associated request. */
125 PVSCSIREQINT pVScsiReq;
126 /** Lun for this I/O request. */
127 PVSCSILUNINT pVScsiLun;
128 /** Transfer direction */
129 VSCSIIOREQTXDIR enmTxDir;
130 /** Direction dependent data. */
131 union
132 {
133 /** Read/Write request. */
134 struct
135 {
136 /** Start offset */
137 uint64_t uOffset;
138 /** Number of bytes to transfer */
139 size_t cbTransfer;
140 /** Number of bytes the S/G list holds */
141 size_t cbSeg;
142 /** Number of segments. */
143 unsigned cSeg;
144 /** Segment array. */
145 PCRTSGSEG paSeg;
146 } Io;
147 /** Unmape request. */
148 struct
149 {
150 /** Array of ranges to unmap. */
151 PVSCSIRANGE paRanges;
152 /** Number of ranges. */
153 unsigned cRanges;
154 } Unmap;
155 } u;
156} VSCSIIOREQINT;
157
158/**
159 * VPD page pool.
160 */
161typedef struct VSCSIVPDPOOL
162{
163 /** List of registered pages. */
164 RTLISTNODE ListPages;
165} VSCSIVPDPOOL;
166/** Pointer to the VSCSI VPD page pool. */
167typedef VSCSIVPDPOOL *PVSCSIVPDPOOL;
168
169/**
170 * Virtual SCSI LUN descriptor.
171 */
172typedef struct VSCSILUNDESC
173{
174 /** Device type this descriptor emulates. */
175 VSCSILUNTYPE enmLunType;
176 /** Descriptor name */
177 const char *pcszDescName;
178 /** LUN type size */
179 size_t cbLun;
180
181 /**
182 * Initialise a Lun instance.
183 *
184 * @returns VBox status code.
185 * @param pVScsiLun The SCSI LUN instance.
186 */
187 DECLR3CALLBACKMEMBER(int, pfnVScsiLunInit, (PVSCSILUNINT pVScsiLun));
188
189 /**
190 * Destroy a Lun instance.
191 *
192 * @returns VBox status code.
193 * @param pVScsiLun The SCSI LUN instance.
194 */
195 DECLR3CALLBACKMEMBER(int, pfnVScsiLunDestroy, (PVSCSILUNINT pVScsiLun));
196
197 /**
198 * Processes a SCSI request.
199 *
200 * @returns VBox status code.
201 * @param pVScsiLun The SCSI LUN instance.
202 * @param pVScsiReq The SCSi request to process.
203 */
204 DECLR3CALLBACKMEMBER(int, pfnVScsiLunReqProcess, (PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq));
205
206} VSCSILUNDESC;
207
208/** Maximum number of LUNs a device can have. */
209#define VSCSI_DEVICE_LUN_MAX 128
210
211/**
212 * Completes a SCSI request and calls the completion handler.
213 *
214 * @returns nothing.
215 * @param pVScsiDevice The virtual SCSI device.
216 * @param pVScsiReq The request which completed.
217 * @param rcScsiCode The status code
218 * One of the SCSI_STATUS_* #defines.
219 * @param fRedoPossible Flag whether redo is possible.
220 * @param rcReq Informational return code of the request.
221 */
222void vscsiDeviceReqComplete(PVSCSIDEVICEINT pVScsiDevice, PVSCSIREQINT pVScsiReq,
223 int rcScsiCode, bool fRedoPossible, int rcReq);
224
225/**
226 * Init the sense data state.
227 *
228 * @returns nothing.
229 * @param pVScsiSense The SCSI sense data state to init.
230 */
231void vscsiSenseInit(PVSCSISENSE pVScsiSense);
232
233/**
234 * Sets a ok sense code.
235 *
236 * @returns SCSI status code.
237 * @param pVScsiSense The SCSI sense state to use.
238 * @param pVScsiReq The SCSI request.
239 */
240int vscsiReqSenseOkSet(PVSCSISENSE pVScsiSense, PVSCSIREQINT pVScsiReq);
241
242/**
243 * Sets a error sense code.
244 *
245 * @returns SCSI status code.
246 * @param pVScsiSense The SCSI sense state to use.
247 * @param pVScsiReq The SCSI request.
248 * @param uSCSISenseKey The SCSI sense key to set.
249 * @param uSCSIASC The ASC value.
250 * @param uSCSIASC The ASCQ value.
251 */
252int vscsiReqSenseErrorSet(PVSCSISENSE pVScsiSense, PVSCSIREQINT pVScsiReq, uint8_t uSCSISenseKey,
253 uint8_t uSCSIASC, uint8_t uSCSIASCQ);
254
255/**
256 * Process a request sense command.
257 *
258 * @returns SCSI status code.
259 * @param pVScsiSense The SCSI sense state to use.
260 * @param pVScsiReq The SCSI request.
261 */
262int vscsiReqSenseCmd(PVSCSISENSE pVScsiSense, PVSCSIREQINT pVScsiReq);
263
264/**
265 * Inits the VPD page pool.
266 *
267 * @returns VBox status code.
268 * @param pVScsiVpdPool The VPD page pool to initialize.
269 */
270int vscsiVpdPagePoolInit(PVSCSIVPDPOOL pVScsiVpdPool);
271
272/**
273 * Destroys the given VPD page pool freeing all pages in it.
274 *
275 * @returns nothing.
276 * @param pVScsiVpdPool The VPD page pool to destroy.
277 */
278void vscsiVpdPagePoolDestroy(PVSCSIVPDPOOL pVScsiVpdPool);
279
280/**
281 * Allocates a new page in the VPD page pool with the given number.
282 *
283 * @returns VBox status code.
284 * @retval VERR_ALREADY_EXIST if the page number is in use.
285 * @param pVScsiVpdPool The VPD page pool the page will belong to.
286 * @param uPage The page number, must be unique.
287 * @param cbPage Size of the page in bytes.
288 * @param ppbPage Where to store the pointer to the raw page data on success.
289 */
290int vscsiVpdPagePoolAllocNewPage(PVSCSIVPDPOOL pVScsiVpdPool, uint8_t uPage, size_t cbPage, uint8_t **ppbPage);
291
292/**
293 * Queries the given page from the pool and cpies it to the buffer given
294 * by the SCSI request.
295 *
296 * @returns VBox status code.
297 * @retval VERR_NOT_FOUND if the page is not in the pool.
298 * @param pVScsiVpdPool The VPD page pool to use.
299 * @param pVScsiReq The SCSI request.
300 * @param uPage Page to query.
301 */
302int vscsiVpdPagePoolQueryPage(PVSCSIVPDPOOL pVScsiVpdPool, PVSCSIREQINT pVScsiReq, uint8_t uPage);
303
304/**
305 * Enqueues a new flush request
306 *
307 * @returns VBox status code.
308 * @param pVScsiLun The LUN instance which issued the request.
309 * @param pVScsiReq The virtual SCSI request associated with the flush.
310 */
311int vscsiIoReqFlushEnqueue(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq);
312
313/**
314 * Enqueue a new data transfer request.
315 *
316 * @returns VBox status code.
317 * @param pVScsiLun The LUN instance which issued the request.
318 * @param pVScsiReq The virtual SCSI request associated with the transfer.
319 * @param enmTxDir Transfer direction.
320 * @param uOffset Start offset of the transfer.
321 * @param cbTransfer Number of bytes to transfer.
322 */
323int vscsiIoReqTransferEnqueue(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq,
324 VSCSIIOREQTXDIR enmTxDir, uint64_t uOffset,
325 size_t cbTransfer);
326
327/**
328 * Enqueue a new unmap request.
329 *
330 * @returns VBox status code.
331 * @param pVScsiLun The LUN instance which issued the request.
332 * @param pVScsiReq The virtual SCSI request associated with the transfer.
333 * @param paRanges The array of ranges to unmap.
334 * @param cRanges Number of ranges in the array.
335 */
336int vscsiIoReqUnmapEnqueue(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq,
337 PVSCSIRANGE paRanges, unsigned cRanges);
338
339/**
340 * Returns the current number of outstanding tasks on the given LUN.
341 *
342 * @returns Number of outstanding tasks.
343 * @param pVScsiLun The LUN to check.
344 */
345uint32_t vscsiIoReqOutstandingCountGet(PVSCSILUNINT pVScsiLun);
346
347/**
348 * Wrapper for the get medium size I/O callback.
349 *
350 * @returns VBox status code.
351 * @param pVScsiLun The LUN.
352 * @param pcbSize Where to store the size on success.
353 */
354DECLINLINE(int) vscsiLunMediumGetSize(PVSCSILUNINT pVScsiLun, uint64_t *pcbSize)
355{
356 return pVScsiLun->pVScsiLunIoCallbacks->pfnVScsiLunMediumGetSize(pVScsiLun,
357 pVScsiLun->pvVScsiLunUser,
358 pcbSize);
359}
360
361/**
362 * Wrapper for the I/O request enqueue I/O callback.
363 *
364 * @returns VBox status code.
365 * @param pVScsiLun The LUN.
366 * @param pVScsiIoReq The I/O request to enqueue.
367 */
368DECLINLINE(int) vscsiLunReqTransferEnqueue(PVSCSILUNINT pVScsiLun, PVSCSIIOREQINT pVScsiIoReq)
369{
370 return pVScsiLun->pVScsiLunIoCallbacks->pfnVScsiLunReqTransferEnqueue(pVScsiLun,
371 pVScsiLun->pvVScsiLunUser,
372 pVScsiIoReq);
373}
374
375/**
376 * Wrapper for the get feature flags I/O callback.
377 *
378 * @returns VBox status code.
379 * @param pVScsiLun The LUN.
380 * @param pVScsiIoReq The I/O request to enqueue.
381 */
382DECLINLINE(int) vscsiLunGetFeatureFlags(PVSCSILUNINT pVScsiLun, uint64_t *pfFeatures)
383{
384 return pVScsiLun->pVScsiLunIoCallbacks->pfnVScsiLunGetFeatureFlags(pVScsiLun,
385 pVScsiLun->pvVScsiLunUser,
386 pfFeatures);
387}
388
389/**
390 * Wrapper around vscsiReqSenseOkSet()
391 */
392DECLINLINE(int) vscsiLunReqSenseOkSet(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq)
393{
394 return vscsiReqSenseOkSet(&pVScsiLun->pVScsiDevice->VScsiSense, pVScsiReq);
395}
396
397/**
398 * Wrapper around vscsiReqSenseErrorSet()
399 */
400DECLINLINE(int) vscsiLunReqSenseErrorSet(PVSCSILUNINT pVScsiLun, PVSCSIREQINT pVScsiReq, uint8_t uSCSISenseKey, uint8_t uSCSIASC, uint8_t uSCSIASCQ)
401{
402 return vscsiReqSenseErrorSet(&pVScsiLun->pVScsiDevice->VScsiSense, pVScsiReq, uSCSISenseKey, uSCSIASC, uSCSIASCQ);
403}
404
405
406#endif /* ___VSCSIInternal_h */
407
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