per Intel policy to include file commit date using git cmd below. The policy does not apply to non-Intel (C) notices. git log --follow -C90% --format=%ad --date default <file> | tail -1 and then pull just the 4 digit year from the result. Intel copyrights were not added to files where Intel either had no contribution ot the contribution lacked substance (ie license header updates, formatting changes, etc). Contribution date used "--follow -C95%" to get the most accurate date. Note that several files in this patch didn't end the license/(c) block with a blank comment line so these were added as the vast majority of files do have this last blank line. Simply there for consistency. Signed-off-by: paul luse <paul.e.luse@intel.com> Change-Id: Id5b7ce4f658fe87132f14139ead58d6e285c04d4 Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/15192 Tested-by: SPDK CI Jenkins <sys_sgci@intel.com> Reviewed-by: Jim Harris <james.r.harris@intel.com> Reviewed-by: Ben Walker <benjamin.walker@intel.com> Community-CI: Mellanox Build Bot
274 lines
5.7 KiB
C
274 lines
5.7 KiB
C
/* SPDX-License-Identifier: BSD-3-Clause
|
|
* Copyright (C) 2008-2012 Daisuke Aoyama <aoyama@peach.ne.jp>.
|
|
* Copyright (C) 2016 Intel Corporation.
|
|
* All rights reserved.
|
|
*/
|
|
|
|
#include "scsi_internal.h"
|
|
#include "spdk/endian.h"
|
|
#include "spdk/env.h"
|
|
#include "spdk/util.h"
|
|
|
|
static void
|
|
scsi_task_free_data(struct spdk_scsi_task *task)
|
|
{
|
|
if (task->alloc_len != 0) {
|
|
spdk_dma_free(task->iov.iov_base);
|
|
task->alloc_len = 0;
|
|
}
|
|
|
|
task->iov.iov_base = NULL;
|
|
task->iov.iov_len = 0;
|
|
}
|
|
|
|
void
|
|
spdk_scsi_task_put(struct spdk_scsi_task *task)
|
|
{
|
|
if (!task) {
|
|
return;
|
|
}
|
|
|
|
assert(task->ref > 0);
|
|
task->ref--;
|
|
|
|
if (task->ref == 0) {
|
|
struct spdk_bdev_io *bdev_io = task->bdev_io;
|
|
|
|
if (bdev_io) {
|
|
spdk_bdev_free_io(bdev_io);
|
|
}
|
|
|
|
scsi_task_free_data(task);
|
|
|
|
task->free_fn(task);
|
|
}
|
|
}
|
|
|
|
void
|
|
spdk_scsi_task_construct(struct spdk_scsi_task *task,
|
|
spdk_scsi_task_cpl cpl_fn,
|
|
spdk_scsi_task_free free_fn)
|
|
{
|
|
assert(task != NULL);
|
|
assert(cpl_fn != NULL);
|
|
assert(free_fn != NULL);
|
|
|
|
task->cpl_fn = cpl_fn;
|
|
task->free_fn = free_fn;
|
|
|
|
task->ref++;
|
|
|
|
/*
|
|
* Pre-fill the iov_buffers to point to the embedded iov
|
|
*/
|
|
assert(task->iov.iov_base == NULL);
|
|
task->iovs = &task->iov;
|
|
task->iovcnt = 1;
|
|
}
|
|
|
|
static void *
|
|
scsi_task_alloc_data(struct spdk_scsi_task *task, uint32_t alloc_len)
|
|
{
|
|
assert(task->alloc_len == 0);
|
|
|
|
task->iov.iov_base = spdk_dma_zmalloc(alloc_len, 0, NULL);
|
|
task->iov.iov_len = alloc_len;
|
|
task->alloc_len = alloc_len;
|
|
|
|
return task->iov.iov_base;
|
|
}
|
|
|
|
int
|
|
spdk_scsi_task_scatter_data(struct spdk_scsi_task *task, const void *src, size_t buf_len)
|
|
{
|
|
size_t len = 0;
|
|
size_t buf_left = buf_len;
|
|
int i;
|
|
struct iovec *iovs = task->iovs;
|
|
const uint8_t *pos;
|
|
|
|
if (buf_len == 0) {
|
|
return 0;
|
|
}
|
|
|
|
if (task->iovcnt == 1 && iovs[0].iov_base == NULL) {
|
|
scsi_task_alloc_data(task, buf_len);
|
|
iovs[0] = task->iov;
|
|
}
|
|
|
|
for (i = 0; i < task->iovcnt; i++) {
|
|
assert(iovs[i].iov_base != NULL);
|
|
len += iovs[i].iov_len;
|
|
}
|
|
|
|
if (len < buf_len) {
|
|
spdk_scsi_task_set_status(task, SPDK_SCSI_STATUS_CHECK_CONDITION,
|
|
SPDK_SCSI_SENSE_ILLEGAL_REQUEST,
|
|
SPDK_SCSI_ASC_INVALID_FIELD_IN_CDB,
|
|
SPDK_SCSI_ASCQ_CAUSE_NOT_REPORTABLE);
|
|
return -1;
|
|
}
|
|
|
|
pos = src;
|
|
|
|
for (i = 0; i < task->iovcnt; i++) {
|
|
len = spdk_min(iovs[i].iov_len, buf_left);
|
|
buf_left -= len;
|
|
memcpy(iovs[i].iov_base, pos, len);
|
|
pos += len;
|
|
}
|
|
|
|
return buf_len;
|
|
}
|
|
|
|
void *
|
|
spdk_scsi_task_gather_data(struct spdk_scsi_task *task, int *len)
|
|
{
|
|
int i;
|
|
struct iovec *iovs = task->iovs;
|
|
size_t buf_len = 0;
|
|
uint8_t *buf, *pos;
|
|
|
|
for (i = 0; i < task->iovcnt; i++) {
|
|
/* It is OK for iov_base to be NULL if iov_len is 0. */
|
|
assert(iovs[i].iov_base != NULL || iovs[i].iov_len == 0);
|
|
buf_len += iovs[i].iov_len;
|
|
}
|
|
|
|
if (buf_len == 0) {
|
|
*len = 0;
|
|
return NULL;
|
|
}
|
|
|
|
buf = calloc(1, buf_len);
|
|
if (buf == NULL) {
|
|
*len = -1;
|
|
return NULL;
|
|
}
|
|
|
|
pos = buf;
|
|
for (i = 0; i < task->iovcnt; i++) {
|
|
memcpy(pos, iovs[i].iov_base, iovs[i].iov_len);
|
|
pos += iovs[i].iov_len;
|
|
}
|
|
|
|
*len = buf_len;
|
|
return buf;
|
|
}
|
|
|
|
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)
|
|
{
|
|
if (sc == SPDK_SCSI_STATUS_CHECK_CONDITION) {
|
|
spdk_scsi_task_build_sense_data(task, sk, asc, ascq);
|
|
}
|
|
task->status = sc;
|
|
}
|
|
|
|
void
|
|
spdk_scsi_task_copy_status(struct spdk_scsi_task *dst,
|
|
struct spdk_scsi_task *src)
|
|
{
|
|
memcpy(dst->sense_data, src->sense_data, src->sense_data_len);
|
|
dst->sense_data_len = src->sense_data_len;
|
|
dst->status = src->status;
|
|
}
|
|
|
|
void
|
|
spdk_scsi_task_process_null_lun(struct spdk_scsi_task *task)
|
|
{
|
|
uint8_t buffer[36];
|
|
uint32_t allocation_len;
|
|
uint32_t data_len;
|
|
|
|
task->length = task->transfer_len;
|
|
if (task->cdb[0] == SPDK_SPC_INQUIRY) {
|
|
/*
|
|
* SPC-4 states that INQUIRY commands to an unsupported LUN
|
|
* must be served with PERIPHERAL QUALIFIER = 0x3 and
|
|
* PERIPHERAL DEVICE TYPE = 0x1F.
|
|
*/
|
|
data_len = sizeof(buffer);
|
|
|
|
memset(buffer, 0, data_len);
|
|
/* PERIPHERAL QUALIFIER(7-5) PERIPHERAL DEVICE TYPE(4-0) */
|
|
buffer[0] = 0x03 << 5 | 0x1f;
|
|
/* ADDITIONAL LENGTH */
|
|
buffer[4] = data_len - 5;
|
|
|
|
allocation_len = from_be16(&task->cdb[3]);
|
|
if (spdk_scsi_task_scatter_data(task, buffer, spdk_min(allocation_len, data_len)) >= 0) {
|
|
task->data_transferred = data_len;
|
|
task->status = SPDK_SCSI_STATUS_GOOD;
|
|
}
|
|
} else {
|
|
/* LOGICAL UNIT NOT SUPPORTED */
|
|
spdk_scsi_task_set_status(task, SPDK_SCSI_STATUS_CHECK_CONDITION,
|
|
SPDK_SCSI_SENSE_ILLEGAL_REQUEST,
|
|
SPDK_SCSI_ASC_LOGICAL_UNIT_NOT_SUPPORTED,
|
|
SPDK_SCSI_ASCQ_CAUSE_NOT_REPORTABLE);
|
|
task->data_transferred = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
spdk_scsi_task_process_abort(struct spdk_scsi_task *task)
|
|
{
|
|
spdk_scsi_task_set_status(task, SPDK_SCSI_STATUS_CHECK_CONDITION,
|
|
SPDK_SCSI_SENSE_ABORTED_COMMAND,
|
|
SPDK_SCSI_ASC_NO_ADDITIONAL_SENSE,
|
|
SPDK_SCSI_ASCQ_CAUSE_NOT_REPORTABLE);
|
|
}
|