/*- * BSD LICENSE * * Copyright (C) 2008-2012 Daisuke Aoyama . * Copyright (c) Intel Corporation. * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * Neither the name of Intel Corporation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "scsi_internal.h" #include "spdk/endian.h" #include "spdk/env.h" void spdk_scsi_task_put(struct spdk_scsi_task *task) { if (!task) { return; } task->ref--; if (task->ref == 0) { struct spdk_bdev_io *bdev_io = task->blockdev_io; if (task->parent) { spdk_scsi_task_put(task->parent); task->parent = NULL; } if (bdev_io) { /* due to lun reset, the bdev_io status could be pending */ if (bdev_io->status == SPDK_BDEV_IO_STATUS_PENDING) { bdev_io->status = SPDK_BDEV_IO_STATUS_FAILED; } spdk_bdev_free_io(bdev_io); } spdk_scsi_task_free_data(task); assert(task->owner_task_ctr != NULL); if (*(task->owner_task_ctr) > 0) { *(task->owner_task_ctr) -= 1; } else { SPDK_ERRLOG("task counter already 0\n"); } task->free_fn(task); } } void spdk_scsi_task_construct(struct spdk_scsi_task *task, uint32_t *owner_task_ctr, struct spdk_scsi_task *parent) { task->ref++; assert(owner_task_ctr != NULL); task->owner_task_ctr = owner_task_ctr; *owner_task_ctr += 1; /* * Pre-fill the iov_buffers to point to the embedded iov */ assert(task->iov.iov_base == NULL); task->iovs = &task->iov; task->iovcnt = 1; if (parent != NULL) { parent->ref++; task->parent = parent; task->type = parent->type; task->dxfer_dir = parent->dxfer_dir; task->transfer_len = parent->transfer_len; task->lun = parent->lun; task->cdb = parent->cdb; task->target_port = parent->target_port; task->initiator_port = parent->initiator_port; task->id = parent->id; } } void spdk_scsi_task_free_data(struct spdk_scsi_task *task) { if (task->alloc_len != 0) { spdk_free(task->iov.iov_base); task->alloc_len = 0; } task->iov.iov_base = NULL; task->iov.iov_len = 0; } void * spdk_scsi_task_alloc_data(struct spdk_scsi_task *task, uint32_t alloc_len) { assert(task->alloc_len == 0); /* Only one buffer is managable */ if (task->iovcnt != 1) { return NULL; } /* This is workaround for buffers shorter than 4kb */ if (task->iov.iov_base == NULL) { task->iov.iov_base = spdk_zmalloc(alloc_len, 0, NULL); task->alloc_len = alloc_len; } task->iov.iov_len = alloc_len; assert(&task->iov == task->iovs); return task->iov.iov_base; } void spdk_scsi_task_set_data(struct spdk_scsi_task *task, void *data, uint32_t len) { assert(task->iovcnt == 1); assert(task->alloc_len == 0); task->iovs[0].iov_base = data; task->iovs[0].iov_len = len; } void spdk_scsi_task_build_sense_data(struct spdk_scsi_task *task, int sk, int asc, int ascq) { uint8_t *cp; int resp_code; resp_code = 0x70; /* Current + Fixed format */ /* Sense Data */ cp = task->sense_data; /* VALID(7) RESPONSE CODE(6-0) */ cp[0] = 0x80 | resp_code; /* Obsolete */ cp[1] = 0; /* FILEMARK(7) EOM(6) ILI(5) SENSE KEY(3-0) */ cp[2] = sk & 0xf; /* INFORMATION */ memset(&cp[3], 0, 4); /* ADDITIONAL SENSE LENGTH */ cp[7] = 10; /* COMMAND-SPECIFIC INFORMATION */ memset(&cp[8], 0, 4); /* ADDITIONAL SENSE CODE */ cp[12] = asc; /* ADDITIONAL SENSE CODE QUALIFIER */ cp[13] = ascq; /* FIELD REPLACEABLE UNIT CODE */ cp[14] = 0; /* SKSV(7) SENSE KEY SPECIFIC(6-0,7-0,7-0) */ cp[15] = 0; cp[16] = 0; cp[17] = 0; /* SenseLength */ task->sense_data_len = 18; } void spdk_scsi_task_set_status(struct spdk_scsi_task *task, int sc, int sk, int asc, int ascq) { spdk_scsi_task_build_sense_data(task, sk, asc, ascq); task->status = sc; }