VirtualBox

source: vbox/trunk/src/VBox/Devices/Storage/ISCSIHDDCore.cpp@ 21094

Last change on this file since 21094 was 18678, checked in by vboxsync, 16 years ago

Storage/iSCSI: fix one case where re-login failed, and also implement basic NOP-In handling in case the target wants to ping the initiator.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 123.1 KB
Line 
1/** @file
2 * iSCSI initiator driver, VD backend.
3 */
4
5/*
6 * Copyright (C) 2006-2009 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
17 * Clara, CA 95054 USA or visit http://www.sun.com if you need
18 * additional information or have any questions.
19 */
20
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25#define LOG_GROUP LOG_GROUP_VD_ISCSI
26#include "VBoxHDD-Internal.h"
27#define VBOX_VDICORE_VD /* Signal that the header is included from here. */
28#include "VDICore.h"
29#include <VBox/err.h>
30
31#include <VBox/log.h>
32#include <iprt/alloc.h>
33#include <iprt/assert.h>
34#include <iprt/uuid.h>
35#include <iprt/file.h>
36#include <iprt/string.h>
37#include <iprt/asm.h>
38#include <iprt/thread.h>
39#include <iprt/semaphore.h>
40#include <iprt/md5.h>
41#include <iprt/tcp.h>
42#include <iprt/time.h>
43#include <VBox/scsi.h>
44
45
46/*******************************************************************************
47* Defined Constants And Macros *
48*******************************************************************************/
49
50/** Default port number to use for iSCSI. */
51#define ISCSI_DEFAULT_PORT 3260
52
53
54/** Converts a number in the range of 0 - 15 into the corresponding hex char. */
55#define NUM_2_HEX(b) ('0' + (b) + (((b) > 9) ? 39 : 0))
56/** Converts a hex char into the corresponding number in the range 0-15. */
57#define HEX_2_NUM(c) (((c) <= '9') ? ((c) - '0') : (((c - 'A' + 10) & 0xf)))
58/* Converts a base64 char into the corresponding number in the range 0-63. */
59#define B64_2_NUM(c) ((c >= 'A' && c <= 'Z') ? (c - 'A') : (c >= 'a' && c <= 'z') ? (c - 'a' + 26) : (c >= '0' && c <= '9') ? (c - '0' + 52) : (c == '+') ? 62 : (c == '/') ? 63 : -1)
60
61
62/** Minumum CHAP_MD5 challenge length in bytes. */
63#define CHAP_MD5_CHALLENGE_MIN 16
64/** Maximum CHAP_MD5 challenge length in bytes. */
65#define CHAP_MD5_CHALLENGE_MAX 24
66
67
68/**
69 * SCSI peripheral device type. */
70typedef enum SCSIDEVTYPE
71{
72 /** direct-access device. */
73 SCSI_DEVTYPE_DISK = 0,
74 /** sequential-access device. */
75 SCSI_DEVTYPE_TAPE,
76 /** printer device. */
77 SCSI_DEVTYPE_PRINTER,
78 /** processor device. */
79 SCSI_DEVTYPE_PROCESSOR,
80 /** write-once device. */
81 SCSI_DEVTYPE_WORM,
82 /** CD/DVD device. */
83 SCSI_DEVTYPE_CDROM,
84 /** scanner device. */
85 SCSI_DEVTYPE_SCANNER,
86 /** optical memory device. */
87 SCSI_DEVTYPE_OPTICAL,
88 /** medium changer. */
89 SCSI_DEVTYPE_CHANGER,
90 /** communications device. */
91 SCSI_DEVTYPE_COMMUNICATION,
92 /** storage array controller device. */
93 SCSI_DEVTYPE_RAIDCTL = 0x0c,
94 /** enclosure services device. */
95 SCSI_DEVTYPE_ENCLOSURE,
96 /** simplified direct-access device. */
97 SCSI_DEVTYPE_SIMPLEDISK,
98 /** optical card reader/writer device. */
99 SCSI_DEVTYPE_OCRW,
100 /** bridge controller device. */
101 SCSI_DEVTYPE_BRIDGE,
102 /** object-based storage device. */
103 SCSI_DEVTYPE_OSD
104} SCSIDEVTYPE;
105
106/** Mask for extracting the SCSI device type out of the first byte of the INQUIRY response. */
107#define SCSI_DEVTYPE_MASK 0x1f
108
109
110/** Maximum PDU size we can handle in one piece. */
111#define ISCSI_RECV_PDU_BUFFER_SIZE (65536 + ISCSI_BHS_SIZE)
112
113
114/** Version of the iSCSI standard which this initiator driver can handle. */
115#define ISCSI_MY_VERSION 0
116
117
118/** Length of ISCSI basic header segment. */
119#define ISCSI_BHS_SIZE 48
120
121
122/** Reserved task tag value. */
123#define ISCSI_TASK_TAG_RSVD 0xffffffff
124
125
126/**
127 * iSCSI opcodes. */
128typedef enum ISCSIOPCODE
129{
130 /** NOP-Out. */
131 ISCSIOP_NOP_OUT = 0x00000000,
132 /** SCSI command. */
133 ISCSIOP_SCSI_CMD = 0x01000000,
134 /** SCSI task management request. */
135 ISCSIOP_SCSI_TASKMGMT_REQ = 0x02000000,
136 /** Login request. */
137 ISCSIOP_LOGIN_REQ = 0x03000000,
138 /** Text request. */
139 ISCSIOP_TEXT_REQ = 0x04000000,
140 /** SCSI Data-Out. */
141 ISCSIOP_SCSI_DATA_OUT = 0x05000000,
142 /** Logout request. */
143 ISCSIOP_LOGOUT_REQ = 0x06000000,
144 /** SNACK request. */
145 ISCSIOP_SNACK_REQ = 0x10000000,
146
147 /** NOP-In. */
148 ISCSIOP_NOP_IN = 0x20000000,
149 /** SCSI response. */
150 ISCSIOP_SCSI_RES = 0x21000000,
151 /** SCSI Task Management response. */
152 ISCSIOP_SCSI_TASKMGMT_RES = 0x22000000,
153 /** Login response. */
154 ISCSIOP_LOGIN_RES = 0x23000000,
155 /** Text response. */
156 ISCSIOP_TEXT_RES = 0x24000000,
157 /** SCSI Data-In. */
158 ISCSIOP_SCSI_DATA_IN = 0x25000000,
159 /** Logout response. */
160 ISCSIOP_LOGOUT_RES = 0x26000000,
161 /** Ready To Transfer (R2T). */
162 ISCSIOP_R2T = 0x31000000,
163 /** Asynchronous message. */
164 ISCSIOP_ASYN_MSG = 0x32000000,
165 /** Reject. */
166 ISCSIOP_REJECT = 0x3f000000
167} ISCSIOPCODE;
168
169/** Mask for extracting the iSCSI opcode out of the first header word. */
170#define ISCSIOP_MASK 0x3f000000
171
172
173/** ISCSI BHS word 0: Request should be processed immediately. */
174#define ISCSI_IMMEDIATE_DELIVERY_BIT 0x40000000
175
176/** ISCSI BHS word 0: This is the final PDU for this request/response. */
177#define ISCSI_FINAL_BIT 0x00800000
178/** ISCSI BHS word 0: Mask for extracting the CSG. */
179#define ISCSI_CSG_MASK 0x000c0000
180/** ISCSI BHS word 0: Shift offset for extracting the CSG. */
181#define ISCSI_CSG_SHIFT 18
182/** ISCSI BHS word 0: Mask for extracting the NSG. */
183#define ISCSI_NSG_MASK 0x00030000
184/** ISCSI BHS word 0: Shift offset for extracting the NSG. */
185#define ISCSI_NSG_SHIFT 16
186
187/** ISCSI BHS word 0: task attribute untagged */
188#define ISCSI_TASK_ATTR_UNTAGGED 0x00000000
189/** ISCSI BHS word 0: task attribute simple */
190#define ISCSI_TASK_ATTR_SIMPLE 0x00010000
191/** ISCSI BHS word 0: task attribute ordered */
192#define ISCSI_TASK_ATTR_ORDERED 0x00020000
193/** ISCSI BHS word 0: task attribute head of queue */
194#define ISCSI_TASK_ATTR_HOQ 0x00030000
195/** ISCSI BHS word 0: task attribute ACA */
196#define ISCSI_TASK_ATTR_ACA 0x00040000
197
198/** ISCSI BHS word 0: transit to next login phase. */
199#define ISCSI_TRANSIT_BIT 0x00800000
200/** ISCSI BHS word 0: continue with login negotiation. */
201#define ISCSI_CONTINUE_BIT 0x00400000
202
203/** ISCSI BHS word 0: residual underflow. */
204#define ISCSI_RESIDUAL_UNFL_BIT 0x00020000
205/** ISCSI BHS word 0: residual overflow. */
206#define ISCSI_RESIDUAL_OVFL_BIT 0x00040000
207/** ISCSI BHS word 0: Bidirectional read residual underflow. */
208#define ISCSI_BI_READ_RESIDUAL_UNFL_BIT 0x00080000
209/** ISCSI BHS word 0: Bidirectional read residual overflow. */
210#define ISCSI_BI_READ_RESIDUAL_OVFL_BIT 0x00100000
211
212/** ISCSI BHS word 0: SCSI response mask. */
213#define ISCSI_SCSI_RESPONSE_MASK 0x0000ff00
214/** ISCSI BHS word 0: SCSI status mask. */
215#define ISCSI_SCSI_STATUS_MASK 0x000000ff
216
217/** ISCSI BHS word 0: response includes status. */
218#define ISCSI_STATUS_BIT 0x00010000
219
220
221/**
222 * iSCSI login status class. */
223typedef enum ISCSILOGINSTATUSCLASS
224{
225 /** Success. */
226 ISCSI_LOGIN_STATUS_CLASS_SUCCESS = 0,
227 /** Redirection. */
228 ISCSI_LOGIN_STATUS_CLASS_REDIRECTION,
229 /** Initiator error. */
230 ISCSI_LOGIN_STATUS_CLASS_INITIATOR_ERROR,
231 /** Target error. */
232 ISCSI_LOGIN_STATUS_CLASS_TARGET_ERROR
233} ISCSILOGINSTATUSCLASS;
234
235
236/**
237 * iSCSI connection state. */
238typedef enum ISCSISTATE
239{
240 /** Not having a connection/session at all. */
241 ISCSISTATE_FREE,
242 /** Currently trying to login. */
243 ISCSISTATE_IN_LOGIN,
244 /** Normal operation, corresponds roughly to the Full Feature Phase. */
245 ISCSISTATE_NORMAL,
246 /** Currently trying to logout. */
247 ISCSISTATE_IN_LOGOUT
248} ISCSISTATE;
249
250
251/*******************************************************************************
252* Structures and Typedefs *
253*******************************************************************************/
254/**
255 * Block driver instance data.
256 */
257typedef struct ISCSIIMAGE
258{
259 /** Pointer to the filename (location). Not really used. */
260 const char *pszFilename;
261 /** Pointer to the initiator name. */
262 char *pszInitiatorName;
263 /** Pointer to the target name. */
264 char *pszTargetName;
265 /** Pointer to the target address. */
266 char *pszTargetAddress;
267 /** Pointer to the user name for authenticating the Initiator. */
268 char *pszInitiatorUsername;
269 /** Pointer to the secret for authenticating the Initiator. */
270 uint8_t *pbInitiatorSecret;
271 /** Length of the secret for authenticating the Initiator. */
272 size_t cbInitiatorSecret;
273 /** Pointer to the user name for authenticating the Target. */
274 char *pszTargetUsername;
275 /** Pointer to the secret for authenticating the Initiator. */
276 uint8_t *pbTargetSecret;
277 /** Length of the secret for authenticating the Initiator. */
278 size_t cbTargetSecret;
279 /** Initiator session identifier. */
280 uint64_t ISID;
281 /** SCSI Logical Unit Number. */
282 uint64_t LUN;
283 /** Pointer to the per-disk VD interface list. */
284 PVDINTERFACE pVDIfsDisk;
285 /** Error interface. */
286 PVDINTERFACE pInterfaceError;
287 /** Error interface callback table. */
288 PVDINTERFACEERROR pInterfaceErrorCallbacks;
289 /** TCP network stack interface. */
290 PVDINTERFACE pInterfaceNet;
291 /** TCP network stack interface callback table. */
292 PVDINTERFACETCPNET pInterfaceNetCallbacks;
293 /** Pointer to the per-image VD interface list. */
294 PVDINTERFACE pVDIfsImage;
295 /** Config interface. */
296 PVDINTERFACE pInterfaceConfig;
297 /** Config interface callback table. */
298 PVDINTERFACECONFIG pInterfaceConfigCallbacks;
299 /** Image open flags. */
300 unsigned uOpenFlags;
301 /** Number of re-login retries when a connection fails. */
302 uint32_t cISCSIRetries;
303 /** Size of volume in sectors. */
304 uint32_t cVolume;
305 /** Sector size on volume. */
306 uint32_t cbSector;
307 /** Total volume size in bytes. Easiert that multiplying the above values all the time. */
308 uint64_t cbSize;
309 /** Current state of the connection/session. */
310 ISCSISTATE state;
311 /** Flag whether the first Login Response PDU has been seen. */
312 bool FirstRecvPDU;
313 /** Initiator Task Tag of the last iSCSI request PDU. */
314 uint32_t ITT;
315 /** Sequence number of the last command. */
316 uint32_t CmdSN;
317 /** Sequence number of the next command expected by the target. */
318 uint32_t ExpCmdSN;
319 /** Maximum sequence number accepted by the target (determines size of window). */
320 uint32_t MaxCmdSN;
321 /** Expected sequence number of next status. */
322 uint32_t ExpStatSN;
323 /** Currently active request. */
324 PISCSIREQ paCurrReq;
325 /** Segment number of currently active request. */
326 uint32_t cnCurrReq;
327 /** Pointer to receive PDU buffer. (Freed by RT) */
328 void *pvRecvPDUBuf;
329 /** Length of receive PDU buffer. */
330 size_t cbRecvPDUBuf;
331 /** Mutex protecting against concurrent use from several threads. */
332 RTSEMMUTEX Mutex;
333
334 /** Pointer to the target hostname. */
335 char *pszHostname;
336 /** Pointer to the target hostname. */
337 uint32_t uPort;
338 /** Socket handle of the TCP connection. */
339 RTSOCKET Socket;
340 /** Timeout for read operations on the TCP connection (in milliseconds). */
341 uint32_t uReadTimeout;
342 /** Flag whether to use the host IP stack or DevINIP. */
343 bool fHostIP;
344} ISCSIIMAGE, *PISCSIIMAGE;
345
346
347/**
348 * SCSI transfer directions.
349 */
350typedef enum SCSIXFER
351{
352 SCSIXFER_NONE = 0,
353 SCSIXFER_TO_TARGET,
354 SCSIXFER_FROM_TARGET,
355 SCSIXFER_TO_FROM_TARGET
356} SCSIXFER, *PSCSIXFER;
357
358
359/**
360 * SCSI request structure.
361 */
362typedef struct SCSIREQ
363{
364 /** Transfer direction. */
365 SCSIXFER enmXfer;
366 /** Length of command block. */
367 size_t cbCmd;
368 /** Length of Initiator2Target data buffer. */
369 size_t cbI2TData;
370 /** Length of Target2Initiator data buffer. */
371 size_t cbT2IData;
372 /** Length of sense buffer. */
373 size_t cbSense;
374 /** Completion status of the command. */
375 uint8_t status;
376 /** Pointer to command block. */
377 void *pvCmd;
378 /** Pointer to Initiator2Target data buffer. */
379 const void *pcvI2TData;
380 /** Pointer to Target2Initiator data buffer. */
381 void *pvT2IData;
382 /** Pointer to sense buffer. */
383 void *pvSense;
384} SCSIREQ, *PSCSIREQ;
385
386
387/*******************************************************************************
388* Static Variables *
389*******************************************************************************/
390
391/** Counter for getting unique instance IDs. */
392static uint32_t s_u32iscsiID = 0;
393
394/** Default LUN. */
395static const char *s_iscsiConfigDefaultLUN = "0";
396
397/** Default initiator name. */
398static const char *s_iscsiConfigDefaultInitiatorName = "iqn.2008-04.com.sun.virtualbox.initiator";
399
400/** Default timeout, 10 seconds. */
401static const char *s_iscsiConfigDefaultTimeout = "10000";
402
403/** Default host IP stack. */
404static const char *s_iscsiConfigDefaultHostIPStack = "1";
405
406/** Description of all accepted config parameters. */
407static const VDCONFIGINFO s_iscsiConfigInfo[] =
408{
409 { "TargetName", NULL, VDCFGVALUETYPE_STRING, VD_CFGKEY_MANDATORY },
410 /* LUN is defined of string type to handle the "enc" prefix. */
411 { "LUN", s_iscsiConfigDefaultLUN, VDCFGVALUETYPE_STRING, VD_CFGKEY_MANDATORY },
412 { "TargetAddress", NULL, VDCFGVALUETYPE_STRING, VD_CFGKEY_MANDATORY },
413 { "InitiatorName", s_iscsiConfigDefaultInitiatorName, VDCFGVALUETYPE_STRING, 0 },
414 { "InitiatorUsername", NULL, VDCFGVALUETYPE_STRING, 0 },
415 { "InitiatorSecret", NULL, VDCFGVALUETYPE_BYTES, 0 },
416 { "TargetUsername", NULL, VDCFGVALUETYPE_STRING, VD_CFGKEY_EXPERT },
417 { "TargetSecret", NULL, VDCFGVALUETYPE_BYTES, VD_CFGKEY_EXPERT },
418 { "Timeout", s_iscsiConfigDefaultTimeout, VDCFGVALUETYPE_INTEGER, VD_CFGKEY_EXPERT },
419 { "HostIPStack", s_iscsiConfigDefaultHostIPStack, VDCFGVALUETYPE_INTEGER, VD_CFGKEY_EXPERT },
420 { NULL, NULL, VDCFGVALUETYPE_INTEGER, 0 }
421};
422
423/*******************************************************************************
424* Internal Functions *
425*******************************************************************************/
426
427/* iSCSI low-level functions (only to be used from the iSCSI high-level functions). */
428static uint32_t iscsiNewITT(PISCSIIMAGE pImage);
429static int iscsiSendPDU(PISCSIIMAGE pImage, PISCSIREQ paReq, uint32_t cnReq);
430static int iscsiRecvPDU(PISCSIIMAGE pImage, uint32_t itt, PISCSIRES paRes, uint32_t cnRes);
431static int drvISCSIValidatePDU(PISCSIRES paRes, uint32_t cnRes);
432static int iscsiTextAddKeyValue(uint8_t *pbBuf, size_t cbBuf, size_t *pcbBufCurr, const char *pcszKey, const char *pcszValue, size_t cbValue);
433static int iscsiTextGetKeyValue(const uint8_t *pbBuf, size_t cbBuf, const char *pcszKey, const char **ppcszValue);
434static int iscsiStrToBinary(const char *pcszValue, uint8_t *pbValue, size_t *pcbValue);
435
436/* Serial number arithmetic comparison. */
437static bool serial_number_less(uint32_t sn1, uint32_t sn2);
438
439/* CHAP-MD5 functions. */
440#ifdef IMPLEMENT_TARGET_AUTH
441static void chap_md5_generate_challenge(uint8_t *pbChallenge, size_t *pcbChallenge);
442#endif
443static void chap_md5_compute_response(uint8_t *pbResponse, uint8_t id, const uint8_t *pbChallenge, size_t cbChallenge,
444 const uint8_t *pbSecret, size_t cbSecret);
445
446
447/**
448 * Internal: signal an error to the frontend.
449 */
450DECLINLINE(int) iscsiError(PISCSIIMAGE pImage, int rc, RT_SRC_POS_DECL,
451 const char *pszFormat, ...)
452{
453 va_list va;
454 va_start(va, pszFormat);
455 if (pImage->pInterfaceError)
456 pImage->pInterfaceErrorCallbacks->pfnError(pImage->pInterfaceError->pvUser, rc, RT_SRC_POS_ARGS,
457 pszFormat, va);
458 va_end(va);
459 return rc;
460}
461
462
463static int iscsiTransportRead(PISCSIIMAGE pImage, PISCSIRES paResponse, unsigned int cnResponse)
464{
465 int rc = VINF_SUCCESS;
466 unsigned int i = 0;
467 size_t cbToRead, cbActuallyRead, residual, cbSegActual = 0, cbAHSLength, cbDataLength;
468 char *pDst;
469
470 LogFlowFunc(("cnResponse=%d (%s:%d)\n", cnResponse, pImage->pszHostname, pImage->uPort));
471 if (pImage->Socket == NIL_RTSOCKET)
472 {
473 /* Attempt to reconnect if the connection was previously broken. */
474 if (pImage->pszHostname != NULL)
475 {
476 rc = pImage->pInterfaceNetCallbacks->pfnClientConnect(pImage->pszHostname, pImage->uPort, &pImage->Socket);
477 if (RT_UNLIKELY( RT_FAILURE(rc)
478 && ( rc == VERR_NET_CONNECTION_REFUSED
479 || rc == VERR_NET_CONNECTION_RESET
480 || rc == VERR_NET_UNREACHABLE
481 || rc == VERR_NET_HOST_UNREACHABLE
482 || rc == VERR_NET_CONNECTION_TIMED_OUT)))
483 {
484 /* Standardize return value for no connection. */
485 rc = VERR_NET_CONNECTION_REFUSED;
486 }
487 }
488 }
489
490 if (RT_SUCCESS(rc) && paResponse[0].cbSeg >= 48)
491 {
492 cbToRead = 0;
493 residual = 48; /* Do not read more than the BHS length before the true PDU length is known. */
494 cbSegActual = residual;
495 pDst = (char *)paResponse[i].pvSeg;
496 uint64_t u64Timeout = RTTimeMilliTS() + pImage->uReadTimeout;
497 do
498 {
499 int64_t cMilliesRemaining = u64Timeout - RTTimeMilliTS();
500 if (cMilliesRemaining <= 0)
501 {
502 rc = VERR_TIMEOUT;
503 break;
504 }
505 Assert(cMilliesRemaining < 1000000);
506 rc = pImage->pInterfaceNetCallbacks->pfnSelectOne(pImage->Socket,
507 cMilliesRemaining);
508 if (RT_FAILURE(rc))
509 break;
510 rc = pImage->pInterfaceNetCallbacks->pfnRead(pImage->Socket,
511 pDst, residual,
512 &cbActuallyRead);
513 if (RT_FAILURE(rc))
514 break;
515 if (cbActuallyRead == 0)
516 {
517 /* The other end has closed the connection. */
518 pImage->pInterfaceNetCallbacks->pfnClientClose(pImage->Socket);
519 pImage->Socket = NIL_RTSOCKET;
520 rc = VERR_NET_CONNECTION_RESET;
521 break;
522 }
523 if (cbToRead == 0)
524 {
525 /* Currently reading the BHS. */
526 residual -= cbActuallyRead;
527 pDst += cbActuallyRead;
528 if (residual <= 40)
529 {
530 /* Enough data read to figure out the actual PDU size. */
531 uint32_t word1 = RT_N2H_U32(((uint32_t *)(paResponse[0].pvSeg))[1]);
532 cbAHSLength = (word1 & 0xff000000) >> 24;
533 cbAHSLength = ((cbAHSLength - 1) | 3) + 1; /* Add padding. */
534 cbDataLength = word1 & 0x00ffffff;
535 cbDataLength = ((cbDataLength - 1) | 3) + 1; /* Add padding. */
536 cbToRead = residual + cbAHSLength + cbDataLength;
537 residual += paResponse[0].cbSeg - 48;
538 if (residual > cbToRead)
539 residual = cbToRead;
540 cbSegActual = 48 + cbAHSLength + cbDataLength;
541 /* Check whether we are already done with this PDU (no payload). */
542 if (cbToRead == 0)
543 break;
544 }
545 }
546 else
547 {
548 cbToRead -= cbActuallyRead;
549 if (cbToRead == 0)
550 break;
551 pDst += cbActuallyRead;
552 residual -= cbActuallyRead;
553 }
554 if (residual == 0)
555 {
556 i++;
557 if (i >= cnResponse)
558 {
559 /* No space left in receive buffers. */
560 rc = VERR_BUFFER_OVERFLOW;
561 break;
562 }
563 pDst = (char *)paResponse[i].pvSeg;
564 residual = paResponse[i].cbSeg;
565 if (residual > cbToRead)
566 residual = cbToRead;
567 cbSegActual = residual;
568 }
569 } while (true);
570 }
571 else
572 {
573 if (RT_SUCCESS(rc))
574 rc = VERR_BUFFER_OVERFLOW;
575 }
576 if (RT_SUCCESS(rc))
577 {
578 paResponse[i].cbSeg = cbSegActual;
579 for (i++; i < cnResponse; i++)
580 paResponse[i].cbSeg = 0;
581 }
582
583 if (RT_UNLIKELY( RT_FAILURE(rc)
584 && ( rc == VERR_NET_CONNECTION_RESET
585 || rc == VERR_NET_CONNECTION_ABORTED
586 || rc == VERR_NET_CONNECTION_RESET_BY_PEER
587 || rc == VERR_NET_CONNECTION_REFUSED
588 || rc == VERR_BROKEN_PIPE)))
589 {
590 /* Standardize return value for broken connection. */
591 rc = VERR_BROKEN_PIPE;
592 }
593
594 LogFlowFunc(("returns %Rrc\n", rc));
595 return rc;
596}
597
598
599static int iscsiTransportWrite(PISCSIIMAGE pImage, PISCSIREQ paRequest, unsigned int cnRequest)
600{
601 int rc = VINF_SUCCESS;
602 uint32_t pad = 0;
603 unsigned int i;
604
605 LogFlow(("drvISCSITransportTcpWrite: cnRequest=%d (%s:%d)\n", cnRequest, pImage->pszHostname, pImage->uPort));
606 if (pImage->Socket == NIL_RTSOCKET)
607 {
608 /* Attempt to reconnect if the connection was previously broken. */
609 if (pImage->pszHostname != NULL)
610 {
611 rc = pImage->pInterfaceNetCallbacks->pfnClientConnect(pImage->pszHostname, pImage->uPort, &pImage->Socket);
612 if (RT_UNLIKELY( RT_FAILURE(rc)
613 && ( rc == VERR_NET_CONNECTION_REFUSED
614 || rc == VERR_NET_CONNECTION_RESET
615 || rc == VERR_NET_UNREACHABLE
616 || rc == VERR_NET_HOST_UNREACHABLE
617 || rc == VERR_NET_CONNECTION_TIMED_OUT)))
618 {
619 /* Standardize return value for no connection. */
620 rc = VERR_NET_CONNECTION_REFUSED;
621 }
622 }
623 }
624
625 if (RT_SUCCESS(rc))
626 {
627 for (i = 0; i < cnRequest; i++)
628 {
629 /* Write one chunk of data. */
630 rc = pImage->pInterfaceNetCallbacks->pfnWrite(pImage->Socket,
631 paRequest[i].pcvSeg,
632 paRequest[i].cbSeg);
633 if (RT_FAILURE(rc))
634 break;
635 /* Insert proper padding before the next chunk us written. */
636 if (paRequest[i].cbSeg & 3)
637 {
638 rc = pImage->pInterfaceNetCallbacks->pfnWrite(pImage->Socket,
639 &pad,
640 4 - (paRequest[i].cbSeg & 3));
641 if (RT_FAILURE(rc))
642 break;
643 }
644 }
645 /* Send out the request as soon as possible, otherwise the target will
646 * answer after an unnecessary delay. */
647 pImage->pInterfaceNetCallbacks->pfnFlush(pImage->Socket);
648 }
649
650 if (RT_UNLIKELY( RT_FAILURE(rc)
651 && ( rc == VERR_NET_CONNECTION_RESET
652 || rc == VERR_NET_CONNECTION_ABORTED
653 || rc == VERR_NET_CONNECTION_RESET_BY_PEER
654 || rc == VERR_NET_CONNECTION_REFUSED
655 || rc == VERR_BROKEN_PIPE)))
656 {
657 /* Standardize return value for broken connection. */
658 rc = VERR_BROKEN_PIPE;
659 }
660
661 LogFlow(("drvISCSITransportTcpWrite: returns %Rrc\n", rc));
662 return rc;
663}
664
665
666static int iscsiTransportOpen(PISCSIIMAGE pImage)
667{
668 int rc = VINF_SUCCESS;
669 size_t cbHostname = 0; /* shut up gcc */
670 const char *pcszPort = NULL; /* shut up gcc */
671 char *pszPortEnd;
672 uint16_t uPort;
673
674 /* Clean up previous connection data. */
675 if (pImage->Socket != NIL_RTSOCKET)
676 {
677 pImage->pInterfaceNetCallbacks->pfnClientClose(pImage->Socket);
678 pImage->Socket = NIL_RTSOCKET;
679 }
680 if (pImage->pszHostname)
681 {
682 RTMemFree(pImage->pszHostname);
683 pImage->pszHostname = NULL;
684 pImage->uPort = 0;
685 }
686
687 /* Locate the port number via the colon separating the hostname from the port. */
688 if (*pImage->pszTargetAddress)
689 {
690 if (*pImage->pszTargetAddress != '[')
691 {
692 /* Normal hostname or IPv4 dotted decimal. */
693 pcszPort = strchr(pImage->pszTargetAddress, ':');
694 if (pcszPort != NULL)
695 {
696 cbHostname = pcszPort - pImage->pszTargetAddress;
697 pcszPort++;
698 }
699 else
700 cbHostname = strlen(pImage->pszTargetAddress);
701 }
702 else
703 {
704 /* IPv6 literal address. Contains colons, so skip to closing square bracket. */
705 pcszPort = strchr(pImage->pszTargetAddress, ']');
706 if (pcszPort != NULL)
707 {
708 pcszPort++;
709 cbHostname = pcszPort - pImage->pszTargetAddress;
710 if (*pcszPort == '\0')
711 pcszPort = NULL;
712 else if (*pcszPort != ':')
713 rc = VERR_PARSE_ERROR;
714 else
715 pcszPort++;
716 }
717 else
718 rc = VERR_PARSE_ERROR;
719 }
720 }
721 else
722 rc = VERR_PARSE_ERROR;
723
724 /* Now split address into hostname and port. */
725 if (RT_SUCCESS(rc))
726 {
727 pImage->pszHostname = (char *)RTMemAlloc(cbHostname + 1);
728 if (!pImage->pszHostname)
729 rc = VERR_NO_MEMORY;
730 else
731 {
732 memcpy(pImage->pszHostname, pImage->pszTargetAddress, cbHostname);
733 pImage->pszHostname[cbHostname] = '\0';
734 if (pcszPort != NULL)
735 {
736 rc = RTStrToUInt16Ex(pcszPort, &pszPortEnd, 0, &uPort);
737 /* Note that RT_SUCCESS() macro to check the rc value is not strict enough in this case. */
738 if (rc == VINF_SUCCESS && *pszPortEnd == '\0' && uPort != 0)
739 {
740 pImage->uPort = uPort;
741 }
742 else
743 {
744 rc = VERR_PARSE_ERROR;
745 }
746 }
747 else
748 pImage->uPort = ISCSI_DEFAULT_PORT;
749 }
750 }
751
752 if (RT_FAILURE(rc))
753 {
754 if (pImage->pszHostname)
755 {
756 RTMemFree(pImage->pszHostname);
757 pImage->pszHostname = NULL;
758 }
759 pImage->uPort = 0;
760 }
761
762 /* Note that in this implementation the actual connection establishment is
763 * delayed until a PDU is read or written. */
764 LogFlowFunc(("returns %Rrc\n", rc));
765 return rc;
766}
767
768
769static int iscsiTransportClose(PISCSIIMAGE pImage)
770{
771 int rc;
772
773 LogFlowFunc(("(%s:%d)\n", pImage->pszHostname, pImage->uPort));
774 if (pImage->Socket != NIL_RTSOCKET)
775 {
776 rc = pImage->pInterfaceNetCallbacks->pfnClientClose(pImage->Socket);
777 pImage->Socket = NIL_RTSOCKET;
778 }
779 else
780 rc = VINF_SUCCESS;
781 LogFlowFunc(("returns %Rrc\n", rc));
782 return rc;
783}
784
785
786/**
787 * Attach to an iSCSI target. Performs all operations necessary to enter
788 * Full Feature Phase.
789 *
790 * @returns VBox status.
791 * @param pImage The iSCSI connection state to be used.
792 */
793static int iscsiAttach(PISCSIIMAGE pImage)
794{
795 int rc;
796 uint32_t itt;
797 uint32_t csg, nsg, substate;
798 uint64_t isid_tsih;
799 uint8_t bBuf[4096]; /* Should be large enough even for large authentication values. */
800 size_t cbBuf;
801 bool transit;
802 uint8_t pbChallenge[1024]; /* RFC3720 specifies this as maximum. */
803 size_t cbChallenge = 0; /* shut up gcc */
804 uint8_t bChapIdx;
805 uint8_t aResponse[RTMD5HASHSIZE];
806 uint32_t cnISCSIReq;
807 ISCSIREQ aISCSIReq[4];
808 uint32_t aReqBHS[12];
809 uint32_t cnISCSIRes;
810 ISCSIRES aISCSIRes[2];
811 uint32_t aResBHS[12];
812 char *pszNext;
813 LogFlowFunc(("entering\n"));
814
815 Assert(pImage->state == ISCSISTATE_FREE);
816
817 RTSemMutexRequest(pImage->Mutex, RT_INDEFINITE_WAIT);
818
819 /* Make 100% sure the connection isn't reused for a new login. */
820 iscsiTransportClose(pImage);
821
822restart:
823 pImage->state = ISCSISTATE_IN_LOGIN;
824 pImage->ITT = 1;
825 pImage->FirstRecvPDU = true;
826 pImage->CmdSN = 1;
827 pImage->ExpCmdSN = 0;
828 pImage->MaxCmdSN = 1;
829 pImage->ExpStatSN = 1;
830
831 /*
832 * Send login request to target.
833 */
834 itt = iscsiNewITT(pImage);
835 csg = 0;
836 nsg = 0;
837 substate = 0;
838 isid_tsih = pImage->ISID << 16; /* TSIH field currently always 0 */
839
840 do {
841 transit = false;
842 cbBuf = 0;
843 /* Handle all cases with a single switch statement. */
844 switch (csg << 8 | substate)
845 {
846 case 0x0000: /* security negotiation, step 0: propose authentication. */
847 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "SessionType", "Normal", 0);
848 if (RT_FAILURE(rc))
849 goto out;
850 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "InitiatorName", pImage->pszInitiatorName, 0);
851 if (RT_FAILURE(rc))
852 goto out;
853 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "TargetName", pImage->pszTargetName, 0);
854 if (RT_FAILURE(rc))
855 goto out;
856 if (pImage->pszInitiatorUsername == NULL)
857 {
858 /* No authentication. Immediately switch to next phase. */
859 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "AuthMethod", "None", 0);
860 if (RT_FAILURE(rc))
861 goto out;
862 nsg = 1;
863 transit = true;
864 }
865 else
866 {
867 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "AuthMethod", "CHAP,None", 0);
868 if (RT_FAILURE(rc))
869 goto out;
870 }
871 break;
872 case 0x0001: /* security negotiation, step 1: propose CHAP_MD5 variant. */
873 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "CHAP_A", "5", 0);
874 if (RT_FAILURE(rc))
875 goto out;
876 break;
877 case 0x0002: /* security negotiation, step 2: send authentication info. */
878 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "CHAP_N", pImage->pszInitiatorUsername, 0);
879 if (RT_FAILURE(rc))
880 goto out;
881 chap_md5_compute_response(aResponse, bChapIdx, pbChallenge, cbChallenge,
882 pImage->pbInitiatorSecret, pImage->cbInitiatorSecret);
883 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "CHAP_R", (const char *)aResponse, RTMD5HASHSIZE);
884 if (RT_FAILURE(rc))
885 goto out;
886 nsg = 1;
887 transit = true;
888 break;
889 case 0x0100: /* login operational negotiation, step 0: set parameters. */
890 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "HeaderDigest", "None", 0);
891 if (RT_FAILURE(rc))
892 goto out;
893 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "DataDigest", "None", 0);
894 if (RT_FAILURE(rc))
895 goto out;
896 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "MaxConnections", "1", 0);
897 if (RT_FAILURE(rc))
898 goto out;
899 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "InitialR2T", "No", 0);
900 if (RT_FAILURE(rc))
901 goto out;
902 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "ImmediateData", "Yes", 0);
903 if (RT_FAILURE(rc))
904 goto out;
905 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "MaxRecvDataSegmentLength", "65536", 0);
906 if (RT_FAILURE(rc))
907 goto out;
908 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "MaxBurstLength", "262144", 0);
909 if (RT_FAILURE(rc))
910 goto out;
911 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "FirstBurstLength", "65536", 0);
912 if (RT_FAILURE(rc))
913 goto out;
914 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "DefaultTime2Wait", "0", 0);
915 if (RT_FAILURE(rc))
916 goto out;
917 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "DefaultTime2Retain", "60", 0);
918 if (RT_FAILURE(rc))
919 goto out;
920 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "DataPDUInOrder", "Yes", 0);
921 if (RT_FAILURE(rc))
922 goto out;
923 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "DataSequenceInOrder", "Yes", 0);
924 if (RT_FAILURE(rc))
925 goto out;
926 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "ErrorRecoveryLevel", "0", 0);
927 if (RT_FAILURE(rc))
928 goto out;
929 rc = iscsiTextAddKeyValue(bBuf, sizeof(bBuf), &cbBuf, "MaxOutstandingR2T", "1", 0);
930 if (RT_FAILURE(rc))
931 goto out;
932 nsg = 3;
933 transit = true;
934 break;
935 case 0x0300: /* full feature phase. */
936 default:
937 /* Should never come here. */
938 AssertMsgFailed(("send: Undefined login state %d substate %d\n", csg, substate));
939 break;
940 }
941
942 aReqBHS[0] = RT_H2N_U32( ISCSI_IMMEDIATE_DELIVERY_BIT
943 | (csg << ISCSI_CSG_SHIFT)
944 | (transit ? (nsg << ISCSI_NSG_SHIFT | ISCSI_TRANSIT_BIT) : 0)
945 | ISCSI_MY_VERSION /* Minimum version. */
946 | (ISCSI_MY_VERSION << 8) /* Maximum version. */
947 | ISCSIOP_LOGIN_REQ); /* C=0 */
948 aReqBHS[1] = RT_H2N_U32((uint32_t)cbBuf); /* TotalAHSLength=0 */
949 aReqBHS[2] = RT_H2N_U32(isid_tsih >> 32);
950 aReqBHS[3] = RT_H2N_U32(isid_tsih & 0xffffffff);
951 aReqBHS[4] = itt;
952 aReqBHS[5] = RT_H2N_U32(1 << 16); /* CID=1,reserved */
953 aReqBHS[6] = RT_H2N_U32(pImage->CmdSN);
954 aReqBHS[7] = RT_H2N_U32(pImage->ExpStatSN);
955 aReqBHS[8] = 0; /* reserved */
956 aReqBHS[9] = 0; /* reserved */
957 aReqBHS[10] = 0; /* reserved */
958 aReqBHS[11] = 0; /* reserved */
959
960 cnISCSIReq = 0;
961 aISCSIReq[cnISCSIReq].pcvSeg = aReqBHS;
962 aISCSIReq[cnISCSIReq].cbSeg = sizeof(aReqBHS);
963 cnISCSIReq++;
964
965 aISCSIReq[cnISCSIReq].pcvSeg = bBuf;
966 aISCSIReq[cnISCSIReq].cbSeg = cbBuf;
967 cnISCSIReq++;
968
969 rc = iscsiSendPDU(pImage, aISCSIReq, cnISCSIReq);
970 if (RT_SUCCESS(rc))
971 {
972 ISCSIOPCODE cmd;
973 ISCSILOGINSTATUSCLASS loginStatusClass;
974
975 /* Place login request in queue. */
976 pImage->paCurrReq = aISCSIReq;
977 pImage->cnCurrReq = cnISCSIReq;
978
979 cnISCSIRes = 0;
980 aISCSIRes[cnISCSIRes].pvSeg = aResBHS;
981 aISCSIRes[cnISCSIRes].cbSeg = sizeof(aResBHS);
982 cnISCSIRes++;
983 aISCSIRes[cnISCSIRes].pvSeg = bBuf;
984 aISCSIRes[cnISCSIRes].cbSeg = sizeof(bBuf);
985 cnISCSIRes++;
986
987 rc = iscsiRecvPDU(pImage, itt, aISCSIRes, cnISCSIRes);
988 if (RT_FAILURE(rc))
989 break;
990 /** @todo collect partial login responses with Continue bit set. */
991 Assert(aISCSIRes[0].pvSeg == aResBHS);
992 Assert(aISCSIRes[0].cbSeg >= ISCSI_BHS_SIZE);
993 Assert((RT_N2H_U32(aResBHS[0]) & ISCSI_CONTINUE_BIT) == 0);
994
995 cmd = (ISCSIOPCODE)(RT_N2H_U32(aResBHS[0]) & ISCSIOP_MASK);
996 if (cmd == ISCSIOP_LOGIN_RES)
997 {
998 if ((RT_N2H_U32(aResBHS[0]) & 0xff) != ISCSI_MY_VERSION)
999 {
1000 iscsiTransportClose(pImage);
1001 rc = VERR_PARSE_ERROR;
1002 break; /* Give up immediately, as a RFC violation in version fields is very serious. */
1003 }
1004
1005 loginStatusClass = (ISCSILOGINSTATUSCLASS)(RT_N2H_U32(aResBHS[9]) >> 24);
1006 switch (loginStatusClass)
1007 {
1008 case ISCSI_LOGIN_STATUS_CLASS_SUCCESS:
1009 uint32_t targetCSG;
1010 uint32_t targetNSG;
1011 bool targetTransit;
1012
1013 if (pImage->FirstRecvPDU)
1014 {
1015 pImage->FirstRecvPDU = false;
1016 pImage->ExpStatSN = RT_N2H_U32(aResBHS[6]) + 1;
1017 }
1018
1019 targetCSG = (RT_N2H_U32(aResBHS[0]) & ISCSI_CSG_MASK) >> ISCSI_CSG_SHIFT;
1020 targetNSG = (RT_N2H_U32(aResBHS[0]) & ISCSI_NSG_MASK) >> ISCSI_NSG_SHIFT;
1021 targetTransit = !!(RT_N2H_U32(aResBHS[0]) & ISCSI_TRANSIT_BIT);
1022
1023 /* Handle all cases with a single switch statement. */
1024 switch (csg << 8 | substate)
1025 {
1026 case 0x0000: /* security negotiation, step 0: receive final authentication. */
1027 const char *pcszAuthMethod;
1028
1029 rc = iscsiTextGetKeyValue(bBuf, aISCSIRes[1].cbSeg, "AuthMethod", &pcszAuthMethod);
1030 if (RT_FAILURE(rc))
1031 {
1032 rc = VERR_PARSE_ERROR;
1033 break;
1034 }
1035 if (strcmp(pcszAuthMethod, "None") == 0)
1036 {
1037 /* Authentication offered, but none required. Skip to operational parameters. */
1038 csg = 1;
1039 nsg = 1;
1040 transit = true;
1041 substate = 0;
1042 break;
1043 }
1044 else if (strcmp(pcszAuthMethod, "CHAP") == 0 && targetNSG == 0 && !targetTransit)
1045 {
1046 /* CHAP authentication required, continue with next substate. */
1047 substate++;
1048 break;
1049 }
1050
1051 /* Unknown auth method or login response PDU headers incorrect. */
1052 rc = VERR_PARSE_ERROR;
1053 break;
1054 case 0x0001: /* security negotiation, step 1: receive final CHAP variant and challenge. */
1055 const char *pcszChapAuthMethod;
1056 const char *pcszChapIdxTarget;
1057 const char *pcszChapChallengeStr;
1058
1059 rc = iscsiTextGetKeyValue(bBuf, aISCSIRes[1].cbSeg, "CHAP_A", &pcszChapAuthMethod);
1060 if (RT_FAILURE(rc))
1061 {
1062 rc = VERR_PARSE_ERROR;
1063 break;
1064 }
1065 if (strcmp(pcszChapAuthMethod, "5") != 0)
1066 {
1067 rc = VERR_PARSE_ERROR;
1068 break;
1069 }
1070 rc = iscsiTextGetKeyValue(bBuf, aISCSIRes[1].cbSeg, "CHAP_I", &pcszChapIdxTarget);
1071 if (RT_FAILURE(rc))
1072 {
1073 rc = VERR_PARSE_ERROR;
1074 break;
1075 }
1076 rc = RTStrToUInt8Ex(pcszChapIdxTarget, &pszNext, 0, &bChapIdx);
1077 if ((rc > VINF_SUCCESS) || *pszNext != '\0')
1078 {
1079 rc = VERR_PARSE_ERROR;
1080 break;
1081 }
1082 rc = iscsiTextGetKeyValue(bBuf, aISCSIRes[1].cbSeg, "CHAP_C", &pcszChapChallengeStr);
1083 if (RT_FAILURE(rc))
1084 {
1085 rc = VERR_PARSE_ERROR;
1086 break;
1087 }
1088 cbChallenge = sizeof(pbChallenge);
1089 rc = iscsiStrToBinary(pcszChapChallengeStr, pbChallenge, &cbChallenge);
1090 if (RT_FAILURE(rc))
1091 break;
1092 substate++;
1093 transit = true;
1094 break;
1095 case 0x0002: /* security negotiation, step 2: check authentication success. */
1096 if (targetCSG == 0 && targetNSG == 1 && targetTransit)
1097 {
1098 /* Target wants to continue in login operational state, authentication success. */
1099 csg = 1;
1100 nsg = 3;
1101 substate = 0;
1102 break;
1103 }
1104 rc = VERR_PARSE_ERROR;
1105 break;
1106 case 0x0100: /* login operational negotiation, step 0: check results. */
1107 if (targetCSG == 1 && targetNSG == 3 && targetTransit)
1108 {
1109 /* Target wants to continue in full feature phase, login finished. */
1110 csg = 3;
1111 nsg = 3;
1112 substate = 0;
1113 break;
1114 }
1115 rc = VERR_PARSE_ERROR;
1116 break;
1117 case 0x0300: /* full feature phase. */
1118 default:
1119 AssertMsgFailed(("recv: Undefined login state %d substate %d\n", csg, substate));
1120 rc = VERR_PARSE_ERROR;
1121 break;
1122 }
1123 break;
1124 case ISCSI_LOGIN_STATUS_CLASS_REDIRECTION:
1125 const char *pcszTargetRedir;
1126
1127 /* Target has moved to some other location, as indicated in the TargetAddress key. */
1128 rc = iscsiTextGetKeyValue(bBuf, aISCSIRes[1].cbSeg, "TargetAddress", &pcszTargetRedir);
1129 if (RT_FAILURE(rc))
1130 {
1131 rc = VERR_PARSE_ERROR;
1132 break;
1133 }
1134 if (pImage->pszTargetAddress)
1135 RTMemFree(pImage->pszTargetAddress);
1136 {
1137 size_t cb = strlen(pcszTargetRedir) + 1;
1138 pImage->pszTargetAddress = (char *)RTMemAlloc(cb);
1139 if (!pImage->pszTargetAddress)
1140 {
1141 rc = VERR_NO_MEMORY;
1142 break;
1143 }
1144 memcpy(pImage->pszTargetAddress, pcszTargetRedir, cb);
1145 }
1146 rc = iscsiTransportOpen(pImage);
1147 goto restart;
1148 case ISCSI_LOGIN_STATUS_CLASS_INITIATOR_ERROR:
1149 iscsiTransportClose(pImage);
1150 pImage->paCurrReq = NULL;
1151 pImage->cnCurrReq = 0;
1152 rc = VERR_IO_GEN_FAILURE;
1153 goto out;
1154 case ISCSI_LOGIN_STATUS_CLASS_TARGET_ERROR:
1155 iscsiTransportClose(pImage);
1156 rc = VINF_EOF;
1157 break;
1158 default:
1159 rc = VERR_PARSE_ERROR;
1160 }
1161
1162 /* Remove login request from queue. */
1163 pImage->paCurrReq = NULL;
1164 pImage->cnCurrReq = 0;
1165
1166 if (csg == 3)
1167 {
1168 /*
1169 * Finished login, continuing with Full Feature Phase.
1170 */
1171 rc = VINF_SUCCESS;
1172 break;
1173 }
1174 }
1175 else
1176 {
1177 AssertMsgFailed(("%s: ignoring unexpected PDU with first word = %#08x\n", __FUNCTION__, RT_N2H_U32(aResBHS[0])));
1178 }
1179 }
1180 else
1181 break;
1182 } while (true);
1183
1184out:
1185 if (RT_FAILURE(rc))
1186 {
1187 /*
1188 * Close connection to target.
1189 */
1190 iscsiTransportClose(pImage);
1191 pImage->state = ISCSISTATE_FREE;
1192 }
1193 else
1194 pImage->state = ISCSISTATE_NORMAL;
1195
1196 RTSemMutexRelease(pImage->Mutex);
1197
1198 LogFlowFunc(("returning %Rrc\n", rc));
1199 LogRel(("iSCSI: login to target %s %s\n", pImage->pszTargetName, RT_SUCCESS(rc) ? "successful" : "failed"));
1200 return rc;
1201}
1202
1203
1204/**
1205 * Detach from an iSCSI target.
1206 *
1207 * @returns VBox status.
1208 * @param pImage The iSCSI connection state to be used.
1209 */
1210static int iscsiDetach(PISCSIIMAGE pImage)
1211{
1212 int rc;
1213 uint32_t itt;
1214 uint32_t cnISCSIReq = 0;
1215 ISCSIREQ aISCSIReq[4];
1216 uint32_t aReqBHS[12];
1217 LogFlow(("drvISCSIDetach: entering\n"));
1218
1219 RTSemMutexRequest(pImage->Mutex, RT_INDEFINITE_WAIT);
1220
1221 if (pImage->state != ISCSISTATE_FREE && pImage->state != ISCSISTATE_IN_LOGOUT)
1222 {
1223 pImage->state = ISCSISTATE_IN_LOGOUT;
1224
1225 /*
1226 * Send logout request to target.
1227 */
1228 itt = iscsiNewITT(pImage);
1229 aReqBHS[0] = RT_H2N_U32(ISCSI_FINAL_BIT | ISCSIOP_LOGOUT_REQ); /* I=0,F=1,Reason=close session */
1230 aReqBHS[1] = RT_H2N_U32(0); /* TotalAHSLength=0,DataSementLength=0 */
1231 aReqBHS[2] = 0; /* reserved */
1232 aReqBHS[3] = 0; /* reserved */
1233 aReqBHS[4] = itt;
1234 aReqBHS[5] = 0; /* reserved */
1235 aReqBHS[6] = RT_H2N_U32(pImage->CmdSN);
1236 aReqBHS[7] = RT_H2N_U32(pImage->ExpStatSN);
1237 aReqBHS[8] = 0; /* reserved */
1238 aReqBHS[9] = 0; /* reserved */
1239 aReqBHS[10] = 0; /* reserved */
1240 aReqBHS[11] = 0; /* reserved */
1241 pImage->CmdSN++;
1242
1243 aISCSIReq[cnISCSIReq].pcvSeg = aReqBHS;
1244 aISCSIReq[cnISCSIReq].cbSeg = sizeof(aReqBHS);
1245 cnISCSIReq++;
1246
1247 rc = iscsiSendPDU(pImage, aISCSIReq, cnISCSIReq);
1248 if (RT_SUCCESS(rc))
1249 {
1250 /* Place logout request in queue. */
1251 pImage->paCurrReq = aISCSIReq;
1252 pImage->cnCurrReq = cnISCSIReq;
1253
1254 /*
1255 * Read logout response from target.
1256 */
1257 ISCSIRES aISCSIRes;
1258 uint32_t aResBHS[12];
1259
1260 aISCSIRes.pvSeg = aResBHS;
1261 aISCSIRes.cbSeg = sizeof(aResBHS);
1262 rc = iscsiRecvPDU(pImage, itt, &aISCSIRes, 1);
1263 if (RT_SUCCESS(rc))
1264 {
1265 if (RT_N2H_U32(aResBHS[0]) != (ISCSI_FINAL_BIT | ISCSIOP_LOGOUT_RES))
1266 AssertMsgFailed(("iSCSI Logout response invalid\n"));
1267 }
1268 else
1269 AssertMsgFailed(("iSCSI Logout response error, rc=%Rrc\n", rc));
1270
1271 /* Remove logout request from queue. */
1272 pImage->paCurrReq = NULL;
1273 pImage->cnCurrReq = 0;
1274 }
1275 else
1276 AssertMsgFailed(("Could not send iSCSI Logout request, rc=%Rrc\n", rc));
1277 }
1278
1279 if (pImage->state != ISCSISTATE_FREE)
1280 {
1281 /*
1282 * Close connection to target.
1283 */
1284 rc = iscsiTransportClose(pImage);
1285 if (RT_FAILURE(rc))
1286 AssertMsgFailed(("Could not close connection to target, rc=%Rrc\n", rc));
1287 }
1288
1289 pImage->state = ISCSISTATE_FREE;
1290
1291 RTSemMutexRelease(pImage->Mutex);
1292
1293 LogFlow(("drvISCSIDetach: leaving\n"));
1294 LogRel(("iSCSI: logout to target %s\n", pImage->pszTargetName));
1295 return VINF_SUCCESS;
1296}
1297
1298
1299/**
1300 * Perform a command on an iSCSI target. Target must be already in
1301 * Full Feature Phase.
1302 *
1303 * @returns VBOX status.
1304 * @param pImage The iSCSI connection state to be used.
1305 * @param pRequest Command descriptor. Contains all information about
1306 * the command, its transfer directions and pointers
1307 * to the buffer(s) used for transferring data and
1308 * status information.
1309 */
1310static int iscsiCommand(PISCSIIMAGE pImage, PSCSIREQ pRequest)
1311{
1312 int rc;
1313 uint32_t itt;
1314 uint32_t cbData;
1315 uint32_t cnISCSIReq = 0;
1316 ISCSIREQ aISCSIReq[4];
1317 uint32_t aReqBHS[12];
1318
1319 uint32_t *pDst = NULL;
1320 size_t cbBufLength;
1321 uint32_t aStat[64];
1322 uint32_t ExpDataSN = 0;
1323 bool final = false;
1324
1325 LogFlow(("iscsiCommand: entering, CmdSN=%d\n", pImage->CmdSN));
1326
1327 Assert(pRequest->enmXfer != SCSIXFER_TO_FROM_TARGET); /**< @todo not yet supported, would require AHS. */
1328 Assert(pRequest->cbI2TData <= 0xffffff); /* larger transfers would require R2T support. */
1329 Assert(pRequest->cbCmd <= 16); /* would cause buffer overrun below. */
1330
1331 /* If not in normal state, then the transport connection was dropped. Try
1332 * to reestablish by logging in, the target might be responsive again. */
1333 if (pImage->state == ISCSISTATE_FREE)
1334 rc = iscsiAttach(pImage);
1335
1336 /* If still not in normal state, then the underlying transport connection
1337 * cannot be established. Get out before bad things happen (and make
1338 * sure the caller suspends the VM again). */
1339 if (pImage->state != ISCSISTATE_NORMAL)
1340 {
1341 rc = VERR_NET_CONNECTION_REFUSED;
1342 goto out;
1343 }
1344
1345 /*
1346 * Send SCSI command to target with all I2T data included.
1347 */
1348 cbData = 0;
1349 if (pRequest->enmXfer == SCSIXFER_FROM_TARGET)
1350 cbData = (uint32_t)pRequest->cbT2IData;
1351 else
1352 cbData = (uint32_t)pRequest->cbI2TData;
1353
1354 RTSemMutexRequest(pImage->Mutex, RT_INDEFINITE_WAIT);
1355
1356 itt = iscsiNewITT(pImage);
1357 memset(aReqBHS, 0, sizeof(aReqBHS));
1358 aReqBHS[0] = RT_H2N_U32( ISCSI_FINAL_BIT | ISCSI_TASK_ATTR_ORDERED | ISCSIOP_SCSI_CMD
1359 | (pRequest->enmXfer << 21)); /* I=0,F=1,Attr=Ordered */
1360 aReqBHS[1] = RT_H2N_U32(0x00000000 | ((uint32_t)pRequest->cbI2TData & 0xffffff)); /* TotalAHSLength=0 */
1361 aReqBHS[2] = RT_H2N_U32(pImage->LUN >> 32);
1362 aReqBHS[3] = RT_H2N_U32(pImage->LUN & 0xffffffff);
1363 aReqBHS[4] = itt;
1364 aReqBHS[5] = RT_H2N_U32(cbData);
1365 aReqBHS[6] = RT_H2N_U32(pImage->CmdSN);
1366 aReqBHS[7] = RT_H2N_U32(pImage->ExpStatSN);
1367 memcpy(aReqBHS + 8, pRequest->pvCmd, pRequest->cbCmd);
1368 pImage->CmdSN++;
1369
1370 aISCSIReq[cnISCSIReq].pcvSeg = aReqBHS;
1371 aISCSIReq[cnISCSIReq].cbSeg = sizeof(aReqBHS);
1372 cnISCSIReq++;
1373
1374 if ( pRequest->enmXfer == SCSIXFER_TO_TARGET
1375 || pRequest->enmXfer == SCSIXFER_TO_FROM_TARGET)
1376 {
1377 aISCSIReq[cnISCSIReq].pcvSeg = pRequest->pcvI2TData;
1378 aISCSIReq[cnISCSIReq].cbSeg = pRequest->cbI2TData; /* Padding done by transport. */
1379 cnISCSIReq++;
1380 }
1381
1382 rc = iscsiSendPDU(pImage, aISCSIReq, cnISCSIReq);
1383 if (RT_FAILURE(rc))
1384 goto out_release;
1385
1386 /* Place SCSI request in queue. */
1387 pImage->paCurrReq = aISCSIReq;
1388 pImage->cnCurrReq = cnISCSIReq;
1389
1390 /*
1391 * Read SCSI response/data in PDUs from target.
1392 */
1393 if ( pRequest->enmXfer == SCSIXFER_FROM_TARGET
1394 || pRequest->enmXfer == SCSIXFER_TO_FROM_TARGET)
1395 {
1396 pDst = (uint32_t *)pRequest->pvT2IData;
1397 cbBufLength = pRequest->cbT2IData;
1398 }
1399 else
1400 cbBufLength = 0;
1401
1402 do {
1403 uint32_t cnISCSIRes = 0;
1404 ISCSIRES aISCSIRes[4];
1405 uint32_t aResBHS[12];
1406
1407 aISCSIRes[cnISCSIRes].pvSeg = aResBHS;
1408 aISCSIRes[cnISCSIRes].cbSeg = sizeof(aResBHS);
1409 cnISCSIRes++;
1410 if (cbBufLength != 0 &&
1411 ( pRequest->enmXfer == SCSIXFER_FROM_TARGET
1412 || pRequest->enmXfer == SCSIXFER_TO_FROM_TARGET))
1413 {
1414 aISCSIRes[cnISCSIRes].pvSeg = pDst;
1415 aISCSIRes[cnISCSIRes].cbSeg = cbBufLength;
1416 cnISCSIRes++;
1417 }
1418 /* Always reserve space for the status - it's impossible to tell
1419 * beforehand whether this will be the final PDU or not. */
1420 aISCSIRes[cnISCSIRes].pvSeg = aStat;
1421 aISCSIRes[cnISCSIRes].cbSeg = sizeof(aStat);
1422 cnISCSIRes++;
1423
1424 rc = iscsiRecvPDU(pImage, itt, aISCSIRes, cnISCSIRes);
1425 if (RT_FAILURE(rc))
1426 break;
1427
1428 final = !!(RT_N2H_U32(aResBHS[0]) & ISCSI_FINAL_BIT);
1429 ISCSIOPCODE cmd = (ISCSIOPCODE)(RT_N2H_U32(aResBHS[0]) & ISCSIOP_MASK);
1430 if (cmd == ISCSIOP_SCSI_RES)
1431 {
1432 /* This is the final PDU which delivers the status (and may be omitted if
1433 * the last Data-In PDU included successful completion status). */
1434 if (!final || ((RT_N2H_U32(aResBHS[0]) & 0x0000ff00) != 0) || (RT_N2H_U32(aResBHS[9]) != ExpDataSN))
1435 {
1436 /* SCSI Response in the wrong place or with a (target) failure. */
1437 rc = VERR_PARSE_ERROR;
1438 break;
1439 }
1440 pRequest->status = RT_N2H_U32(aResBHS[0]) & 0x000000ff;
1441 uint32_t cbData = RT_N2H_U32(aResBHS[1]) & 0x00ffffff;
1442 if (cbData >= 2)
1443 {
1444 uint32_t cbStat = RT_N2H_U32(aStat[0]) >> 16;
1445 if (cbStat + 2 > cbData || cbStat > pRequest->cbSense)
1446 {
1447 rc = VERR_BUFFER_OVERFLOW;
1448 break;
1449 }
1450 pRequest->cbSense = RT_N2H_U32(aStat[0]) >> 16;
1451 memcpy(pRequest->pvSense, ((const uint8_t *)aStat) + 2, pRequest->cbSense);
1452 }
1453 else if (cbData == 1)
1454 {
1455 rc = VERR_PARSE_ERROR;
1456 break;
1457 }
1458 break;
1459 }
1460 else if (cmd == ISCSIOP_SCSI_DATA_IN)
1461 {
1462 /* A Data-In PDU carries some data that needs to be added to the received
1463 * data in response to the command. There may be both partial and complete
1464 * Data-In PDUs, so collect data until the status is included or the status
1465 * is sent in a separate SCSI Result frame (see above). */
1466 if (final && aISCSIRes[2].cbSeg != 0)
1467 {
1468 /* The received PDU is partially stored in the buffer for status.
1469 * Must not happen under normal circumstances and is a target error. */
1470 rc = VERR_BUFFER_OVERFLOW;
1471 break;
1472 }
1473 uint32_t len = RT_N2H_U32(aResBHS[1]) & 0x00ffffff;
1474 pDst = (uint32_t *)((char *)pDst + len);
1475 cbBufLength -= len;
1476 ExpDataSN++;
1477 if (final && (RT_N2H_U32(aResBHS[0]) & ISCSI_STATUS_BIT) != 0)
1478 {
1479 pRequest->status = RT_N2H_U32(aResBHS[0]) & 0x000000ff;
1480 pRequest->cbSense = 0;
1481 break;
1482 }
1483 }
1484 else
1485 {
1486 rc = VERR_PARSE_ERROR;
1487 break;
1488 }
1489 } while (true);
1490
1491 /* Remove SCSI request from queue. */
1492 pImage->paCurrReq = NULL;
1493 pImage->cnCurrReq = 0;
1494
1495out_release:
1496 if (rc == VERR_TIMEOUT)
1497 {
1498 /* Drop connection in case the target plays dead. Much better than
1499 * delaying the next requests until the timed out command actually
1500 * finishes. Also keep in mind that command shouldn't take longer than
1501 * about 30-40 seconds, or the guest will lose its patience. */
1502 iscsiTransportClose(pImage);
1503 pImage->state = ISCSISTATE_FREE;
1504 }
1505 RTSemMutexRelease(pImage->Mutex);
1506
1507out:
1508 LogFlow(("iscsiCommand: returns %Rrc\n", rc));
1509 return rc;
1510}
1511
1512
1513/**
1514 * Generate a new Initiator Task Tag.
1515 *
1516 * @returns Initiator Task Tag.
1517 * @param pImage The iSCSI connection state to be used.
1518 */
1519static uint32_t iscsiNewITT(PISCSIIMAGE pImage)
1520{
1521 uint32_t next_itt;
1522
1523 next_itt = pImage->ITT++;
1524 if (pImage->ITT == ISCSI_TASK_TAG_RSVD)
1525 pImage->ITT = 0;
1526 return RT_H2N_U32(next_itt);
1527}
1528
1529
1530/**
1531 * Send an iSCSI request. The request can consist of several segments, which
1532 * are padded to 4 byte boundaries and concatenated.
1533 *
1534 * @returns VBOX status
1535 * @param pImage The iSCSI connection state to be used.
1536 * @param paReq Pointer to array of iSCSI request sections.
1537 * @param cnReq Number of valid iSCSI request sections in the array.
1538 */
1539static int iscsiSendPDU(PISCSIIMAGE pImage, PISCSIREQ paReq, uint32_t cnReq)
1540{
1541 int rc = VINF_SUCCESS;
1542 uint32_t i;
1543 /** @todo return VERR_VD_ISCSI_INVALID_STATE in the appropriate situations,
1544 * needs cleaning up of timeout/disconnect handling a bit, as otherwise
1545 * too many incorrect errors are signalled. */
1546 Assert(pImage->paCurrReq == NULL);
1547 Assert(cnReq >= 1);
1548 Assert(paReq[0].cbSeg >= ISCSI_BHS_SIZE);
1549
1550 for (i = 0; i < pImage->cISCSIRetries; i++)
1551 {
1552 rc = iscsiTransportWrite(pImage, paReq, cnReq);
1553 if (RT_SUCCESS(rc))
1554 break;
1555 if (rc != VERR_BROKEN_PIPE && rc != VERR_NET_CONNECTION_REFUSED)
1556 break;
1557 RTThreadSleep(500);
1558 if ( pImage->state != ISCSISTATE_IN_LOGIN
1559 && pImage->state != ISCSISTATE_IN_LOGOUT)
1560 {
1561 /* Attempt to re-login when a connection fails, but only when not
1562 * currently logging in or logging out. */
1563 rc = iscsiAttach(pImage);
1564 if (RT_FAILURE(rc))
1565 break;
1566 }
1567 }
1568 return rc;
1569}
1570
1571
1572/**
1573 * Wait for an iSCSI response with a matching Initiator Target Tag. The response is
1574 * split into several segments, as requested by the caller-provided buffer specification.
1575 * Remember that the response can be split into several PDUs by the sender, so make
1576 * sure that all parts are collected and processed appropriately by the caller.
1577 *
1578 * @returns VBOX status
1579 * @param pImage The iSCSI connection state to be used.
1580 * @param paRes Pointer to array of iSCSI response sections.
1581 * @param cnRes Number of valid iSCSI response sections in the array.
1582 */
1583static int iscsiRecvPDU(PISCSIIMAGE pImage, uint32_t itt, PISCSIRES paRes, uint32_t cnRes)
1584{
1585 int rc = VINF_SUCCESS;
1586 uint32_t i;
1587 ISCSIRES aResBuf;
1588
1589 for (i = 0; i < pImage->cISCSIRetries; i++)
1590 {
1591 aResBuf.pvSeg = pImage->pvRecvPDUBuf;
1592 aResBuf.cbSeg = pImage->cbRecvPDUBuf;
1593 rc = iscsiTransportRead(pImage, &aResBuf, 1);
1594 if (RT_FAILURE(rc))
1595 {
1596 if (rc == VERR_BROKEN_PIPE || rc == VERR_NET_CONNECTION_REFUSED)
1597 {
1598 /* Connection broken while waiting for a response - wait a while and
1599 * try to restart by re-sending the original request (if any).
1600 * This also handles the connection reestablishment (login etc.). */
1601 RTThreadSleep(500);
1602 if ( pImage->state != ISCSISTATE_IN_LOGIN
1603 && pImage->state != ISCSISTATE_IN_LOGOUT)
1604 {
1605 /* Attempt to re-login when a connection fails, but only when not
1606 * currently logging in or logging out. */
1607 rc = iscsiAttach(pImage);
1608 if (RT_FAILURE(rc))
1609 break;
1610 }
1611 if (pImage->paCurrReq != NULL)
1612 {
1613 rc = iscsiSendPDU(pImage, pImage->paCurrReq, pImage->cnCurrReq);
1614 if (RT_FAILURE(rc))
1615 break;
1616 }
1617 }
1618 else
1619 {
1620 /* Signal other errors (VERR_BUFFER_OVERFLOW etc.) to the caller. */
1621 break;
1622 }
1623 }
1624 else
1625 {
1626 ISCSIOPCODE cmd;
1627 const uint32_t *pcvResSeg = (const uint32_t *)aResBuf.pvSeg;
1628
1629 /* Check whether the received PDU is valid, and update the internal state of
1630 * the iSCSI connection/session. */
1631 rc = drvISCSIValidatePDU(&aResBuf, 1);
1632 if (RT_FAILURE(rc))
1633 continue;
1634 cmd = (ISCSIOPCODE)(RT_N2H_U32(pcvResSeg[0]) & ISCSIOP_MASK);
1635 switch (cmd)
1636 {
1637 case ISCSIOP_SCSI_RES:
1638 case ISCSIOP_SCSI_TASKMGMT_RES:
1639 case ISCSIOP_SCSI_DATA_IN:
1640 case ISCSIOP_R2T:
1641 case ISCSIOP_ASYN_MSG:
1642 case ISCSIOP_TEXT_RES:
1643 case ISCSIOP_LOGIN_RES:
1644 case ISCSIOP_LOGOUT_RES:
1645 case ISCSIOP_REJECT:
1646 case ISCSIOP_NOP_IN:
1647 if (serial_number_less(pImage->MaxCmdSN, RT_N2H_U32(pcvResSeg[8])))
1648 pImage->MaxCmdSN = RT_N2H_U32(pcvResSeg[8]);
1649 if (serial_number_less(pImage->ExpCmdSN, RT_N2H_U32(pcvResSeg[7])))
1650 pImage->ExpCmdSN = RT_N2H_U32(pcvResSeg[7]);
1651 break;
1652 default:
1653 rc = VERR_PARSE_ERROR;
1654 }
1655 if (RT_FAILURE(rc))
1656 continue;
1657 if ( !pImage->FirstRecvPDU
1658 && (cmd != ISCSIOP_SCSI_DATA_IN || (RT_N2H_U32(pcvResSeg[0]) & ISCSI_STATUS_BIT)))
1659 {
1660 if (pImage->ExpStatSN == RT_N2H_U32(pcvResSeg[6]))
1661 {
1662 /* StatSN counter is not advanced on R2T and on a target SN update NOP-In. */
1663 if ( (cmd != ISCSIOP_R2T)
1664 && ((cmd != ISCSIOP_NOP_IN) || (RT_N2H_U32(pcvResSeg[4]) != ISCSI_TASK_TAG_RSVD)))
1665 pImage->ExpStatSN++;
1666 }
1667 else
1668 {
1669 rc = VERR_PARSE_ERROR;
1670 continue;
1671 }
1672 }
1673 /* Finally check whether the received PDU matches what the caller wants. */
1674 if (itt == pcvResSeg[4])
1675 {
1676 /* Copy received PDU (one segment) to caller-provided buffers. */
1677 uint32_t i;
1678 size_t cbSeg;
1679 const uint8_t *pSrc;
1680
1681 pSrc = (const uint8_t *)aResBuf.pvSeg;
1682 cbSeg = aResBuf.cbSeg;
1683 for (i = 0; i < cnRes; i++)
1684 {
1685 if (cbSeg > paRes[i].cbSeg)
1686 {
1687 memcpy(paRes[i].pvSeg, pSrc, paRes[i].cbSeg);
1688 pSrc += paRes[i].cbSeg;
1689 cbSeg -= paRes[i].cbSeg;
1690 }
1691 else
1692 {
1693 memcpy(paRes[i].pvSeg, pSrc, cbSeg);
1694 paRes[i].cbSeg = cbSeg;
1695 cbSeg = 0;
1696 break;
1697 }
1698 }
1699 if (cbSeg != 0)
1700 {
1701 rc = VERR_BUFFER_OVERFLOW;
1702 break;
1703 }
1704 for (i++; i < cnRes; i++)
1705 paRes[i].cbSeg = 0;
1706 break;
1707 }
1708 else if ( cmd == ISCSIOP_NOP_IN
1709 && RT_N2H_U32(pcvResSeg[5]) != ISCSI_TASK_TAG_RSVD)
1710 {
1711 const uint32_t *pcvResSeg = (const uint32_t *)aResBuf.pvSeg;
1712 uint32_t cnISCSIReq;
1713 ISCSIREQ aISCSIReq[4];
1714 uint32_t aReqBHS[12];
1715
1716 aReqBHS[0] = RT_H2N_U32(ISCSI_IMMEDIATE_DELIVERY_BIT | ISCSI_FINAL_BIT | ISCSIOP_NOP_OUT);
1717 aReqBHS[1] = RT_H2N_U32(0); /* TotalAHSLength=0,DataSementLength=0 */
1718 aReqBHS[2] = pcvResSeg[2]; /* copy LUN from NOP-In */
1719 aReqBHS[3] = pcvResSeg[3]; /* copy LUN from NOP-In */
1720 aReqBHS[4] = RT_H2N_U32(ISCSI_TASK_TAG_RSVD); /* ITT, reply */
1721 aReqBHS[5] = pcvResSeg[5]; /* copy TTT from NOP-In */
1722 aReqBHS[6] = RT_H2N_U32(pImage->CmdSN);
1723 aReqBHS[7] = RT_H2N_U32(pImage->ExpStatSN);
1724 aReqBHS[8] = 0; /* reserved */
1725 aReqBHS[9] = 0; /* reserved */
1726 aReqBHS[10] = 0; /* reserved */
1727 aReqBHS[11] = 0; /* reserved */
1728
1729 cnISCSIReq = 0;
1730 aISCSIReq[cnISCSIReq].pcvSeg = aReqBHS;
1731 aISCSIReq[cnISCSIReq].cbSeg = sizeof(aReqBHS);
1732 cnISCSIReq++;
1733
1734 iscsiSendPDU(pImage, aISCSIReq, cnISCSIReq);
1735 }
1736 }
1737 }
1738 return rc;
1739}
1740
1741
1742/**
1743 * Check the static (not dependent on the connection/session state) validity of an iSCSI response PDU.
1744 *
1745 * @returns VBOX status
1746 * @param paRes Pointer to array of iSCSI response sections.
1747 * @param cnRes Number of valid iSCSI response sections in the array.
1748 */
1749static int drvISCSIValidatePDU(PISCSIRES paRes, uint32_t cnRes)
1750{
1751 const uint32_t *pcrgResBHS;
1752 uint32_t hw0;
1753 Assert(cnRes >= 1);
1754 Assert(paRes[0].cbSeg >= ISCSI_BHS_SIZE);
1755
1756 pcrgResBHS = (const uint32_t *)(paRes[0].pvSeg);
1757 hw0 = RT_N2H_U32(pcrgResBHS[0]);
1758 switch (hw0 & ISCSIOP_MASK)
1759 {
1760 case ISCSIOP_NOP_IN:
1761 /* NOP-In responses must not be split into several PDUs nor it may contain
1762 * ping data for target-initiated pings nor may both task tags be valid task tags. */
1763 if ( (hw0 & ISCSI_FINAL_BIT) == 0
1764 || ( RT_N2H_U32(pcrgResBHS[4]) == ISCSI_TASK_TAG_RSVD
1765 && RT_N2H_U32(pcrgResBHS[1]) != 0)
1766 || ( RT_N2H_U32(pcrgResBHS[4]) != ISCSI_TASK_TAG_RSVD
1767 && RT_N2H_U32(pcrgResBHS[5]) != ISCSI_TASK_TAG_RSVD))
1768 return VERR_PARSE_ERROR;
1769 break;
1770 case ISCSIOP_SCSI_RES:
1771 /* SCSI responses must not be split into several PDUs nor must the residual
1772 * bits be contradicting each other nor may the residual bits be set for PDUs
1773 * containing anything else but a completed command response. Underflow
1774 * is no reason for declaring a PDU as invalid, as the target may choose
1775 * to return less data than we assume to get. */
1776 if ( (hw0 & ISCSI_FINAL_BIT) == 0
1777 || ((hw0 & ISCSI_BI_READ_RESIDUAL_OVFL_BIT) && (hw0 & ISCSI_BI_READ_RESIDUAL_UNFL_BIT))
1778 || ((hw0 & ISCSI_RESIDUAL_OVFL_BIT) && (hw0 & ISCSI_RESIDUAL_UNFL_BIT))
1779 || ( ((hw0 & ISCSI_SCSI_RESPONSE_MASK) == 0)
1780 && ((hw0 & ISCSI_SCSI_STATUS_MASK) == SCSI_STATUS_OK)
1781 && (hw0 & ( ISCSI_BI_READ_RESIDUAL_OVFL_BIT | ISCSI_BI_READ_RESIDUAL_UNFL_BIT
1782 | ISCSI_RESIDUAL_OVFL_BIT))))
1783 return VERR_PARSE_ERROR;
1784 break;
1785 case ISCSIOP_LOGIN_RES:
1786 /* Login responses must not contain contradicting transit and continue bits. */
1787 if ((hw0 & ISCSI_CONTINUE_BIT) && ((hw0 & ISCSI_TRANSIT_BIT) != 0))
1788 return VERR_PARSE_ERROR;
1789 break;
1790 case ISCSIOP_TEXT_RES:
1791 /* Text responses must not contain contradicting final and continue bits nor
1792 * may the final bit be set for PDUs containing a target transfer tag other than
1793 * the reserved transfer tag (and vice versa). */
1794 if ( (((hw0 & ISCSI_CONTINUE_BIT) && (hw0 & ISCSI_FINAL_BIT) != 0))
1795 || (((hw0 & ISCSI_FINAL_BIT) && (RT_N2H_U32(pcrgResBHS[5]) != ISCSI_TASK_TAG_RSVD)))
1796 || (((hw0 & ISCSI_FINAL_BIT) == 0) && (RT_N2H_U32(pcrgResBHS[5]) == ISCSI_TASK_TAG_RSVD)))
1797 return VERR_PARSE_ERROR;
1798 break;
1799 case ISCSIOP_SCSI_DATA_IN:
1800 /* SCSI Data-in responses must not contain contradicting residual bits when
1801 * status bit is set. */
1802 if ((hw0 & ISCSI_STATUS_BIT) && (hw0 & ISCSI_RESIDUAL_OVFL_BIT) && (hw0 & ISCSI_RESIDUAL_UNFL_BIT))
1803 return VERR_PARSE_ERROR;
1804 break;
1805 case ISCSIOP_LOGOUT_RES:
1806 /* Logout responses must not have the final bit unset and may not contain any
1807 * data or additional header segments. */
1808 if ( ((hw0 & ISCSI_FINAL_BIT) == 0)
1809 || (RT_N2H_U32(pcrgResBHS[1]) != 0))
1810 return VERR_PARSE_ERROR;
1811 break;
1812 case ISCSIOP_ASYN_MSG:
1813 /* Asynchronous Messages must not have the final bit unser and may not contain
1814 * an initiator task tag. */
1815 if ( ((hw0 & ISCSI_FINAL_BIT) == 0)
1816 || (RT_N2H_U32(pcrgResBHS[4]) != ISCSI_TASK_TAG_RSVD))
1817 return VERR_PARSE_ERROR;
1818 break;
1819 case ISCSIOP_SCSI_TASKMGMT_RES:
1820 case ISCSIOP_R2T:
1821 case ISCSIOP_REJECT:
1822 default:
1823 /* Do some logging, ignore PDU. */
1824 LogFlow(("drvISCSIValidatePDU: ignore unhandled PDU, first word %#08x\n", RT_N2H_U32(pcrgResBHS[0])));
1825 return VERR_PARSE_ERROR;
1826 }
1827 /* A target must not send PDUs with MaxCmdSN less than ExpCmdSN-1. */
1828
1829 if (serial_number_less(RT_N2H_U32(pcrgResBHS[8]), RT_N2H_U32(pcrgResBHS[7])-1))
1830 return VERR_PARSE_ERROR;
1831
1832 return VINF_SUCCESS;
1833}
1834
1835
1836/**
1837 * Appends a key-value pair to the buffer. Normal ASCII strings (cbValue == 0) and large binary values
1838 * of a given length (cbValue > 0) are directly supported. Other value types must be converted to ASCII
1839 * by the caller. Strings must be in UTF-8 encoding.
1840 *
1841 * @returns VBOX status
1842 * @param pbBuf Pointer to the key-value buffer.
1843 * @param cbBuf Length of the key-value buffer.
1844 * @param pcbBufCurr Currently used portion of the key-value buffer.
1845 * @param pszKey Pointer to a string containing the key.
1846 * @param pszValue Pointer to either a string containing the value or to a large binary value.
1847 * @param cbValue Length of the binary value if applicable.
1848 */
1849static int iscsiTextAddKeyValue(uint8_t *pbBuf, size_t cbBuf, size_t *pcbBufCurr, const char *pcszKey,
1850 const char *pcszValue, size_t cbValue)
1851{
1852 size_t cbBufTmp = *pcbBufCurr;
1853 size_t cbKey = strlen(pcszKey);
1854 size_t cbValueEnc;
1855 uint8_t *pbCurr;
1856
1857 if (cbValue == 0)
1858 cbValueEnc = strlen(pcszValue);
1859 else
1860 cbValueEnc = cbValue * 2 + 2; /* 2 hex bytes per byte, 2 bytes prefix */
1861
1862 if (cbBuf < cbBufTmp + cbKey + 1 + cbValueEnc + 1)
1863 {
1864 /* Buffer would overflow, signal error. */
1865 return VERR_BUFFER_OVERFLOW;
1866 }
1867
1868 /*
1869 * Append a key=value pair (zero terminated string) to the end of the buffer.
1870 */
1871 pbCurr = pbBuf + cbBufTmp;
1872 memcpy(pbCurr, pcszKey, cbKey);
1873 pbCurr += cbKey;
1874 *pbCurr++ = '=';
1875 if (cbValue == 0)
1876 {
1877 memcpy(pbCurr, pcszValue, cbValueEnc);
1878 pbCurr += cbValueEnc;
1879 }
1880 else
1881 {
1882 *pbCurr++ = '0';
1883 *pbCurr++ = 'x';
1884 for (uint32_t i = 0; i < cbValue; i++)
1885 {
1886 uint8_t b;
1887 b = pcszValue[i];
1888 *pbCurr++ = NUM_2_HEX(b >> 4);
1889 *pbCurr++ = NUM_2_HEX(b & 0xf);
1890 }
1891 }
1892 *pbCurr = '\0';
1893 *pcbBufCurr = cbBufTmp + cbKey + 1 + cbValueEnc + 1;
1894
1895 return VINF_SUCCESS;
1896}
1897
1898
1899/**
1900 * Retrieve the value for a given key from the key=value buffer.
1901 *
1902 * @returns VBOX status.
1903 * @param pbBuf Buffer containing key=value pairs.
1904 * @param cbBuf Length of buffer with key=value pairs.
1905 * @param pszKey Pointer to key for which to retrieve the value.
1906 * @param ppszValue Pointer to value string pointer.
1907 */
1908static int iscsiTextGetKeyValue(const uint8_t *pbBuf, size_t cbBuf, const char *pcszKey, const char **ppcszValue)
1909{
1910 size_t cbKey = strlen(pcszKey);
1911
1912 while (cbBuf != 0)
1913 {
1914 size_t cbKeyValNull = strlen((const char *)pbBuf) + 1;
1915
1916 if (strncmp(pcszKey, (const char *)pbBuf, cbKey) == 0 && pbBuf[cbKey] == '=')
1917 {
1918 *ppcszValue = (const char *)(pbBuf + cbKey + 1);
1919 return VINF_SUCCESS;
1920 }
1921 pbBuf += cbKeyValNull;
1922 cbBuf -= cbKeyValNull;
1923 }
1924 return VERR_INVALID_NAME;
1925}
1926
1927
1928/**
1929 * Convert a long-binary value from a value string to the binary representation.
1930 *
1931 * @returns VBOX status
1932 * @param pszValue Pointer to a string containing the textual value representation.
1933 * @param pbValue Pointer to the value buffer for the binary value.
1934 * @param pcbValue In: length of value buffer, out: actual length of binary value.
1935 */
1936static int iscsiStrToBinary(const char *pcszValue, uint8_t *pbValue, size_t *pcbValue)
1937{
1938 size_t cbValue = *pcbValue;
1939 char c1, c2, c3, c4;
1940 Assert(cbValue >= 1);
1941
1942 if (strlen(pcszValue) < 3)
1943 return VERR_PARSE_ERROR;
1944 if (*pcszValue++ != '0')
1945 return VERR_PARSE_ERROR;
1946 switch (*pcszValue++)
1947 {
1948 case 'x':
1949 case 'X':
1950 if (strlen(pcszValue) & 1)
1951 {
1952 c1 = *pcszValue++;
1953 *pbValue++ = HEX_2_NUM(c1);
1954 cbValue--;
1955 }
1956 while (*pcszValue != '\0')
1957 {
1958 if (cbValue == 0)
1959 return VERR_BUFFER_OVERFLOW;
1960 c1 = *pcszValue++;
1961 if ((c1 < '0' || c1 > '9') && (c1 < 'a' || c1 > 'f') && (c1 < 'A' || c1 > 'F'))
1962 return VERR_PARSE_ERROR;
1963 c2 = *pcszValue++;
1964 if ((c2 < '0' || c2 > '9') && (c2 < 'a' || c2 > 'f') && (c2 < 'A' || c2 > 'F'))
1965 return VERR_PARSE_ERROR;
1966 *pbValue++ = (HEX_2_NUM(c1) << 4) | HEX_2_NUM(c2);
1967 cbValue--;
1968 }
1969 *pcbValue -= cbValue;
1970 break;
1971 case 'b':
1972 case 'B':
1973 if ((strlen(pcszValue) & 3) != 0)
1974 return VERR_PARSE_ERROR;
1975 while (*pcszValue != '\0')
1976 {
1977 uint32_t temp;
1978 if (cbValue == 0)
1979 return VERR_BUFFER_OVERFLOW;
1980 c1 = *pcszValue++;
1981 if ((c1 < 'A' || c1 > 'Z') && (c1 < 'a' || c1 >'z') && (c1 < '0' || c1 > '9') && (c1 != '+') && (c1 != '/'))
1982 return VERR_PARSE_ERROR;
1983 c2 = *pcszValue++;
1984 if ((c2 < 'A' || c2 > 'Z') && (c2 < 'a' || c2 >'z') && (c2 < '0' || c2 > '9') && (c2 != '+') && (c2 != '/'))
1985 return VERR_PARSE_ERROR;
1986 c3 = *pcszValue++;
1987 if ((c3 < 'A' || c3 > 'Z') && (c3 < 'a' || c3 >'z') && (c3 < '0' || c3 > '9') && (c3 != '+') && (c3 != '/') && (c3 != '='))
1988 return VERR_PARSE_ERROR;
1989 c4 = *pcszValue++;
1990 if ( (c3 == '=' && c4 != '=')
1991 || ((c4 < 'A' || c4 > 'Z') && (c4 < 'a' || c4 >'z') && (c4 < '0' || c4 > '9') && (c4 != '+') && (c4 != '/') && (c4 != '=')))
1992 return VERR_PARSE_ERROR;
1993 temp = (B64_2_NUM(c1) << 18) | (B64_2_NUM(c2) << 12);
1994 if (c3 == '=') {
1995 if (*pcszValue != '\0')
1996 return VERR_PARSE_ERROR;
1997 *pbValue++ = temp >> 16;
1998 cbValue--;
1999 } else {
2000 temp |= B64_2_NUM(c3) << 6;
2001 if (c4 == '=') {
2002 if (*pcszValue != '\0')
2003 return VERR_PARSE_ERROR;
2004 if (cbValue < 2)
2005 return VERR_BUFFER_OVERFLOW;
2006 *pbValue++ = temp >> 16;
2007 *pbValue++ = (temp >> 8) & 0xff;
2008 cbValue -= 2;
2009 }
2010 else
2011 {
2012 temp |= B64_2_NUM(c4);
2013 if (cbValue < 3)
2014 return VERR_BUFFER_OVERFLOW;
2015 *pbValue++ = temp >> 16;
2016 *pbValue++ = (temp >> 8) & 0xff;
2017 *pbValue++ = temp & 0xff;
2018 cbValue -= 3;
2019 }
2020 }
2021 }
2022 *pcbValue -= cbValue;
2023 break;
2024 default:
2025 return VERR_PARSE_ERROR;
2026 }
2027 return VINF_SUCCESS;
2028}
2029
2030
2031static bool serial_number_less(uint32_t s1, uint32_t s2)
2032{
2033 return (s1 < s2 && s2 - s1 < 0x80000000) || (s1 > s2 && s1 - s2 > 0x80000000);
2034}
2035
2036
2037#ifdef IMPLEMENT_TARGET_AUTH
2038static void chap_md5_generate_challenge(uint8_t *pbChallenge, size_t *pcbChallenge)
2039{
2040 uint8_t cbChallenge;
2041
2042 cbChallenge = RTrand_U8(CHAP_MD5_CHALLENGE_MIN, CHAP_MD5_CHALLENGE_MAX);
2043 RTrand_bytes(pbChallenge, cbChallenge);
2044 *pcbChallenge = cbChallenge;
2045}
2046#endif
2047
2048
2049static void chap_md5_compute_response(uint8_t *pbResponse, uint8_t id, const uint8_t *pbChallenge, size_t cbChallenge,
2050 const uint8_t *pbSecret, size_t cbSecret)
2051{
2052 RTMD5CONTEXT ctx;
2053 uint8_t bId;
2054
2055 bId = id;
2056 RTMd5Init(&ctx);
2057 RTMd5Update(&ctx, &bId, 1);
2058 RTMd5Update(&ctx, pbSecret, cbSecret);
2059 RTMd5Update(&ctx, pbChallenge, cbChallenge);
2060 RTMd5Final(pbResponse, &ctx);
2061}
2062
2063/**
2064 * Internal. Free all allocated space for representing an image, and optionally
2065 * delete the image from disk.
2066 */
2067static void iscsiFreeImage(PISCSIIMAGE pImage, bool fDelete)
2068{
2069 Assert(pImage);
2070 Assert(!fDelete); /* This MUST be false, the flag isn't supported. */
2071
2072 if (pImage->Mutex != NIL_RTSEMMUTEX)
2073 {
2074 /* Detaching only makes sense when the mutex is there. Otherwise the
2075 * failure happened long before we could attach to the target. */
2076 iscsiDetach(pImage);
2077 RTSemMutexDestroy(pImage->Mutex);
2078 pImage->Mutex = NIL_RTSEMMUTEX;
2079 }
2080 if (pImage->pszTargetName)
2081 {
2082 RTMemFree(pImage->pszTargetName);
2083 pImage->pszTargetName = NULL;
2084 }
2085 if (pImage->pszInitiatorName)
2086 {
2087 RTMemFree(pImage->pszInitiatorName);
2088 pImage->pszInitiatorName = NULL;
2089 }
2090 if (pImage->pszInitiatorUsername)
2091 {
2092 RTMemFree(pImage->pszInitiatorUsername);
2093 pImage->pszInitiatorUsername = NULL;
2094 }
2095 if (pImage->pbInitiatorSecret)
2096 {
2097 RTMemFree(pImage->pbInitiatorSecret);
2098 pImage->pbInitiatorSecret = NULL;
2099 }
2100 if (pImage->pszTargetUsername)
2101 {
2102 RTMemFree(pImage->pszTargetUsername);
2103 pImage->pszTargetUsername = NULL;
2104 }
2105 if (pImage->pbTargetSecret)
2106 {
2107 RTMemFree(pImage->pbTargetSecret);
2108 pImage->pbTargetSecret = NULL;
2109 }
2110 if (pImage->pvRecvPDUBuf)
2111 {
2112 RTMemFree(pImage->pvRecvPDUBuf);
2113 pImage->pvRecvPDUBuf = NULL;
2114 }
2115}
2116
2117/**
2118 * Internal: Open an image, constructing all necessary data structures.
2119 */
2120static int iscsiOpenImage(PISCSIIMAGE pImage, unsigned uOpenFlags)
2121{
2122 int rc;
2123 char *pszLUN = NULL, *pszLUNInitial = NULL;
2124 bool fLunEncoded = false;
2125 uint32_t uTimeoutDef = 0;
2126 uint64_t uHostIPTmp = 0;
2127 bool fHostIPDef = 0;
2128 rc = RTStrToUInt32Full(s_iscsiConfigDefaultTimeout, 0, &uTimeoutDef);
2129 AssertRC(rc);
2130 rc = RTStrToUInt64Full(s_iscsiConfigDefaultHostIPStack, 0, &uHostIPTmp);
2131 AssertRC(rc);
2132 fHostIPDef = !!uHostIPTmp;
2133
2134 pImage->uOpenFlags = uOpenFlags;
2135
2136 /* Get error signalling interface. */
2137 pImage->pInterfaceError = VDInterfaceGet(pImage->pVDIfsDisk, VDINTERFACETYPE_ERROR);
2138 if (pImage->pInterfaceError)
2139 pImage->pInterfaceErrorCallbacks = VDGetInterfaceError(pImage->pInterfaceError);
2140
2141 /* Get TCP network stack interface. */
2142 pImage->pInterfaceNet = VDInterfaceGet(pImage->pVDIfsDisk, VDINTERFACETYPE_TCPNET);
2143 if (pImage->pInterfaceNet)
2144 pImage->pInterfaceNetCallbacks = VDGetInterfaceTcpNet(pImage->pInterfaceNet);
2145 else
2146 {
2147 rc = iscsiError(pImage, VERR_VD_ISCSI_UNKNOWN_INTERFACE,
2148 RT_SRC_POS, N_("iSCSI: TCP network stack interface missing"));
2149 goto out;
2150 }
2151
2152 /* Get configuration interface. */
2153 pImage->pInterfaceConfig = VDInterfaceGet(pImage->pVDIfsImage, VDINTERFACETYPE_CONFIG);
2154 if (pImage->pInterfaceConfig)
2155 pImage->pInterfaceConfigCallbacks = VDGetInterfaceConfig(pImage->pInterfaceConfig);
2156 else
2157 {
2158 rc = iscsiError(pImage, VERR_VD_ISCSI_UNKNOWN_INTERFACE,
2159 RT_SRC_POS, N_("iSCSI: configuration interface missing"));
2160 goto out;
2161 }
2162
2163 pImage->ISID = 0x800000000000ULL | 0x001234560000ULL | (0x00000000cba0ULL + ASMAtomicIncU32(&s_u32iscsiID));
2164 pImage->cISCSIRetries = 10;
2165 pImage->state = ISCSISTATE_FREE;
2166 pImage->pvRecvPDUBuf = RTMemAlloc(ISCSI_RECV_PDU_BUFFER_SIZE);
2167 pImage->cbRecvPDUBuf = ISCSI_RECV_PDU_BUFFER_SIZE;
2168 if (pImage->pvRecvPDUBuf == NULL)
2169 {
2170 rc = VERR_NO_MEMORY;
2171 goto out;
2172 }
2173 pImage->Mutex = NIL_RTSEMMUTEX;
2174 rc = RTSemMutexCreate(&pImage->Mutex);
2175 if (RT_FAILURE(rc))
2176 goto out;
2177
2178 /* Validate configuration, detect unknown keys. */
2179 if (!VDCFGAreKeysValid(pImage->pInterfaceConfigCallbacks,
2180 pImage->pInterfaceConfig->pvUser,
2181 "TargetName\0InitiatorName\0LUN\0TargetAddress\0InitiatorUsername\0InitiatorSecret\0TargetUsername\0TargetSecret\0Timeout\0HostIPStack\0"))
2182 {
2183 rc = iscsiError(pImage, VERR_VD_ISCSI_UNKNOWN_CFG_VALUES, RT_SRC_POS, N_("iSCSI: configuration error: unknown configuration keys present"));
2184 goto out;
2185 }
2186
2187 /* Query the iSCSI upper level configuration. */
2188 rc = VDCFGQueryStringAlloc(pImage->pInterfaceConfigCallbacks,
2189 pImage->pInterfaceConfig->pvUser,
2190 "TargetName", &pImage->pszTargetName);
2191 if (RT_FAILURE(rc))
2192 {
2193 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read TargetName as string"));
2194 goto out;
2195 }
2196 rc = VDCFGQueryStringAllocDef(pImage->pInterfaceConfigCallbacks,
2197 pImage->pInterfaceConfig->pvUser,
2198 "InitiatorName", &pImage->pszInitiatorName,
2199 s_iscsiConfigDefaultInitiatorName);
2200 if (RT_FAILURE(rc))
2201 {
2202 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read InitiatorName as string"));
2203 goto out;
2204 }
2205 rc = VDCFGQueryStringAllocDef(pImage->pInterfaceConfigCallbacks,
2206 pImage->pInterfaceConfig->pvUser,
2207 "LUN", &pszLUN, s_iscsiConfigDefaultLUN);
2208 if (RT_FAILURE(rc))
2209 {
2210 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read LUN as string"));
2211 goto out;
2212 }
2213 pszLUNInitial = pszLUN;
2214 if (!strncmp(pszLUN, "enc", 3))
2215 {
2216 fLunEncoded = true;
2217 pszLUN += 3;
2218 }
2219 rc = RTStrToUInt64Full(pszLUN, 0, &pImage->LUN);
2220 if (RT_FAILURE(rc))
2221 {
2222 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to convert LUN to integer"));
2223 goto out;
2224 }
2225 if (!fLunEncoded)
2226 {
2227 if (pImage->LUN <= 255)
2228 {
2229 pImage->LUN = pImage->LUN << 48; /* uses peripheral device addressing method */
2230 }
2231 else if (pImage->LUN <= 16383)
2232 {
2233 pImage->LUN = (pImage->LUN << 48) | RT_BIT_64(62); /* uses flat space addressing method */
2234 }
2235 else
2236 {
2237 rc = VERR_OUT_OF_RANGE;
2238 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: LUN number out of range (0-16383)"));
2239 goto out;
2240 }
2241 }
2242 rc = VDCFGQueryStringAlloc(pImage->pInterfaceConfigCallbacks,
2243 pImage->pInterfaceConfig->pvUser,
2244 "TargetAddress", &pImage->pszTargetAddress);
2245 if (RT_FAILURE(rc))
2246 {
2247 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read TargetAddress as string"));
2248 goto out;
2249 }
2250 pImage->pszInitiatorUsername = NULL;
2251 rc = VDCFGQueryStringAlloc(pImage->pInterfaceConfigCallbacks,
2252 pImage->pInterfaceConfig->pvUser,
2253 "InitiatorUsername",
2254 &pImage->pszInitiatorUsername);
2255 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
2256 rc = VINF_SUCCESS;
2257 if (RT_FAILURE(rc))
2258 {
2259 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read InitiatorUsername as string"));
2260 goto out;
2261 }
2262 pImage->pbInitiatorSecret = NULL;
2263 pImage->cbInitiatorSecret = 0;
2264 rc = VDCFGQueryBytesAlloc(pImage->pInterfaceConfigCallbacks,
2265 pImage->pInterfaceConfig->pvUser,
2266 "InitiatorSecret",
2267 (void **)&pImage->pbInitiatorSecret,
2268 &pImage->cbInitiatorSecret);
2269 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
2270 rc = VINF_SUCCESS;
2271 if (RT_FAILURE(rc))
2272 {
2273 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read InitiatorSecret as byte string"));
2274 goto out;
2275 }
2276 pImage->pszTargetUsername = NULL;
2277 rc = VDCFGQueryStringAlloc(pImage->pInterfaceConfigCallbacks,
2278 pImage->pInterfaceConfig->pvUser,
2279 "TargetUsername",
2280 &pImage->pszTargetUsername);
2281 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
2282 rc = VINF_SUCCESS;
2283 if (RT_FAILURE(rc))
2284 {
2285 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read TargetUsername as string"));
2286 goto out;
2287 }
2288 pImage->pbTargetSecret = NULL;
2289 pImage->cbTargetSecret = 0;
2290 rc = VDCFGQueryBytesAlloc(pImage->pInterfaceConfigCallbacks,
2291 pImage->pInterfaceConfig->pvUser,
2292 "TargetSecret", (void **)&pImage->pbTargetSecret,
2293 &pImage->cbTargetSecret);
2294 if (rc == VERR_CFGM_VALUE_NOT_FOUND || rc == VERR_CFGM_NO_PARENT)
2295 rc = VINF_SUCCESS;
2296 if (RT_FAILURE(rc))
2297 {
2298 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read TargetSecret as byte string"));
2299 goto out;
2300 }
2301
2302 pImage->pszHostname = NULL;
2303 pImage->uPort = 0;
2304 pImage->Socket = NIL_RTSOCKET;
2305 /* Query the iSCSI lower level configuration. */
2306 rc = VDCFGQueryU32Def(pImage->pInterfaceConfigCallbacks,
2307 pImage->pInterfaceConfig->pvUser,
2308 "Timeout", &pImage->uReadTimeout,
2309 uTimeoutDef);
2310 if (RT_FAILURE(rc))
2311 {
2312 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read Timeout as U32"));
2313 goto out;
2314 }
2315 rc = VDCFGQueryBoolDef(pImage->pInterfaceConfigCallbacks,
2316 pImage->pInterfaceConfig->pvUser,
2317 "HostIPStack", &pImage->fHostIP,
2318 fHostIPDef);
2319 if (RT_FAILURE(rc))
2320 {
2321 rc = iscsiError(pImage, rc, RT_SRC_POS, N_("iSCSI: configuration error: failed to read HostIPStack as boolean"));
2322 goto out;
2323 }
2324
2325 /* Don't actually establish iSCSI transport connection if this is just an
2326 * open to query the image information and the host IP stack isn't used.
2327 * Even trying is rather useless, as in this context the InTnet IP stack
2328 * isn't present. Returning dummies is the best possible result anyway. */
2329 if ((uOpenFlags & VD_OPEN_FLAGS_INFO) && !pImage->fHostIP)
2330 {
2331 LogFunc(("Not opening the transport connection as IntNet IP stack is not available. Will return dummies\n"));
2332 goto out;
2333 }
2334
2335 /*
2336 * Establish the iSCSI transport connection.
2337 */
2338 rc = iscsiTransportOpen(pImage);
2339 if (RT_SUCCESS(rc))
2340 rc = iscsiAttach(pImage);
2341
2342 if (RT_FAILURE(rc))
2343 {
2344 LogRel(("iSCSI: could not open target %s, rc=%Rrc\n", pImage->pszTargetName, rc));
2345 goto out;
2346 }
2347 LogFlowFunc(("target '%s' opened successfully\n", pImage->pszTargetName));
2348
2349 SCSIREQ sr;
2350 uint8_t sense[32];
2351 uint8_t data8[8];
2352
2353 /*
2354 * Inquire available LUNs - purely dummy request.
2355 */
2356 uint8_t cdb_rlun[12];
2357 uint8_t rlundata[16];
2358 cdb_rlun[0] = SCSI_REPORT_LUNS;
2359 cdb_rlun[1] = 0; /* reserved */
2360 cdb_rlun[2] = 0; /* reserved */
2361 cdb_rlun[3] = 0; /* reserved */
2362 cdb_rlun[4] = 0; /* reserved */
2363 cdb_rlun[5] = 0; /* reserved */
2364 cdb_rlun[6] = sizeof(rlundata) >> 24;
2365 cdb_rlun[7] = (sizeof(rlundata) >> 16) & 0xff;
2366 cdb_rlun[8] = (sizeof(rlundata) >> 8) & 0xff;
2367 cdb_rlun[9] = sizeof(rlundata) & 0xff;
2368 cdb_rlun[10] = 0; /* reserved */
2369 cdb_rlun[11] = 0; /* control */
2370
2371 sr.enmXfer = SCSIXFER_FROM_TARGET;
2372 sr.cbCmd = sizeof(cdb_rlun);
2373 sr.pvCmd = cdb_rlun;
2374 sr.cbI2TData = 0;
2375 sr.pcvI2TData = NULL;
2376 sr.cbT2IData = sizeof(rlundata);
2377 sr.pvT2IData = rlundata;
2378 sr.cbSense = sizeof(sense);
2379 sr.pvSense = sense;
2380
2381 rc = iscsiCommand(pImage, &sr);
2382 if (RT_FAILURE(rc))
2383 {
2384 LogRel(("iSCSI: Could not get LUN info for target %s, rc=%Rrc\n", pImage->pszTargetName, rc));
2385 return rc;
2386 }
2387
2388 /*
2389 * Inquire device characteristics - no tapes, scanners etc., please.
2390 */
2391 uint8_t cdb_inq[6];
2392 cdb_inq[0] = SCSI_INQUIRY;
2393 cdb_inq[1] = 0; /* reserved */
2394 cdb_inq[2] = 0; /* reserved */
2395 cdb_inq[3] = 0; /* reserved */
2396 cdb_inq[4] = sizeof(data8);
2397 cdb_inq[5] = 0; /* control */
2398
2399 sr.enmXfer = SCSIXFER_FROM_TARGET;
2400 sr.cbCmd = sizeof(cdb_inq);
2401 sr.pvCmd = cdb_inq;
2402 sr.cbI2TData = 0;
2403 sr.pcvI2TData = NULL;
2404 sr.cbT2IData = sizeof(data8);
2405 sr.pvT2IData = data8;
2406 sr.cbSense = sizeof(sense);
2407 sr.pvSense = sense;
2408
2409 rc = iscsiCommand(pImage, &sr);
2410 if (RT_SUCCESS(rc))
2411 {
2412 if ((data8[0] & SCSI_DEVTYPE_MASK) != SCSI_DEVTYPE_DISK)
2413 {
2414 rc = iscsiError(pImage, VERR_VD_ISCSI_INVALID_TYPE,
2415 RT_SRC_POS, N_("iSCSI: target address %s, target name %s, SCSI LUN %lld reports device type=%u"),
2416 pImage->pszTargetAddress, pImage->pszTargetName,
2417 pImage->LUN, data8[0]);
2418 LogRel(("iSCSI: Unsupported SCSI peripheral device type %d for target %s\n", data8[0] & SCSI_DEVTYPE_MASK, pImage->pszTargetName));
2419 goto out;
2420 }
2421 }
2422 else
2423 {
2424 LogRel(("iSCSI: Could not get INQUIRY info for target %s, rc=%Rrc\n", pImage->pszTargetName, rc));
2425 goto out;
2426 }
2427
2428 /*
2429 * Query write disable bit in the device specific parameter entry in the
2430 * mode parameter header. Refuse read/write opening of read only disks.
2431 */
2432
2433 uint8_t cdb_ms[6];
2434 uint8_t data4[4];
2435 cdb_ms[0] = SCSI_MODE_SENSE_6;
2436 cdb_ms[1] = 0; /* dbd=0/reserved */
2437 cdb_ms[2] = 0x3f; /* pc=0/page code=0x3f, ask for all pages */
2438 cdb_ms[3] = 0; /* subpage code=0, return everything in page_0 format */
2439 cdb_ms[4] = sizeof(data4); /* allocation length=4 */
2440 cdb_ms[5] = 0; /* control */
2441
2442 sr.enmXfer = SCSIXFER_FROM_TARGET;
2443 sr.cbCmd = sizeof(cdb_ms);
2444 sr.pvCmd = cdb_ms;
2445 sr.cbI2TData = 0;
2446 sr.pcvI2TData = NULL;
2447 sr.cbT2IData = sizeof(data4);
2448 sr.pvT2IData = data4;
2449 sr.cbSense = sizeof(sense);
2450 sr.pvSense = sense;
2451
2452 rc = iscsiCommand(pImage, &sr);
2453 if (RT_SUCCESS(rc))
2454 {
2455 if (!(uOpenFlags & VD_OPEN_FLAGS_READONLY) && data4[2] & 0x80)
2456 {
2457 rc = VERR_VD_IMAGE_READ_ONLY;
2458 goto out;
2459 }
2460 }
2461 else
2462 {
2463 LogRel(("iSCSI: Could not get MODE SENSE info for target %s, rc=%Rrc\n", pImage->pszTargetName, rc));
2464 goto out;
2465 }
2466
2467 /*
2468 * Determine sector size and capacity of the volume immediately.
2469 */
2470 uint8_t cdb_cap[10];
2471
2472 cdb_cap[0] = SCSI_READ_CAPACITY;
2473 cdb_cap[1] = 0; /* reserved */
2474 cdb_cap[2] = 0; /* reserved */
2475 cdb_cap[3] = 0; /* reserved */
2476 cdb_cap[4] = 0; /* reserved */
2477 cdb_cap[5] = 0; /* reserved */
2478 cdb_cap[6] = 0; /* reserved */
2479 cdb_cap[7] = 0; /* reserved */
2480 cdb_cap[8] = 0; /* reserved */
2481 cdb_cap[9] = 0; /* control */
2482
2483 sr.enmXfer = SCSIXFER_FROM_TARGET;
2484 sr.cbCmd = sizeof(cdb_cap);
2485 sr.pvCmd = cdb_cap;
2486 sr.cbI2TData = 0;
2487 sr.pcvI2TData = NULL;
2488 sr.cbT2IData = sizeof(data8);
2489 sr.pvT2IData = data8;
2490 sr.cbSense = sizeof(sense);
2491 sr.pvSense = sense;
2492
2493 rc = iscsiCommand(pImage, &sr);
2494 if (RT_SUCCESS(rc))
2495 {
2496 pImage->cVolume = (data8[0] << 24) | (data8[1] << 16) | (data8[2] << 8) | data8[3];
2497 pImage->cVolume++;
2498 pImage->cbSector = (data8[4] << 24) | (data8[5] << 16) | (data8[6] << 8) | data8[7];
2499 pImage->cbSize = (uint64_t)(pImage->cVolume) * pImage->cbSector;
2500 if (pImage->cVolume == 0 || pImage->cbSector == 0)
2501 {
2502 rc = iscsiError(pImage, VERR_VD_ISCSI_INVALID_TYPE,
2503 RT_SRC_POS, N_("iSCSI: target address %s, target name %s, SCSI LUN %lld reports media sector count=%lu sector size=%lu"),
2504 pImage->pszTargetAddress, pImage->pszTargetName,
2505 pImage->LUN, pImage->cVolume, pImage->cbSector);
2506 }
2507 }
2508 else
2509 {
2510 LogRel(("iSCSI: Could not determine capacity of target %s, rc=%Rrc\n", pImage->pszTargetName, rc));
2511 goto out;
2512 }
2513
2514 /*
2515 * Check the read and write cache bits.
2516 * Try to enable the cache if it is disabled.
2517 *
2518 * We already checked that this is a block access device. No need
2519 * to do it again.
2520 */
2521 uint8_t aCachingModePage[32];
2522 uint8_t aCDBModeSense6[6];
2523
2524 memset(aCachingModePage, '\0', sizeof(aCachingModePage));
2525 aCDBModeSense6[0] = SCSI_MODE_SENSE_6;
2526 aCDBModeSense6[1] = 0;
2527 aCDBModeSense6[2] = (0x00 << 6) | (0x08 & 0x3f); /* Current values and caching mode page */
2528 aCDBModeSense6[3] = 0; /* Sub page code. */
2529 aCDBModeSense6[4] = sizeof(aCachingModePage) & 0xff;
2530 aCDBModeSense6[5] = 0;
2531 sr.enmXfer = SCSIXFER_FROM_TARGET;
2532 sr.cbCmd = sizeof(aCDBModeSense6);
2533 sr.pvCmd = aCDBModeSense6;
2534 sr.cbI2TData = 0;
2535 sr.pcvI2TData = NULL;
2536 sr.cbT2IData = sizeof(aCachingModePage);
2537 sr.pvT2IData = aCachingModePage;
2538 sr.cbSense = sizeof(sense);
2539 sr.pvSense = sense;
2540 rc = iscsiCommand(pImage, &sr);
2541 if ( RT_SUCCESS(rc)
2542 && (sr.status == SCSI_STATUS_OK)
2543 && (aCachingModePage[0] >= 15)
2544 && (aCachingModePage[4 + aCachingModePage[3]] & 0x3f) == 0x08
2545 && (aCachingModePage[4 + aCachingModePage[3] + 1] > 3))
2546 {
2547 uint32_t Offset = 4 + aCachingModePage[3];
2548 /*
2549 * Check if the read and/or the write cache is disabled.
2550 * The write cache is disabled if bit 2 (WCE) is zero and
2551 * the read cache is disabled if bit 0 (RCD) is set.
2552 */
2553 if (!ASMBitTest(&aCachingModePage[Offset + 2], 2) || ASMBitTest(&aCachingModePage[Offset + 2], 0))
2554 {
2555 /*
2556 * Write Cache Enable (WCE) bit is zero or the Read Cache Disable (RCD) is one
2557 * So one of the caches is disabled. Enable both caches.
2558 * The rest is unchanged.
2559 */
2560 ASMBitSet(&aCachingModePage[Offset + 2], 2);
2561 ASMBitClear(&aCachingModePage[Offset + 2], 0);
2562
2563 uint8_t aCDBCaching[6];
2564 aCDBCaching[0] = SCSI_MODE_SELECT_6;
2565 aCDBCaching[1] = 0; /* Don't write the page into NV RAM. */
2566 aCDBCaching[2] = 0;
2567 aCDBCaching[3] = 0;
2568 aCDBCaching[4] = sizeof(aCachingModePage) & 0xff;
2569 aCDBCaching[5] = 0;
2570 sr.enmXfer = SCSIXFER_TO_TARGET;
2571 sr.cbCmd = sizeof(aCDBCaching);
2572 sr.pvCmd = aCDBCaching;
2573 sr.cbI2TData = sizeof(aCachingModePage);
2574 sr.pcvI2TData = aCachingModePage;
2575 sr.cbT2IData = 0;
2576 sr.pvT2IData = NULL;
2577 sr.cbSense = sizeof(sense);
2578 sr.pvSense = sense;
2579 sr.status = 0;
2580 rc = iscsiCommand(pImage, &sr);
2581 if ( RT_SUCCESS(rc)
2582 && (sr.status == SCSI_STATUS_OK))
2583 {
2584 LogRel(("iSCSI: Enabled read and write cache of target %s\n", pImage->pszTargetName));
2585 }
2586 else
2587 {
2588 /* Log failures but continue. */
2589 LogRel(("iSCSI: Could not enable read and write cache of target %s, rc=%Rrc status=%#x\n",
2590 pImage->pszTargetName, rc, sr.status));
2591 LogRel(("iSCSI: Sense:\n%.*Rhxd\n", sr.cbSense, sense));
2592 rc = VINF_SUCCESS;
2593 }
2594 }
2595 }
2596 else
2597 {
2598 /* Log errors but continue. */
2599 LogRel(("iSCSI: Could not check write cache of target %s, rc=%Rrc, got mode page %#x\n", pImage->pszTargetName, rc,aCachingModePage[0] & 0x3f));
2600 LogRel(("iSCSI: Sense:\n%.*Rhxd\n", sr.cbSense, sense));
2601 rc = VINF_SUCCESS;
2602 }
2603
2604
2605out:
2606 if (RT_FAILURE(rc))
2607 iscsiFreeImage(pImage, false);
2608 return rc;
2609}
2610
2611
2612/** @copydoc VBOXHDDBACKEND::pfnCheckIfValid */
2613static int iscsiCheckIfValid(const char *pszFilename)
2614{
2615 LogFlowFunc(("pszFilename=\"%s\"\n", pszFilename));
2616
2617 /* iSCSI images can't be checked for validity this way, as the filename
2618 * just can't supply enough configuration information. */
2619 int rc = VERR_VD_ISCSI_INVALID_HEADER;
2620
2621 LogFlowFunc(("returns %Rrc\n", rc));
2622 return rc;
2623}
2624
2625
2626/** @copydoc VBOXHDDBACKEND::pfnOpen */
2627static int iscsiOpen(const char *pszFilename, unsigned uOpenFlags,
2628 PVDINTERFACE pVDIfsDisk, PVDINTERFACE pVDIfsImage,
2629 void **ppBackendData)
2630{
2631 LogFlowFunc(("pszFilename=\"%s\" uOpenFlags=%#x pVDIfsDisk=%#p pVDIfsImage=%#p ppBackendData=%#p\n", pszFilename, uOpenFlags, pVDIfsDisk, pVDIfsImage, ppBackendData));
2632 int rc;
2633 PISCSIIMAGE pImage;
2634
2635 /* Check open flags. All valid flags are supported. */
2636 if (uOpenFlags & ~VD_OPEN_FLAGS_MASK)
2637 {
2638 rc = VERR_INVALID_PARAMETER;
2639 goto out;
2640 }
2641
2642 /* Check remaining arguments. */
2643 if ( !VALID_PTR(pszFilename)
2644 || !*pszFilename
2645 || strchr(pszFilename, '"'))
2646 {
2647 rc = VERR_INVALID_PARAMETER;
2648 goto out;
2649 }
2650
2651 pImage = (PISCSIIMAGE)RTMemAllocZ(sizeof(ISCSIIMAGE));
2652 if (!pImage)
2653 {
2654 rc = VERR_NO_MEMORY;
2655 goto out;
2656 }
2657
2658 pImage->pszFilename = pszFilename;
2659 pImage->pszInitiatorName = NULL;
2660 pImage->pszTargetName = NULL;
2661 pImage->pszTargetAddress = NULL;
2662 pImage->pszInitiatorUsername = NULL;
2663 pImage->pbInitiatorSecret = NULL;
2664 pImage->pszTargetUsername = NULL;
2665 pImage->pbTargetSecret = NULL;
2666 pImage->paCurrReq = NULL;
2667 pImage->pvRecvPDUBuf = NULL;
2668 pImage->pszHostname = NULL;
2669 pImage->pVDIfsDisk = pVDIfsDisk;
2670 pImage->pVDIfsImage = pVDIfsImage;
2671
2672 rc = iscsiOpenImage(pImage, uOpenFlags);
2673 if (RT_SUCCESS(rc))
2674 *ppBackendData = pImage;
2675
2676out:
2677 if (RT_SUCCESS(rc))
2678 {
2679 LogFlowFunc(("target %s cVolume %d, cbSector %d\n", pImage->pszTargetName, pImage->cVolume, pImage->cbSector));
2680 LogRel(("iSCSI: target address %s, target name %s, SCSI LUN %lld\n", pImage->pszTargetAddress, pImage->pszTargetName, pImage->LUN));
2681 }
2682 LogFlowFunc(("returns %Rrc (pBackendData=%#p)\n", rc, *ppBackendData));
2683 return rc;
2684}
2685
2686
2687/** @copydoc VBOXHDDBACKEND::pfnCreate */
2688static int iscsiCreate(const char *pszFilename, uint64_t cbSize,
2689 unsigned uImageFlags, const char *pszComment,
2690 PCPDMMEDIAGEOMETRY pPCHSGeometry,
2691 PCPDMMEDIAGEOMETRY pLCHSGeometry, PCRTUUID pUuid,
2692 unsigned uOpenFlags, unsigned uPercentStart,
2693 unsigned uPercentSpan, PVDINTERFACE pVDIfsDisk,
2694 PVDINTERFACE pVDIfsImage, PVDINTERFACE pVDIfsOperation,
2695 void **ppBackendData)
2696{
2697 LogFlowFunc(("pszFilename=\"%s\" cbSize=%llu uImageFlags=%#x pszComment=\"%s\" pPCHSGeometry=%#p pLCHSGeometry=%#p Uuid=%RTuuid uOpenFlags=%#x uPercentStart=%u uPercentSpan=%u pVDIfsDisk=%#p pVDIfsImage=%#p pVDIfsOperation=%#p ppBackendData=%#p", pszFilename, cbSize, uImageFlags, pszComment, pPCHSGeometry, pLCHSGeometry, pUuid, uOpenFlags, uPercentStart, uPercentSpan, pVDIfsDisk, pVDIfsImage, pVDIfsOperation, ppBackendData));
2698 int rc = VERR_NOT_SUPPORTED;
2699
2700 LogFlowFunc(("returns %Rrc (pBackendData=%#p)\n", rc, *ppBackendData));
2701 return rc;
2702}
2703
2704
2705/** @copydoc VBOXHDDBACKEND::pfnRename */
2706static int iscsiRename(void *pBackendData, const char *pszFilename)
2707{
2708 LogFlowFunc(("pBackendData=%#p pszFilename=%#p\n", pBackendData, pszFilename));
2709 int rc = VERR_NOT_SUPPORTED;
2710
2711 LogFlowFunc(("returns %Rrc\n", rc));
2712 return rc;
2713}
2714
2715
2716/** @copydoc VBOXHDDBACKEND::pfnClose */
2717static int iscsiClose(void *pBackendData, bool fDelete)
2718{
2719 LogFlowFunc(("pBackendData=%#p fDelete=%d\n", pBackendData, fDelete));
2720 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2721 int rc = VINF_SUCCESS;
2722
2723 Assert(!fDelete); /* This flag is unsupported. */
2724
2725 /* Freeing a never allocated image (e.g. because the open failed) is
2726 * not signalled as an error. After all nothing bad happens. */
2727 if (pImage)
2728 iscsiFreeImage(pImage, fDelete);
2729
2730 LogFlowFunc(("returns %Rrc\n", rc));
2731 return rc;
2732}
2733
2734
2735/** @copydoc VBOXHDDBACKEND::pfnRead */
2736static int iscsiRead(void *pBackendData, uint64_t uOffset, void *pvBuf,
2737 size_t cbToRead, size_t *pcbActuallyRead)
2738{
2739 /** @todo reinstate logging of the target everywhere - dropped temporarily */
2740 LogFlowFunc(("pBackendData=%#p uOffset=%llu pvBuf=%#p cbToRead=%zu pcbActuallyRead=%#p\n", pBackendData, uOffset, pvBuf, cbToRead, pcbActuallyRead));
2741 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2742 uint64_t lba;
2743 uint16_t tls;
2744 int rc;
2745
2746 Assert(pImage);
2747 Assert(uOffset % 512 == 0);
2748 Assert(cbToRead % 512 == 0);
2749
2750 Assert(pImage->cbSector);
2751 AssertPtr(pvBuf);
2752
2753 if ( uOffset + cbToRead > pImage->cbSize
2754 || cbToRead == 0)
2755 {
2756 rc = VERR_INVALID_PARAMETER;
2757 goto out;
2758 }
2759
2760 lba = uOffset / pImage->cbSector;
2761 tls = (uint16_t)(cbToRead / pImage->cbSector);
2762 SCSIREQ sr;
2763 uint8_t cdb[10];
2764 uint8_t sense[32];
2765
2766 cdb[0] = SCSI_READ_10;
2767 cdb[1] = 0; /* reserved */
2768 cdb[2] = (lba >> 24) & 0xff;
2769 cdb[3] = (lba >> 16) & 0xff;
2770 cdb[4] = (lba >> 8) & 0xff;
2771 cdb[5] = lba & 0xff;
2772 cdb[6] = 0; /* reserved */
2773 cdb[7] = (tls >> 8) & 0xff;
2774 cdb[8] = tls & 0xff;
2775 cdb[9] = 0; /* control */
2776
2777 sr.enmXfer = SCSIXFER_FROM_TARGET;
2778 sr.cbCmd = sizeof(cdb);
2779 sr.pvCmd = cdb;
2780 sr.cbI2TData = 0;
2781 sr.pcvI2TData = NULL;
2782 sr.cbT2IData = cbToRead;
2783 sr.pvT2IData = pvBuf;
2784 sr.cbSense = sizeof(sense);
2785 sr.pvSense = sense;
2786
2787 rc = iscsiCommand(pImage, &sr);
2788 if (RT_FAILURE(rc))
2789 AssertMsgFailed(("iscsiCommand(%s, %#llx) -> %Rrc\n", pImage->pszTargetName, uOffset, rc));
2790
2791out:
2792 LogFlowFunc(("returns %Rrc\n", rc));
2793 return rc;
2794}
2795
2796
2797/** @copydoc VBOXHDDBACKEND::pfnWrite */
2798static int iscsiWrite(void *pBackendData, uint64_t uOffset, const void *pvBuf,
2799 size_t cbToWrite, size_t *pcbWriteProcess,
2800 size_t *pcbPreRead, size_t *pcbPostRead, unsigned fWrite)
2801{
2802 LogFlowFunc(("pBackendData=%#p uOffset=%llu pvBuf=%#p cbToWrite=%zu pcbWriteProcess=%#p pcbPreRead=%#p pcbPostRead=%#p\n", pBackendData, uOffset, pvBuf, cbToWrite, pcbWriteProcess, pcbPreRead, pcbPostRead));
2803 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2804 uint64_t lba;
2805 uint16_t tls;
2806 int rc;
2807
2808 Assert(pImage);
2809 Assert(uOffset % 512 == 0);
2810 Assert(cbToWrite % 512 == 0);
2811
2812 Assert(pImage->cbSector);
2813 Assert(pvBuf);
2814
2815 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2816 {
2817 rc = VERR_VD_IMAGE_READ_ONLY;
2818 goto out;
2819 }
2820
2821 *pcbPreRead = 0;
2822 *pcbPostRead = 0;
2823
2824 lba = uOffset / pImage->cbSector;
2825 tls = (uint16_t)(cbToWrite / pImage->cbSector);
2826 SCSIREQ sr;
2827 uint8_t cdb[10];
2828 uint8_t sense[32];
2829
2830 cdb[0] = SCSI_WRITE_10;
2831 cdb[1] = 0; /* reserved */
2832 cdb[2] = (lba >> 24) & 0xff;
2833 cdb[3] = (lba >> 16) & 0xff;
2834 cdb[4] = (lba >> 8) & 0xff;
2835 cdb[5] = lba & 0xff;
2836 cdb[6] = 0; /* reserved */
2837 cdb[7] = (tls >> 8) & 0xff;
2838 cdb[8] = tls & 0xff;
2839 cdb[9] = 0; /* control */
2840
2841 sr.enmXfer = SCSIXFER_TO_TARGET;
2842 sr.cbCmd = sizeof(cdb);
2843 sr.pvCmd = cdb;
2844 sr.cbI2TData = cbToWrite;
2845 sr.pcvI2TData = pvBuf;
2846 sr.cbT2IData = 0;
2847 sr.pvT2IData = NULL;
2848 sr.cbSense = sizeof(sense);
2849 sr.pvSense = sense;
2850
2851 rc = iscsiCommand(pImage, &sr);
2852 if (RT_FAILURE(rc))
2853 {
2854 AssertMsgFailed(("iscsiCommand(%s, %#llx) -> %Rrc\n", pImage->pszTargetName, uOffset, rc));
2855 *pcbWriteProcess = 0;
2856 }
2857 else
2858 *pcbWriteProcess = cbToWrite;
2859
2860out:
2861 LogFlowFunc(("returns %Rrc\n", rc));
2862 return rc;
2863}
2864
2865
2866/** @copydoc VBOXHDDBACKEND::pfnFlush */
2867static int iscsiFlush(void *pBackendData)
2868{
2869 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
2870 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2871 int rc;
2872
2873 Assert(pImage);
2874
2875 SCSIREQ sr;
2876 uint8_t cdb[10];
2877 uint8_t sense[32];
2878
2879 cdb[0] = SCSI_SYNCHRONIZE_CACHE;
2880 cdb[1] = 0; /* reserved */
2881 cdb[2] = 0; /* LBA 0 */
2882 cdb[3] = 0; /* LBA 0 */
2883 cdb[4] = 0; /* LBA 0 */
2884 cdb[5] = 0; /* LBA 0 */
2885 cdb[6] = 0; /* reserved */
2886 cdb[7] = 0; /* transfer everything to disk */
2887 cdb[8] = 0; /* transfer everything to disk */
2888 cdb[9] = 0; /* control */
2889
2890 sr.enmXfer = SCSIXFER_TO_TARGET;
2891 sr.cbCmd = sizeof(cdb);
2892 sr.pvCmd = cdb;
2893 sr.cbI2TData = 0;
2894 sr.pcvI2TData = NULL;
2895 sr.cbT2IData = 0;
2896 sr.pvT2IData = NULL;
2897 sr.cbSense = sizeof(sense);
2898 sr.pvSense = sense;
2899
2900 rc = iscsiCommand(pImage, &sr);
2901 if (RT_FAILURE(rc))
2902 AssertMsgFailed(("iscsiCommand(%s) -> %Rrc\n", pImage->pszTargetName, rc));
2903 LogFlowFunc(("returns %Rrc\n", rc));
2904 return rc;
2905}
2906
2907
2908/** @copydoc VBOXHDDBACKEND::pfnGetVersion */
2909static unsigned iscsiGetVersion(void *pBackendData)
2910{
2911 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
2912 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2913
2914 Assert(pImage);
2915 NOREF(pImage);
2916
2917 return 0;
2918}
2919
2920
2921/** @copydoc VBOXHDDBACKEND::pfnGetSize */
2922static uint64_t iscsiGetSize(void *pBackendData)
2923{
2924 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
2925 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2926
2927 Assert(pImage);
2928
2929 if (pImage)
2930 return pImage->cbSize;
2931 else
2932 return 0;
2933}
2934
2935
2936/** @copydoc VBOXHDDBACKEND::pfnGetFileSize */
2937static uint64_t iscsiGetFileSize(void *pBackendData)
2938{
2939 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
2940 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2941
2942 Assert(pImage);
2943 NOREF(pImage);
2944
2945 if (pImage)
2946 return pImage->cbSize;
2947 else
2948 return 0;
2949}
2950
2951
2952/** @copydoc VBOXHDDBACKEND::pfnGetPCHSGeometry */
2953static int iscsiGetPCHSGeometry(void *pBackendData,
2954 PPDMMEDIAGEOMETRY pPCHSGeometry)
2955{
2956 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p\n", pBackendData, pPCHSGeometry));
2957 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2958 int rc;
2959
2960 Assert(pImage);
2961
2962 if (pImage)
2963 rc = VERR_VD_GEOMETRY_NOT_SET;
2964 else
2965 rc = VERR_VD_NOT_OPENED;
2966
2967 LogFlowFunc(("returns %Rrc (PCHS=%u/%u/%u)\n", rc, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
2968 return rc;
2969}
2970
2971
2972/** @copydoc VBOXHDDBACKEND::pfnSetPCHSGeometry */
2973static int iscsiSetPCHSGeometry(void *pBackendData,
2974 PCPDMMEDIAGEOMETRY pPCHSGeometry)
2975{
2976 LogFlowFunc(("pBackendData=%#p pPCHSGeometry=%#p PCHS=%u/%u/%u\n", pBackendData, pPCHSGeometry, pPCHSGeometry->cCylinders, pPCHSGeometry->cHeads, pPCHSGeometry->cSectors));
2977 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
2978 int rc;
2979
2980 Assert(pImage);
2981
2982 if (pImage)
2983 {
2984 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
2985 {
2986 rc = VERR_VD_IMAGE_READ_ONLY;
2987 goto out;
2988 }
2989 rc = VERR_VD_GEOMETRY_NOT_SET;
2990 }
2991 else
2992 rc = VERR_VD_NOT_OPENED;
2993
2994out:
2995 LogFlowFunc(("returns %Rrc\n", rc));
2996 return rc;
2997}
2998
2999
3000/** @copydoc VBOXHDDBACKEND::pfnGetLCHSGeometry */
3001static int iscsiGetLCHSGeometry(void *pBackendData,
3002 PPDMMEDIAGEOMETRY pLCHSGeometry)
3003{
3004 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p\n", pBackendData, pLCHSGeometry));
3005 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3006 int rc;
3007
3008 Assert(pImage);
3009
3010 if (pImage)
3011 rc = VERR_VD_GEOMETRY_NOT_SET;
3012 else
3013 rc = VERR_VD_NOT_OPENED;
3014
3015 LogFlowFunc(("returns %Rrc (LCHS=%u/%u/%u)\n", rc, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
3016 return rc;
3017}
3018
3019
3020/** @copydoc VBOXHDDBACKEND::pfnGetImageFlags */
3021static unsigned iscsiGetImageFlags(void *pBackendData)
3022{
3023 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
3024 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3025 unsigned uImageFlags;
3026
3027 Assert(pImage);
3028 NOREF(pImage);
3029
3030 uImageFlags = VD_IMAGE_FLAGS_FIXED;
3031
3032 LogFlowFunc(("returns %#x\n", uImageFlags));
3033 return uImageFlags;
3034}
3035
3036
3037/** @copydoc VBOXHDDBACKEND::pfnGetOpenFlags */
3038static unsigned iscsiGetOpenFlags(void *pBackendData)
3039{
3040 LogFlowFunc(("pBackendData=%#p\n", pBackendData));
3041 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3042 unsigned uOpenFlags;
3043
3044 Assert(pImage);
3045
3046 if (pImage)
3047 uOpenFlags = pImage->uOpenFlags;
3048 else
3049 uOpenFlags = 0;
3050
3051 LogFlowFunc(("returns %#x\n", uOpenFlags));
3052 return uOpenFlags;
3053}
3054
3055
3056/** @copydoc VBOXHDDBACKEND::pfnSetOpenFlags */
3057static int iscsiSetOpenFlags(void *pBackendData, unsigned uOpenFlags)
3058{
3059 LogFlowFunc(("pBackendData=%#p\n uOpenFlags=%#x", pBackendData, uOpenFlags));
3060 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3061 int rc;
3062
3063 /* Image must be opened and the new flags must be valid. Just readonly and
3064 * info flags are supported. */
3065 if (!pImage || (uOpenFlags & ~(VD_OPEN_FLAGS_READONLY | VD_OPEN_FLAGS_INFO)))
3066 {
3067 rc = VERR_INVALID_PARAMETER;
3068 goto out;
3069 }
3070
3071 /* Implement this operation via reopening the image. */
3072 iscsiFreeImage(pImage, false);
3073 rc = iscsiOpenImage(pImage, uOpenFlags);
3074
3075out:
3076 LogFlowFunc(("returns %Rrc\n", rc));
3077 return rc;
3078}
3079
3080
3081/** @copydoc VBOXHDDBACKEND::pfnSetLCHSGeometry */
3082static int iscsiSetLCHSGeometry(void *pBackendData,
3083 PCPDMMEDIAGEOMETRY pLCHSGeometry)
3084{
3085 LogFlowFunc(("pBackendData=%#p pLCHSGeometry=%#p LCHS=%u/%u/%u\n", pBackendData, pLCHSGeometry, pLCHSGeometry->cCylinders, pLCHSGeometry->cHeads, pLCHSGeometry->cSectors));
3086 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3087 int rc;
3088
3089 Assert(pImage);
3090
3091 if (pImage)
3092 {
3093 if (pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY)
3094 {
3095 rc = VERR_VD_IMAGE_READ_ONLY;
3096 goto out;
3097 }
3098 rc = VERR_VD_GEOMETRY_NOT_SET;
3099 }
3100 else
3101 rc = VERR_VD_NOT_OPENED;
3102
3103out:
3104 LogFlowFunc(("returns %Rrc\n", rc));
3105 return rc;
3106}
3107
3108
3109/** @copydoc VBOXHDDBACKEND::pfnGetComment */
3110static int iscsiGetComment(void *pBackendData, char *pszComment,
3111 size_t cbComment)
3112{
3113 LogFlowFunc(("pBackendData=%#p pszComment=%#p cbComment=%zu\n", pBackendData, pszComment, cbComment));
3114 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3115 int rc;
3116
3117 Assert(pImage);
3118
3119 if (pImage)
3120 rc = VERR_NOT_SUPPORTED;
3121 else
3122 rc = VERR_VD_NOT_OPENED;
3123
3124 LogFlowFunc(("returns %Rrc comment='%s'\n", rc, pszComment));
3125 return rc;
3126}
3127
3128
3129/** @copydoc VBOXHDDBACKEND::pfnSetComment */
3130static int iscsiSetComment(void *pBackendData, const char *pszComment)
3131{
3132 LogFlowFunc(("pBackendData=%#p pszComment=\"%s\"\n", pBackendData, pszComment));
3133 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3134 int rc;
3135
3136 Assert(pImage);
3137
3138 if (pImage)
3139 {
3140 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
3141 rc = VERR_NOT_SUPPORTED;
3142 else
3143 rc = VERR_VD_IMAGE_READ_ONLY;
3144 }
3145 else
3146 rc = VERR_VD_NOT_OPENED;
3147
3148 LogFlowFunc(("returns %Rrc\n", rc));
3149 return rc;
3150}
3151
3152
3153/** @copydoc VBOXHDDBACKEND::pfnGetUuid */
3154static int iscsiGetUuid(void *pBackendData, PRTUUID pUuid)
3155{
3156 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
3157 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3158 int rc;
3159
3160 Assert(pImage);
3161
3162 if (pImage)
3163 rc = VERR_NOT_SUPPORTED;
3164 else
3165 rc = VERR_VD_NOT_OPENED;
3166
3167 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
3168 return rc;
3169}
3170
3171
3172/** @copydoc VBOXHDDBACKEND::pfnSetUuid */
3173static int iscsiSetUuid(void *pBackendData, PCRTUUID pUuid)
3174{
3175 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
3176 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3177 int rc;
3178
3179 LogFlowFunc(("%RTuuid\n", pUuid));
3180 Assert(pImage);
3181
3182 if (pImage)
3183 {
3184 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
3185 rc = VERR_NOT_SUPPORTED;
3186 else
3187 rc = VERR_VD_IMAGE_READ_ONLY;
3188 }
3189 else
3190 rc = VERR_VD_NOT_OPENED;
3191
3192 LogFlowFunc(("returns %Rrc\n", rc));
3193 return rc;
3194}
3195
3196
3197/** @copydoc VBOXHDDBACKEND::pfnGetModificationUuid */
3198static int iscsiGetModificationUuid(void *pBackendData, PRTUUID pUuid)
3199{
3200 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
3201 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3202 int rc;
3203
3204 Assert(pImage);
3205
3206 if (pImage)
3207 rc = VERR_NOT_SUPPORTED;
3208 else
3209 rc = VERR_VD_NOT_OPENED;
3210
3211 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
3212 return rc;
3213}
3214
3215
3216/** @copydoc VBOXHDDBACKEND::pfnSetModificationUuid */
3217static int iscsiSetModificationUuid(void *pBackendData, PCRTUUID pUuid)
3218{
3219 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
3220 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3221 int rc;
3222
3223 LogFlowFunc(("%RTuuid\n", pUuid));
3224 Assert(pImage);
3225
3226 if (pImage)
3227 {
3228 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
3229 rc = VERR_NOT_SUPPORTED;
3230 else
3231 rc = VERR_VD_IMAGE_READ_ONLY;
3232 }
3233 else
3234 rc = VERR_VD_NOT_OPENED;
3235
3236 LogFlowFunc(("returns %Rrc\n", rc));
3237 return rc;
3238}
3239
3240
3241/** @copydoc VBOXHDDBACKEND::pfnGetParentUuid */
3242static int iscsiGetParentUuid(void *pBackendData, PRTUUID pUuid)
3243{
3244 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
3245 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3246 int rc;
3247
3248 Assert(pImage);
3249
3250 if (pImage)
3251 rc = VERR_NOT_SUPPORTED;
3252 else
3253 rc = VERR_VD_NOT_OPENED;
3254
3255 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
3256 return rc;
3257}
3258
3259
3260/** @copydoc VBOXHDDBACKEND::pfnSetParentUuid */
3261static int iscsiSetParentUuid(void *pBackendData, PCRTUUID pUuid)
3262{
3263 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
3264 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3265 int rc;
3266
3267 LogFlowFunc(("%RTuuid\n", pUuid));
3268 Assert(pImage);
3269
3270 if (pImage)
3271 {
3272 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
3273 rc = VERR_NOT_SUPPORTED;
3274 else
3275 rc = VERR_VD_IMAGE_READ_ONLY;
3276 }
3277 else
3278 rc = VERR_VD_NOT_OPENED;
3279
3280 LogFlowFunc(("returns %Rrc\n", rc));
3281 return rc;
3282}
3283
3284
3285/** @copydoc VBOXHDDBACKEND::pfnGetParentModificationUuid */
3286static int iscsiGetParentModificationUuid(void *pBackendData, PRTUUID pUuid)
3287{
3288 LogFlowFunc(("pBackendData=%#p pUuid=%#p\n", pBackendData, pUuid));
3289 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3290 int rc;
3291
3292 Assert(pImage);
3293
3294 if (pImage)
3295 rc = VERR_NOT_SUPPORTED;
3296 else
3297 rc = VERR_VD_NOT_OPENED;
3298
3299 LogFlowFunc(("returns %Rrc (%RTuuid)\n", rc, pUuid));
3300 return rc;
3301}
3302
3303
3304/** @copydoc VBOXHDDBACKEND::pfnSetParentModificationUuid */
3305static int iscsiSetParentModificationUuid(void *pBackendData, PCRTUUID pUuid)
3306{
3307 LogFlowFunc(("pBackendData=%#p Uuid=%RTuuid\n", pBackendData, pUuid));
3308 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3309 int rc;
3310
3311 LogFlowFunc(("%RTuuid\n", pUuid));
3312 Assert(pImage);
3313
3314 if (pImage)
3315 {
3316 if (!(pImage->uOpenFlags & VD_OPEN_FLAGS_READONLY))
3317 rc = VERR_NOT_SUPPORTED;
3318 else
3319 rc = VERR_VD_IMAGE_READ_ONLY;
3320 }
3321 else
3322 rc = VERR_VD_NOT_OPENED;
3323
3324 LogFlowFunc(("returns %Rrc\n", rc));
3325 return rc;
3326}
3327
3328
3329/** @copydoc VBOXHDDBACKEND::pfnDump */
3330static void iscsiDump(void *pBackendData)
3331{
3332 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3333
3334 Assert(pImage);
3335 if (pImage)
3336 {
3337 /** @todo put something useful here */
3338 RTLogPrintf("Header: cVolume=%u\n", pImage->cVolume);
3339 }
3340}
3341
3342
3343/** @copydoc VBOXHDDBACKEND::pfnGetTimeStamp */
3344static int iscsiGetTimeStamp(void *pBackendData, PRTTIMESPEC pTimeStamp)
3345{
3346 LogFlowFunc(("pBackendData=%#p pTimeStamp=%#p\n", pBackendData, pTimeStamp));
3347 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3348 int rc = VERR_NOT_SUPPORTED;
3349
3350 Assert(pImage);
3351 NOREF(pImage);
3352
3353 LogFlowFunc(("returns %Rrc\n", rc));
3354 return rc;
3355}
3356
3357
3358/** @copydoc VBOXHDDBACKEND::pfnGetParentTimeStamp */
3359static int iscsiGetParentTimeStamp(void *pBackendData, PRTTIMESPEC pTimeStamp)
3360{
3361 LogFlowFunc(("pBackendData=%#p pTimeStamp=%#p\n", pBackendData, pTimeStamp));
3362 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3363 int rc = VERR_NOT_SUPPORTED;
3364
3365 Assert(pImage);
3366 NOREF(pImage);
3367
3368 LogFlowFunc(("returns %Rrc\n", rc));
3369 return rc;
3370}
3371
3372
3373/** @copydoc VBOXHDDBACKEND::pfnSetParentTimeStamp */
3374static int iscsiSetParentTimeStamp(void *pBackendData, PCRTTIMESPEC pTimeStamp)
3375{
3376 LogFlowFunc(("pBackendData=%#p pTimeStamp=%#p\n", pBackendData, pTimeStamp));
3377 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3378 int rc = VERR_NOT_SUPPORTED;
3379
3380 Assert(pImage);
3381 NOREF(pImage);
3382
3383 LogFlowFunc(("returns %Rrc\n", rc));
3384 return rc;
3385}
3386
3387
3388/** @copydoc VBOXHDDBACKEND::pfnGetParentFilename */
3389static int iscsiGetParentFilename(void *pBackendData, char **ppszParentFilename)
3390{
3391 LogFlowFunc(("pBackendData=%#p ppszParentFilename=%#p\n", pBackendData, ppszParentFilename));
3392 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3393 int rc = VERR_NOT_SUPPORTED;
3394
3395 Assert(pImage);
3396 NOREF(pImage);
3397
3398 LogFlowFunc(("returns %Rrc\n", rc));
3399 return rc;
3400}
3401
3402
3403/** @copydoc VBOXHDDBACKEND::pfnSetParentFilename */
3404static int iscsiSetParentFilename(void *pBackendData, const char *pszParentFilename)
3405{
3406 LogFlowFunc(("pBackendData=%#p pszParentFilename=%s\n", pBackendData, pszParentFilename));
3407 PISCSIIMAGE pImage = (PISCSIIMAGE)pBackendData;
3408 int rc = VERR_NOT_SUPPORTED;
3409
3410 Assert(pImage);
3411 NOREF(pImage);
3412
3413 LogFlowFunc(("returns %Rrc\n", rc));
3414 return rc;
3415}
3416
3417/** @copydoc VBOXHDDBACKEND::pfnComposeLocation */
3418static int iscsiComposeLocation(PVDINTERFACE pConfig, char **pszLocation)
3419{
3420 char *pszTarget = NULL;
3421 char *pszLUN = NULL;
3422 char *pszAddress = NULL;
3423 int rc = VDCFGQueryStringAlloc(VDGetInterfaceConfig(pConfig), pConfig->pvUser, "TargetName", &pszTarget);
3424 if (RT_SUCCESS(rc))
3425 {
3426 rc = VDCFGQueryStringAlloc(VDGetInterfaceConfig(pConfig), pConfig->pvUser, "LUN", &pszLUN);
3427 if (RT_SUCCESS(rc))
3428 {
3429 rc = VDCFGQueryStringAlloc(VDGetInterfaceConfig(pConfig), pConfig->pvUser, "TargetAddress", &pszAddress);
3430 if (RT_SUCCESS(rc))
3431 {
3432 if (RTStrAPrintf(pszLocation, "iscsi://%s/%s/%s",
3433 pszAddress, pszTarget, pszLUN) < 0)
3434 rc = VERR_NO_MEMORY;
3435 }
3436 }
3437 }
3438 RTMemFree(pszTarget);
3439 RTMemFree(pszLUN);
3440 RTMemFree(pszAddress);
3441 return rc;
3442}
3443
3444/** @copydoc VBOXHDDBACKEND::pfnComposeName */
3445static int iscsiComposeName(PVDINTERFACE pConfig, char **pszName)
3446{
3447 char *pszTarget = NULL;
3448 char *pszLUN = NULL;
3449 char *pszAddress = NULL;
3450 int rc = VDCFGQueryStringAlloc(VDGetInterfaceConfig(pConfig), pConfig->pvUser, "TargetName", &pszTarget);
3451 if (RT_SUCCESS(rc))
3452 {
3453 rc = VDCFGQueryStringAlloc(VDGetInterfaceConfig(pConfig), pConfig->pvUser, "LUN", &pszLUN);
3454 if (RT_SUCCESS(rc))
3455 {
3456 rc = VDCFGQueryStringAlloc(VDGetInterfaceConfig(pConfig), pConfig->pvUser, "TargetAddress", &pszAddress);
3457 if (RT_SUCCESS(rc))
3458 {
3459 /** @todo think about a nicer looking location scheme for iSCSI */
3460 if (RTStrAPrintf(pszName, "%s/%s/%s",
3461 pszAddress, pszTarget, pszLUN) < 0)
3462 rc = VERR_NO_MEMORY;
3463 }
3464 }
3465 }
3466 RTMemFree(pszTarget);
3467 RTMemFree(pszLUN);
3468 RTMemFree(pszAddress);
3469
3470 return rc;
3471}
3472
3473
3474VBOXHDDBACKEND g_ISCSIBackend =
3475{
3476 /* pszBackendName */
3477 "iSCSI",
3478 /* cbSize */
3479 sizeof(VBOXHDDBACKEND),
3480 /* uBackendCaps */
3481 VD_CAP_CONFIG | VD_CAP_TCPNET,
3482 /* papszFileExtensions */
3483 NULL,
3484 /* paConfigInfo */
3485 s_iscsiConfigInfo,
3486 /* hPlugin */
3487 NIL_RTLDRMOD,
3488 /* pfnCheckIfValid */
3489 iscsiCheckIfValid,
3490 /* pfnOpen */
3491 iscsiOpen,
3492 /* pfnCreate */
3493 iscsiCreate,
3494 /* pfnRename */
3495 iscsiRename,
3496 /* pfnClose */
3497 iscsiClose,
3498 /* pfnRead */
3499 iscsiRead,
3500 /* pfnWrite */
3501 iscsiWrite,
3502 /* pfnFlush */
3503 iscsiFlush,
3504 /* pfnGetVersion */
3505 iscsiGetVersion,
3506 /* pfnGetSize */
3507 iscsiGetSize,
3508 /* pfnGetFileSize */
3509 iscsiGetFileSize,
3510 /* pfnGetPCHSGeometry */
3511 iscsiGetPCHSGeometry,
3512 /* pfnSetPCHSGeometry */
3513 iscsiSetPCHSGeometry,
3514 /* pfnGetLCHSGeometry */
3515 iscsiGetLCHSGeometry,
3516 /* pfnSetLCHSGeometry */
3517 iscsiSetLCHSGeometry,
3518 /* pfnGetImageFlags */
3519 iscsiGetImageFlags,
3520 /* pfnGetOpenFlags */
3521 iscsiGetOpenFlags,
3522 /* pfnSetOpenFlags */
3523 iscsiSetOpenFlags,
3524 /* pfnGetComment */
3525 iscsiGetComment,
3526 /* pfnSetComment */
3527 iscsiSetComment,
3528 /* pfnGetUuid */
3529 iscsiGetUuid,
3530 /* pfnSetUuid */
3531 iscsiSetUuid,
3532 /* pfnGetModificationUuid */
3533 iscsiGetModificationUuid,
3534 /* pfnSetModificationUuid */
3535 iscsiSetModificationUuid,
3536 /* pfnGetParentUuid */
3537 iscsiGetParentUuid,
3538 /* pfnSetParentUuid */
3539 iscsiSetParentUuid,
3540 /* pfnGetParentModificationUuid */
3541 iscsiGetParentModificationUuid,
3542 /* pfnSetParentModificationUuid */
3543 iscsiSetParentModificationUuid,
3544 /* pfnDump */
3545 iscsiDump,
3546 /* pfnGetTimeStamp */
3547 iscsiGetTimeStamp,
3548 /* pfnGetParentTimeStamp */
3549 iscsiGetParentTimeStamp,
3550 /* pfnSetParentTimeStamp */
3551 iscsiSetParentTimeStamp,
3552 /* pfnGetParentFilename */
3553 iscsiGetParentFilename,
3554 /* pfnSetParentFilename */
3555 iscsiSetParentFilename,
3556 /* pfnIsAsyncIOSupported */
3557 NULL,
3558 /* pfnAsyncRead */
3559 NULL,
3560 /* pfnAsyncWrite */
3561 NULL,
3562 /* pfnComposeLocation */
3563 iscsiComposeLocation,
3564 /* pfnComposeName */
3565 iscsiComposeName
3566};
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