VirtualBox

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

Last change on this file since 27436 was 27181, checked in by vboxsync, 15 years ago

Storage/iSCSI: Increase sense data buffers, and truncate sense data if the buffers are too small. Also fixes a potential buffer overrun when collecting the sense data.

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