| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436 |
- /*
- Copyright (C) 2010 by Ronnie Sahlberg <ronniesahlberg@gmail.com>
-
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 3 of the License, or
- (at your option) any later version.
-
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
-
- You should have received a copy of the GNU General Public License
- along with this program; if not, see <http://www.gnu.org/licenses/>.
- */
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <arpa/inet.h>
- #include "iscsi.h"
- #include "iscsi-private.h"
- #include "scsi-lowlevel.h"
- #include "dlinklist.h"
- struct iscsi_scsi_cbdata {
- struct iscsi_scsi_cbdata *prev, *next;
- iscsi_command_cb callback;
- void *private_data;
- struct scsi_task *task;
- };
- void iscsi_free_scsi_cbdata(struct iscsi_scsi_cbdata *scsi_cbdata)
- {
- if (scsi_cbdata == NULL) {
- return;
- }
- if (scsi_cbdata->task == NULL) {
- scsi_free_scsi_task(scsi_cbdata->task);
- scsi_cbdata->task = NULL;
- }
- free(scsi_cbdata);
- }
- static void iscsi_scsi_response_cb(struct iscsi_context *iscsi, int status, void *command_data, void *private_data)
- {
- struct iscsi_scsi_cbdata *scsi_cbdata = (struct iscsi_scsi_cbdata *)private_data;
- struct scsi_task *task = command_data;
- switch (status) {
- case ISCSI_STATUS_CHECK_CONDITION:
- scsi_cbdata->callback(iscsi, ISCSI_STATUS_CHECK_CONDITION, task, scsi_cbdata->private_data);
- return;
-
- case ISCSI_STATUS_GOOD:
- scsi_cbdata->callback(iscsi, ISCSI_STATUS_GOOD, task, scsi_cbdata->private_data);
- return;
- default:
- printf("Cant handle scsi status %d yet\n", status);
- scsi_cbdata->callback(iscsi, ISCSI_STATUS_ERROR, task, scsi_cbdata->private_data);
- }
- }
- static int iscsi_scsi_command_async(struct iscsi_context *iscsi, int lun, struct scsi_task *task, iscsi_command_cb cb, struct iscsi_data *data, void *private_data)
- {
- struct iscsi_pdu *pdu;
- struct iscsi_scsi_cbdata *scsi_cbdata;
- int flags;
- if (iscsi == NULL) {
- printf("trying to send command on NULL context\n");
- scsi_free_scsi_task(task);
- return -1;
- }
- if (iscsi->session_type != ISCSI_SESSION_NORMAL) {
- printf("Trying to send command on discovery session\n");
- scsi_free_scsi_task(task);
- return -2;
- }
- if (iscsi->is_loggedin == 0) {
- printf("Trying to send command while not logged in\n");
- scsi_free_scsi_task(task);
- return -3;
- }
- scsi_cbdata = malloc(sizeof(struct iscsi_scsi_cbdata));
- if (scsi_cbdata == NULL) {
- printf("failed to allocate scsi cbdata\n");
- scsi_free_scsi_task(task);
- return -4;
- }
- bzero(scsi_cbdata, sizeof(struct iscsi_scsi_cbdata));
- scsi_cbdata->task = task;
- scsi_cbdata->callback = cb;
- scsi_cbdata->private_data = private_data;
- pdu = iscsi_allocate_pdu(iscsi, ISCSI_PDU_SCSI_REQUEST, ISCSI_PDU_SCSI_RESPONSE);
- if (pdu == NULL) {
- printf("Failed to allocate text pdu\n");
- iscsi_free_scsi_cbdata(scsi_cbdata);
- return -5;
- }
- pdu->scsi_cbdata = scsi_cbdata;
- /* flags */
- flags = ISCSI_PDU_SCSI_FINAL|ISCSI_PDU_SCSI_ATTR_SIMPLE;
- switch (task->xfer_dir) {
- case SCSI_XFER_NONE:
- break;
- case SCSI_XFER_READ:
- flags |= ISCSI_PDU_SCSI_READ;
- break;
- case SCSI_XFER_WRITE:
- flags |= ISCSI_PDU_SCSI_WRITE;
- if (data == NULL) {
- printf("DATA-OUT command but data == NULL\n");
- iscsi_free_pdu(iscsi, pdu);
- return -5;
- }
- if (data->size != task->expxferlen) {
- printf("data size:%d is not same as expected data transfer length:%d\n", data->size, task->expxferlen);
- iscsi_free_pdu(iscsi, pdu);
- return -7;
- }
- if (iscsi_pdu_add_data(iscsi, pdu, data->data, data->size) != 0) {
- printf("Failed to add outdata to the pdu\n");
- iscsi_free_pdu(iscsi, pdu);
- return -6;
- }
- break;
- }
- iscsi_pdu_set_pduflags(pdu, flags);
- /* lun */
- iscsi_pdu_set_lun(pdu, lun);
- /* expxferlen */
- iscsi_pdu_set_expxferlen(pdu, task->expxferlen);
- /* cmdsn */
- iscsi_pdu_set_cmdsn(pdu, iscsi->cmdsn);
- pdu->cmdsn = iscsi->cmdsn;
- iscsi->cmdsn++;
- /* exp statsn */
- iscsi_pdu_set_expstatsn(pdu, iscsi->statsn+1);
-
- /* cdb */
- iscsi_pdu_set_cdb(pdu, task);
- pdu->callback = iscsi_scsi_response_cb;
- pdu->private_data = scsi_cbdata;
- if (iscsi_queue_pdu(iscsi, pdu) != 0) {
- printf("failed to queue iscsi scsi pdu\n");
- iscsi_free_pdu(iscsi, pdu);
- return -6;
- }
- return 0;
- }
- int iscsi_process_scsi_reply(struct iscsi_context *iscsi, struct iscsi_pdu *pdu, const unsigned char *hdr, int size)
- {
- int statsn, flags, response, status;
- struct iscsi_scsi_cbdata *scsi_cbdata = pdu->scsi_cbdata;
- struct scsi_task *task = scsi_cbdata->task;
- statsn = ntohl(*(uint32_t *)&hdr[24]);
- if (statsn > (int)iscsi->statsn) {
- iscsi->statsn = statsn;
- }
- flags = hdr[1];
- if ((flags&ISCSI_PDU_DATA_FINAL) == 0) {
- printf("scsi response pdu but Final bit is not set: 0x%02x\n", flags);
- pdu->callback(iscsi, ISCSI_STATUS_ERROR, task, pdu->private_data);
- return -1;
- }
- if ((flags&ISCSI_PDU_DATA_ACK_REQUESTED) != 0) {
- printf("scsi response asked for ACK 0x%02x\n", flags);
- pdu->callback(iscsi, ISCSI_STATUS_ERROR, task, pdu->private_data);
- return -1;
- }
- /* for now, we ignore all overflow/underflow flags. We just print/log them so we can tweak
- * libiscsi to not generate under/over flows
- */
- if ((flags&ISCSI_PDU_DATA_BIDIR_OVERFLOW) != 0) {
- printf("scsi response contains bidir overflow 0x%02x\n", flags);
- }
- if ((flags&ISCSI_PDU_DATA_BIDIR_UNDERFLOW) != 0) {
- printf("scsi response contains bidir underflow 0x%02x\n", flags);
- }
- if ((flags&ISCSI_PDU_DATA_RESIDUAL_OVERFLOW) != 0) {
- printf("scsi response contains residual overflow 0x%02x\n", flags);
- }
- if ((flags&ISCSI_PDU_DATA_RESIDUAL_UNDERFLOW) != 0) {
- printf("scsi response contains residual underflow 0x%02x\n", flags);
- }
- response = hdr[2];
- if (response != 0x00) {
- printf("scsi reply response field:%d\n", response);
- }
- status = hdr[3];
- switch (status) {
- case ISCSI_STATUS_GOOD:
- task->datain.data = pdu->indata.data;
- task->datain.size = pdu->indata.size;
- pdu->callback(iscsi, ISCSI_STATUS_GOOD, &pdu->indata, pdu->private_data);
- break;
- case ISCSI_STATUS_CHECK_CONDITION:
- task->datain.data = discard_const(hdr + ISCSI_HEADER_SIZE);
- task->datain.size = size - ISCSI_HEADER_SIZE;
- task->sense.error_type = task->datain.data[2] & 0x7f;
- task->sense.key = task->datain.data[4] & 0x0f;
- task->sense.ascq = ntohs(*(uint16_t *)&(task->datain.data[14]));
- pdu->callback(iscsi, ISCSI_STATUS_CHECK_CONDITION, task, pdu->private_data);
- break;
- default:
- printf("Unknown status :%d\n", status);
- pdu->callback(iscsi, ISCSI_STATUS_ERROR, task, pdu->private_data);
- return -1;
- }
- return 0;
- }
- int iscsi_process_scsi_data_in(struct iscsi_context *iscsi, struct iscsi_pdu *pdu, const unsigned char *hdr, int size, int *is_finished)
- {
- int statsn, flags, status;
- struct iscsi_scsi_cbdata *scsi_cbdata = pdu->scsi_cbdata;
- struct scsi_task *task = scsi_cbdata->task;
- int dsl;
- statsn = ntohl(*(uint32_t *)&hdr[24]);
- if (statsn > (int)iscsi->statsn) {
- iscsi->statsn = statsn;
- }
- flags = hdr[1];
- if ((flags&ISCSI_PDU_DATA_ACK_REQUESTED) != 0) {
- printf("scsi response asked for ACK 0x%02x\n", flags);
- pdu->callback(iscsi, ISCSI_STATUS_ERROR, task, pdu->private_data);
- return -1;
- }
- /* for now, we ignore all overflow/underflow flags. We just print/log them so we can tweak
- * libiscsi to not generate under/over flows
- */
- if ((flags&ISCSI_PDU_DATA_RESIDUAL_OVERFLOW) != 0) {
- printf("scsi response contains residual overflow 0x%02x\n", flags);
- }
- if ((flags&ISCSI_PDU_DATA_RESIDUAL_UNDERFLOW) != 0) {
- printf("scsi response contains residual underflow 0x%02x\n", flags);
- }
- dsl = ntohl(*(uint32_t *)&hdr[4])&0x00ffffff;
- if (dsl > size - ISCSI_HEADER_SIZE) {
- printf ("dsl is :%d, while buffser size if %d\n", dsl, size - ISCSI_HEADER_SIZE);
- }
- if (iscsi_add_data(&pdu->indata, discard_const(hdr + ISCSI_HEADER_SIZE), dsl, 0) != 0) {
- printf("failed to add data to pdu in buffer\n");
- return -3;
- }
- if ((flags&ISCSI_PDU_DATA_FINAL) == 0) {
- printf("scsi data-in without Final bit: 0x%02x\n", flags);
- *is_finished = 0;
- }
- if ((flags&ISCSI_PDU_DATA_CONTAINS_STATUS) == 0) {
- printf("scsi data-in without Status bit: 0x%02x\n", flags);
- *is_finished = 0;
- }
- if (*is_finished == 0) {
- return 0;
- }
- /* this was the final data-in packet in the sequence and it has the s-bit set, so invoke the
- * callback.
- */
- status = hdr[3];
- task->datain.data = pdu->indata.data;
- task->datain.size = pdu->indata.size;
- pdu->callback(iscsi, status, task, pdu->private_data);
- return 0;
- }
- /*
- * SCSI commands
- */
- int iscsi_testunitready_async(struct iscsi_context *iscsi, int lun, iscsi_command_cb cb, void *private_data)
- {
- struct scsi_task *task;
- int ret;
- if ((task = scsi_cdb_testunitready()) == NULL) {
- printf("Failed to create testunitready cdb\n");
- return -1;
- }
- ret = iscsi_scsi_command_async(iscsi, lun, task, cb, NULL, private_data);
- return ret;
- }
- int iscsi_reportluns_async(struct iscsi_context *iscsi, iscsi_command_cb cb, int report_type, int alloc_len, void *private_data)
- {
- struct scsi_task *task;
- int ret;
- if (alloc_len < 16) {
- printf("Minimum allowed alloc len for reportluns is 16. You specified %d\n", alloc_len);
- return -1;
- }
- if ((task = scsi_reportluns_cdb(report_type, alloc_len)) == NULL) {
- printf("Failed to create reportluns cdb\n");
- return -2;
- }
- /* report luns are always sent to lun 0 */
- ret = iscsi_scsi_command_async(iscsi, 0, task, cb, NULL, private_data);
- return ret;
- }
- int iscsi_inquiry_async(struct iscsi_context *iscsi, int lun, iscsi_command_cb cb, int evpd, int page_code, int maxsize, void *private_data)
- {
- struct scsi_task *task;
- int ret;
- if ((task = scsi_cdb_inquiry(evpd, page_code, maxsize)) == NULL) {
- printf("Failed to create inquiry cdb\n");
- return -1;
- }
- ret = iscsi_scsi_command_async(iscsi, lun, task, cb, NULL, private_data);
- return ret;
- }
- int iscsi_readcapacity10_async(struct iscsi_context *iscsi, int lun, iscsi_command_cb cb, int lba, int pmi, void *private_data)
- {
- struct scsi_task *task;
- int ret;
- if ((task = scsi_cdb_readcapacity10(lba, pmi)) == NULL) {
- printf("Failed to create readcapacity10 cdb\n");
- return -1;
- }
- ret = iscsi_scsi_command_async(iscsi, lun, task, cb, NULL, private_data);
- return ret;
- }
- int iscsi_read10_async(struct iscsi_context *iscsi, int lun, iscsi_command_cb cb, int lba, int datalen, int blocksize, void *private_data)
- {
- struct scsi_task *task;
- int ret;
- if (datalen % blocksize != 0) {
- printf("datalen:%d is not a multiple of the blocksize:%d\n", datalen, blocksize);
- return -1;
- }
- if ((task = scsi_cdb_read10(lba, datalen, blocksize)) == NULL) {
- printf("Failed to create read10 cdb\n");
- return -2;
- }
- ret = iscsi_scsi_command_async(iscsi, lun, task, cb, NULL, private_data);
- return ret;
- }
- int iscsi_write10_async(struct iscsi_context *iscsi, int lun, iscsi_command_cb cb, unsigned char *data, int datalen, int lba, int fua, int fuanv, int blocksize, void *private_data)
- {
- struct scsi_task *task;
- struct iscsi_data outdata;
- int ret;
- if (datalen % blocksize != 0) {
- printf("datalen:%d is not a multiple of the blocksize:%d\n", datalen, blocksize);
- return -1;
- }
- if ((task = scsi_cdb_write10(lba, datalen, fua, fuanv, blocksize)) == NULL) {
- printf("Failed to create read10 cdb\n");
- return -2;
- }
- outdata.data = data;
- outdata.size = datalen;
- ret = iscsi_scsi_command_async(iscsi, lun, task, cb, &outdata, private_data);
- return ret;
- }
- int iscsi_modesense6_async(struct iscsi_context *iscsi, int lun, iscsi_command_cb cb, int dbd, int pc, int page_code, int sub_page_code, unsigned char alloc_len, void *private_data)
- {
- struct scsi_task *task;
- int ret;
- if ((task = scsi_cdb_modesense6(dbd, pc, page_code, sub_page_code, alloc_len)) == NULL) {
- printf("Failed to create modesense6 cdb\n");
- return -2;
- }
- ret = iscsi_scsi_command_async(iscsi, lun, task, cb, NULL, private_data);
- return ret;
- }
|