mirror of
https://github.com/SCST-project/scst.git
synced 2026-05-22 05:01:27 +00:00
svn+ssh://yanb123@svn.code.sf.net/p/scst/svn/trunk
........
r5510 | vlnb | 2014-05-09 06:51:10 +0300 (Fri, 09 May 2014) | 8 lines
scst: Make pr path configurable
Make the path of the file in which persistent reservation information
is stored configurable via sysfs.
Signed-off-by: Bart Van Assche <bvanassche@acm.org> with some improvements and fixes
........
r5511 | vlnb | 2014-05-09 06:57:20 +0300 (Fri, 09 May 2014) | 9 lines
scst_vdisk: Introduce three helper functions
Introduce the vdisk_bio_alloc(), vdisk_bio_set_failfast() and
vdisk_bio_set_hoq() helper functions. This patch does not change any
functionality.
Signed-off-by: Bart Van Assche <bvanassche@acm.org>
........
r5512 | vlnb | 2014-05-09 07:14:39 +0300 (Fri, 09 May 2014) | 3 lines
Cleanup
........
r5513 | vlnb | 2014-05-10 02:20:49 +0300 (Sat, 10 May 2014) | 7 lines
Fix sense code for invalid service actions
According to T10, multibyte opcode commands with not supported service
actions must be refused with INVALID FIELD IN CDB instead of INVALID
OPCODE
........
r5514 | vlnb | 2014-05-10 02:25:13 +0300 (Sat, 10 May 2014) | 3 lines
Cleanup
........
r5515 | vlnb | 2014-05-10 05:10:22 +0300 (Sat, 10 May 2014) | 3 lines
Improve tracing of Unit Attentions
........
r5516 | vlnb | 2014-05-10 06:55:11 +0300 (Sat, 10 May 2014) | 3 lines
Follow up for r5513
........
r5517 | bvassche | 2014-05-10 09:20:01 +0300 (Sat, 10 May 2014) | 6 lines
scst_pres: Fix a recently introduced checkpatch warning
Avoid that checkpatch reports the following:
WARNING: do {} while (0) macros should not be semicolon terminated
........
r5518 | bvassche | 2014-05-12 18:56:45 +0300 (Mon, 12 May 2014) | 1 line
ib_srpt: Source code comment spelling fix
........
r5520 | vlnb | 2014-05-15 04:39:12 +0300 (Thu, 15 May 2014) | 11 lines
iscsi-scst: One major number per thread pool
Assign one major number per thread pool instead of as many major
numbers as there are threads in a thread pool. Do not increment
'major' if thread pool allocation fails. Micro-optimize
iscsi_threads_pool_get() by eliminating the assignment to 'fn'
and the write via snprintf() into name[].
Signed-off-by: Bart Van Assche <bvanassche@acm.org>
........
r5521 | vlnb | 2014-05-16 05:05:38 +0300 (Fri, 16 May 2014) | 3 lines
Cleanups
........
r5522 | vlnb | 2014-05-16 05:37:29 +0300 (Fri, 16 May 2014) | 5 lines
Cleanup
Those functions might be called on some corner cases without pr_mutex held
........
r5523 | vlnb | 2014-05-17 03:18:42 +0300 (Sat, 17 May 2014) | 8 lines
scst_lib: Clarify scst_init_cmd() documentation
The possible return values of scst_init_cmd() are -1, 0 and 1. Mention
this in the comment header above that function.
Signed-off-by: Bart Van Assche <bvanassche@acm.org>
........
r5524 | vlnb | 2014-05-17 04:04:08 +0300 (Sat, 17 May 2014) | 29 lines
scst_main: Fix race between scst_resume_activity() and scst_init_thread()
After SCST_FLAG_SUSPENDED has been cleared it is essential that
scst_do_job_init() reexamines scst_init_cmd_list to avoid that
commands get stuck in the command init list. This patch fixes the
following race condition that can occur if SCST_FLAG_SUSPENDED
has been set and if scst_init_cmd_list is not empty:
* scst_do_job_init() returns to scst_init_thread() and leaves the
commands that were on the init list on that list.
* scst_init_thread() invokes test_init_cmd_list().
* test_init_cmd_list() returns false because SCST_FLAG_SUSPENDED
has been set.
* scst_resume_activity() clears SCST_FLAG_SUSPENDED and invokes
wake_up_all(&scst_init_cmd_list_waitQ). However, since
scst_init_thread() has not yet added the init thread back to
scst_init_cmd_list_waitQ this wake_up_all() call doesn't do
anything.
* scst_init_thread() adds the init thread to scst_init_cmd_list_waitQ
and unlocks scst_init_lock.
Additionally, remove an unneeded smp_mb__after_clear_bit() call.
wake_up_all() guarantees that if it wakes up a thread that that
thread sees all store operations that were performed by the thread
that invoked wake_up_all() and that preceeded the wake_up_all()
invocation.
Signed-off-by: Bart Van Assche <bvanassche@acm.org>
........
r5525 | vlnb | 2014-05-17 04:16:33 +0300 (Sat, 17 May 2014) | 7 lines
scst_vdisk: Introduce vdisk_reexamine() and vdisk_close_fd()
This patch does not change any functionality
Signed-off-by: Bart Van Assche <bvanassche@acm.org>
........
r5526 | bvassche | 2014-05-18 14:26:42 +0300 (Sun, 18 May 2014) | 1 line
scst_vdisk: Handle attach failures properly (follow-up for r5525)
........
r5527 | bvassche | 2014-05-18 19:27:01 +0300 (Sun, 18 May 2014) | 1 line
nightly build: Update kernel versions
........
r5532 | vlnb | 2014-05-21 02:39:57 +0300 (Wed, 21 May 2014) | 3 lines
Prevent potential deadlock between scst_del_threads() and commands taking scst_mutex
........
r5533 | vlnb | 2014-05-22 05:56:20 +0300 (Thu, 22 May 2014) | 3 lines
Improve handling of aborted internal commands
........
git-svn-id: http://svn.code.sf.net/p/scst/svn/branches/iser@5554 d57e44dd-8a1f-0410-8b47-8ef2f437770f
1701 lines
42 KiB
C
1701 lines
42 KiB
C
/*
|
|
* common.c
|
|
*
|
|
* Copyright (C) 2007 - 2014 Vladislav Bolkhovitin <vst@vlnb.net>
|
|
* Copyright (C) 2007 - 2014 Fusion-io, Inc.
|
|
*
|
|
* 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, version 2
|
|
* of the License.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include <ctype.h>
|
|
#include <errno.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <unistd.h>
|
|
#include <stdint.h>
|
|
#include <getopt.h>
|
|
#include <inttypes.h>
|
|
|
|
#include <sys/ioctl.h>
|
|
#include <sys/poll.h>
|
|
|
|
#include <arpa/inet.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include "common.h"
|
|
|
|
static void exec_inquiry(struct vdisk_cmd *vcmd);
|
|
static void exec_request_sense(struct vdisk_cmd *vcmd);
|
|
static void exec_mode_sense(struct vdisk_cmd *vcmd);
|
|
static void exec_mode_select(struct vdisk_cmd *vcmd);
|
|
static void exec_read_capacity(struct vdisk_cmd *vcmd);
|
|
static void exec_read_capacity16(struct vdisk_cmd *vcmd);
|
|
static void exec_read_toc(struct vdisk_cmd *vcmd);
|
|
static void exec_prevent_allow_medium_removal(struct vdisk_cmd *vcmd);
|
|
static int exec_fsync(struct vdisk_cmd *vcmd);
|
|
static void exec_read(struct vdisk_cmd *vcmd, loff_t loff);
|
|
static void exec_write(struct vdisk_cmd *vcmd, loff_t loff);
|
|
static void exec_verify(struct vdisk_cmd *vcmd, loff_t loff);
|
|
|
|
static inline void set_cmd_error_status(struct scst_user_scsi_cmd_reply_exec *reply,
|
|
int status)
|
|
{
|
|
reply->status = status;
|
|
reply->resp_data_len = 0;
|
|
return;
|
|
}
|
|
|
|
static int set_sense(uint8_t *buffer, int len, int key, int asc, int ascq)
|
|
{
|
|
int res = 18;
|
|
|
|
EXTRACHECKS_BUG_ON(len < res);
|
|
|
|
memset(buffer, 0, res);
|
|
|
|
buffer[0] = 0x70; /* Error Code */
|
|
buffer[2] = key; /* Sense Key */
|
|
buffer[7] = 0x0a; /* Additional Sense Length */
|
|
buffer[12] = asc; /* ASC */
|
|
buffer[13] = ascq; /* ASCQ */
|
|
|
|
TRACE_BUFFER("Sense set", buffer, res);
|
|
return res;
|
|
}
|
|
|
|
/*
|
|
* ToDo: implement analogs of scst_set_invalid_field_in_cdb() and
|
|
* scst_set_invalid_field_in_parm_list()
|
|
*/
|
|
|
|
void set_cmd_error(struct vdisk_cmd *vcmd, int key, int asc, int ascq)
|
|
{
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
EXTRACHECKS_BUG_ON(vcmd->cmd->subcode != SCST_USER_EXEC);
|
|
|
|
set_cmd_error_status(reply, SAM_STAT_CHECK_CONDITION);
|
|
reply->sense_len = set_sense(vcmd->sense, sizeof(vcmd->sense), key,
|
|
asc, ascq);
|
|
reply->psense_buffer = (unsigned long)vcmd->sense;
|
|
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
void set_busy(struct scst_user_scsi_cmd_reply_exec *reply)
|
|
{
|
|
TRACE_ENTRY();
|
|
|
|
set_cmd_error_status(reply, SAM_STAT_TASK_SET_FULL);
|
|
TRACE_MGMT_DBG("%s", "Sending QUEUE_FULL status");
|
|
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static int do_parse(struct vdisk_cmd *vcmd)
|
|
{
|
|
int res = 0;
|
|
struct scst_user_scsi_cmd_parse *cmd = &vcmd->cmd->parse_cmd;
|
|
struct scst_user_scsi_cmd_reply_parse *reply = &vcmd->reply->parse_reply;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
memset(reply, 0, sizeof(*reply));
|
|
vcmd->reply->cmd_h = vcmd->cmd->cmd_h;
|
|
vcmd->reply->subcode = vcmd->cmd->subcode;
|
|
|
|
if (cmd->expected_values_set == 0) {
|
|
PRINT_ERROR("%s", "Oops, expected values are not set");
|
|
reply->bufflen = -1; /* invalid value */
|
|
goto out;
|
|
}
|
|
|
|
reply->queue_type = cmd->queue_type;
|
|
reply->data_direction = cmd->expected_data_direction;
|
|
reply->lba = cmd->lba;
|
|
reply->data_len = cmd->expected_transfer_len;
|
|
reply->bufflen = cmd->expected_transfer_len;
|
|
reply->out_bufflen = cmd->expected_out_transfer_len;
|
|
reply->cdb_len = cmd->cdb_len;
|
|
|
|
if (cmd->op_flags & SCST_INFO_VALID)
|
|
reply->op_flags = cmd->op_flags;
|
|
else {
|
|
TRACE_DBG("Extra parse (op %x)", cmd->cdb[0]);
|
|
|
|
if (reply->data_direction & SCST_DATA_WRITE)
|
|
reply->op_flags |= SCST_WRITE_MEDIUM;
|
|
reply->op_flags |= SCST_INFO_VALID;
|
|
}
|
|
|
|
out:
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
struct vdisk_tgt_dev *find_tgt_dev(struct vdisk_dev *dev, uint64_t sess_h)
|
|
{
|
|
unsigned int i;
|
|
struct vdisk_tgt_dev *res = NULL;
|
|
|
|
for(i = 0; i < ARRAY_SIZE(dev->tgt_devs); i++) {
|
|
if (dev->tgt_devs[i].sess_h == sess_h) {
|
|
res = &dev->tgt_devs[i];
|
|
break;
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
struct vdisk_tgt_dev *find_empty_tgt_dev(struct vdisk_dev *dev)
|
|
{
|
|
unsigned int i;
|
|
struct vdisk_tgt_dev *res = NULL;
|
|
|
|
for(i = 0; i < ARRAY_SIZE(dev->tgt_devs); i++) {
|
|
if (dev->tgt_devs[i].sess_h == 0) {
|
|
res = &dev->tgt_devs[i];
|
|
break;
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
static int do_exec(struct vdisk_cmd *vcmd)
|
|
{
|
|
int res = 0;
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
uint64_t lba_start = cmd->lba;
|
|
int64_t data_len = cmd->data_len;
|
|
uint8_t *cdb = cmd->cdb;
|
|
int opcode = cdb[0];
|
|
loff_t loff;
|
|
int fua = 0;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
switch(cmd->queue_type) {
|
|
case SCST_CMD_QUEUE_ORDERED:
|
|
TRACE(TRACE_ORDER, "ORDERED cmd_h %d", vcmd->cmd->cmd_h);
|
|
break;
|
|
case SCST_CMD_QUEUE_HEAD_OF_QUEUE:
|
|
TRACE(TRACE_ORDER, "HQ cmd_h %d", vcmd->cmd->cmd_h);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
memset(reply, 0, sizeof(*reply));
|
|
vcmd->reply->cmd_h = vcmd->cmd->cmd_h;
|
|
vcmd->reply->subcode = vcmd->cmd->subcode;
|
|
reply->reply_type = SCST_EXEC_REPLY_COMPLETED;
|
|
|
|
#ifdef DEBUG_SENSE
|
|
if ((random() % 100000) == 75) {
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
goto out;
|
|
}
|
|
#endif
|
|
|
|
#ifdef DEBUG_TM_IGNORE
|
|
if (dev->debug_tm_ignore && (random() % 10000) == 75) {
|
|
TRACE_MGMT_DBG("Ignore cmd op %x (h=%d)", cdb[0],
|
|
vcmd->cmd->cmd_h);
|
|
res = 150;
|
|
goto out;
|
|
}
|
|
#endif
|
|
|
|
if ((cmd->pbuf == 0) && (cmd->alloc_len != 0)) {
|
|
#ifdef DEBUG_NOMEM
|
|
if ((random() % 100) == 75)
|
|
cmd->pbuf = 0;
|
|
else
|
|
#endif
|
|
cmd->pbuf = (unsigned long)dev->alloc_fn(cmd->alloc_len);
|
|
TRACE_MEM("Buf %"PRIx64" alloced, len %d", cmd->pbuf,
|
|
cmd->alloc_len);
|
|
reply->pbuf = cmd->pbuf;
|
|
if (cmd->pbuf == 0) {
|
|
TRACE(TRACE_OUT_OF_MEM, "Unable to allocate buffer "
|
|
"(len %d)", cmd->alloc_len);
|
|
#ifndef DEBUG_NOMEM
|
|
set_busy(reply);
|
|
#endif
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
if (cmd->data_direction & SCST_DATA_READ)
|
|
reply->resp_data_len = cmd->bufflen;
|
|
|
|
loff = (loff_t)lba_start << dev->block_shift;
|
|
TRACE_DBG("cmd %d, buf %"PRIx64", lba_start %"PRId64", loff %"PRId64
|
|
", data_len %"PRId64, vcmd->cmd->cmd_h, cmd->pbuf, lba_start,
|
|
(uint64_t)loff, data_len);
|
|
if ((loff < 0) || (data_len < 0) || ((loff + data_len) > dev->file_size)) {
|
|
PRINT_INFO("Access beyond the end of the device "
|
|
"(%"PRId64" of %"PRId64", len %"PRId64")", (uint64_t)loff,
|
|
(uint64_t)dev->file_size, data_len);
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(
|
|
scst_sense_block_out_range_error));
|
|
goto out;
|
|
}
|
|
|
|
switch (opcode) {
|
|
case WRITE_10:
|
|
case WRITE_12:
|
|
case WRITE_16:
|
|
fua = (cdb[1] & 0x8);
|
|
if (cdb[1] & 0x8) {
|
|
TRACE(TRACE_ORDER, "FUA(%d): loff=%"PRId64", "
|
|
"data_len=%"PRId64, fua, (uint64_t)loff,
|
|
data_len);
|
|
}
|
|
break;
|
|
}
|
|
|
|
switch (opcode) {
|
|
case READ_6:
|
|
case READ_10:
|
|
case READ_12:
|
|
case READ_16:
|
|
exec_read(vcmd, loff);
|
|
break;
|
|
case WRITE_6:
|
|
case WRITE_10:
|
|
case WRITE_12:
|
|
case WRITE_16:
|
|
if (!dev->rd_only_flag) {
|
|
struct vdisk_tgt_dev *tgt_dev;
|
|
|
|
tgt_dev = find_tgt_dev(dev, cmd->sess_h);
|
|
if (tgt_dev == NULL) {
|
|
PRINT_ERROR("Session %"PRIx64" not found",
|
|
cmd->sess_h);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
goto out;
|
|
}
|
|
|
|
exec_write(vcmd, loff);
|
|
/* O_DSYNC flag is used for WT devices */
|
|
if (fua)
|
|
exec_fsync(vcmd);
|
|
} else {
|
|
PRINT_WARNING("Attempt to write to read-only "
|
|
"device %s", dev->name);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_data_protect));
|
|
}
|
|
break;
|
|
case WRITE_VERIFY:
|
|
case WRITE_VERIFY_12:
|
|
case WRITE_VERIFY_16:
|
|
if (!dev->rd_only_flag) {
|
|
struct vdisk_tgt_dev *tgt_dev;
|
|
|
|
tgt_dev = find_tgt_dev(dev, cmd->sess_h);
|
|
if (tgt_dev == NULL) {
|
|
PRINT_ERROR("Session %"PRIx64" not found",
|
|
cmd->sess_h);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
goto out;
|
|
}
|
|
|
|
exec_write(vcmd, loff);
|
|
/* O_DSYNC flag is used for WT devices */
|
|
if (reply->status == 0)
|
|
exec_verify(vcmd, loff);
|
|
else if (fua)
|
|
exec_fsync(vcmd);
|
|
} else {
|
|
PRINT_WARNING("Attempt to write to read-only "
|
|
"device %s", dev->name);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_data_protect));
|
|
}
|
|
break;
|
|
case SYNCHRONIZE_CACHE:
|
|
case SYNCHRONIZE_CACHE_16:
|
|
{
|
|
int immed = cdb[1] & 0x2;
|
|
if (data_len == 0)
|
|
data_len = dev->file_size -
|
|
((loff_t)lba_start << dev->block_shift);
|
|
TRACE(TRACE_ORDER, "SYNCHRONIZE_CACHE: "
|
|
"loff=%"PRId64", data_len=%"PRId64", immed=%d",
|
|
(uint64_t)loff, data_len, immed);
|
|
if (immed) {
|
|
/* ToDo: backgroung exec */
|
|
exec_fsync(vcmd);
|
|
break;
|
|
} else {
|
|
exec_fsync(vcmd);
|
|
break;
|
|
}
|
|
}
|
|
case VERIFY:
|
|
case VERIFY_12:
|
|
case VERIFY_16:
|
|
exec_verify(vcmd, loff);
|
|
break;
|
|
case MODE_SENSE:
|
|
case MODE_SENSE_10:
|
|
exec_mode_sense(vcmd);
|
|
break;
|
|
case MODE_SELECT:
|
|
case MODE_SELECT_10:
|
|
exec_mode_select(vcmd);
|
|
break;
|
|
case ALLOW_MEDIUM_REMOVAL:
|
|
exec_prevent_allow_medium_removal(vcmd);
|
|
break;
|
|
case READ_TOC:
|
|
exec_read_toc(vcmd);
|
|
break;
|
|
case START_STOP:
|
|
exec_fsync(vcmd/*, 0, dev->file_size*/);
|
|
break;
|
|
case RESERVE:
|
|
case RESERVE_10:
|
|
case RELEASE:
|
|
case RELEASE_10:
|
|
case TEST_UNIT_READY:
|
|
break;
|
|
case INQUIRY:
|
|
exec_inquiry(vcmd);
|
|
break;
|
|
case REQUEST_SENSE:
|
|
exec_request_sense(vcmd);
|
|
break;
|
|
case READ_CAPACITY:
|
|
exec_read_capacity(vcmd);
|
|
break;
|
|
case SERVICE_ACTION_IN:
|
|
if ((cmd->cdb[1] & 0x1f) == SAI_READ_CAPACITY_16)
|
|
exec_read_capacity16(vcmd);
|
|
else {
|
|
TRACE_DBG("Invalid service action %d for SERVICE "
|
|
"ACTION IN", cmd->cdb[1] & 0x1f);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
}
|
|
break;
|
|
case REPORT_LUNS:
|
|
default:
|
|
TRACE_DBG("Invalid opcode %d", opcode);
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(scst_sense_invalid_opcode));
|
|
break;
|
|
}
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return res;
|
|
}
|
|
|
|
static int do_alloc_mem(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct scst_user_get_cmd *cmd = vcmd->cmd;
|
|
struct scst_user_reply_cmd *reply = vcmd->reply;
|
|
int res = 0;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
TRACE_MEM("Alloc mem (cmd %d, sess_h %"PRIx64", cdb_len %d, "
|
|
"alloc_len %d, queue_type %d, data_direction %d)",
|
|
cmd->cmd_h, cmd->alloc_cmd.sess_h,
|
|
cmd->alloc_cmd.cdb_len, cmd->alloc_cmd.alloc_len,
|
|
cmd->alloc_cmd.queue_type, cmd->alloc_cmd.data_direction);
|
|
|
|
TRACE_BUFF_FLAG(TRACE_MEMORY, "CDB", cmd->alloc_cmd.cdb,
|
|
cmd->alloc_cmd.cdb_len);
|
|
|
|
memset(reply, 0, sizeof(*reply));
|
|
reply->cmd_h = cmd->cmd_h;
|
|
reply->subcode = cmd->subcode;
|
|
#ifdef DEBUG_NOMEM
|
|
if ((random() % 100) == 75)
|
|
reply->alloc_reply.pbuf = 0;
|
|
else
|
|
#endif
|
|
reply->alloc_reply.pbuf = (unsigned long)vcmd->dev->alloc_fn(
|
|
cmd->alloc_cmd.alloc_len);
|
|
TRACE_MEM("Buf %"PRIx64" alloced, len %d", reply->alloc_reply.pbuf,
|
|
cmd->alloc_cmd.alloc_len);
|
|
if (reply->alloc_reply.pbuf == 0) {
|
|
TRACE(TRACE_OUT_OF_MEM, "Unable to allocate buffer (len %d)",
|
|
cmd->alloc_cmd.alloc_len);
|
|
}
|
|
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
static int do_cached_mem_free(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct scst_user_get_cmd *cmd = vcmd->cmd;
|
|
struct scst_user_reply_cmd *reply = vcmd->reply;
|
|
int res = 0;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
TRACE_MEM("Cached mem free (cmd %x, buf %"PRIx64")", cmd->cmd_h,
|
|
cmd->on_cached_mem_free.pbuf);
|
|
|
|
free((void *)(unsigned long)cmd->on_cached_mem_free.pbuf);
|
|
|
|
memset(reply, 0, sizeof(*reply));
|
|
reply->cmd_h = cmd->cmd_h;
|
|
reply->subcode = cmd->subcode;
|
|
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
static int do_on_free_cmd(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct scst_user_get_cmd *cmd = vcmd->cmd;
|
|
struct scst_user_reply_cmd *reply = vcmd->reply;
|
|
int res = 0;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
TRACE_DBG("On free cmd (cmd %d, resp_data_len %d, aborted %d, "
|
|
"status %d, delivery_status %d)", cmd->cmd_h,
|
|
cmd->on_free_cmd.resp_data_len, cmd->on_free_cmd.aborted,
|
|
cmd->on_free_cmd.status, cmd->on_free_cmd.delivery_status);
|
|
|
|
TRACE_MEM("On free cmd (cmd %d, buf %"PRIx64", buffer_cached %d)",
|
|
cmd->cmd_h, cmd->on_free_cmd.pbuf,
|
|
cmd->on_free_cmd.buffer_cached);
|
|
|
|
if (!cmd->on_free_cmd.buffer_cached && (cmd->on_free_cmd.pbuf != 0)) {
|
|
TRACE_MEM("Freeing buf %"PRIx64, cmd->on_free_cmd.pbuf);
|
|
free((void *)(unsigned long)cmd->on_free_cmd.pbuf);
|
|
}
|
|
|
|
memset(reply, 0, sizeof(*reply));
|
|
reply->cmd_h = cmd->cmd_h;
|
|
reply->subcode = cmd->subcode;
|
|
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
static int do_tm(struct vdisk_cmd *vcmd, int done)
|
|
{
|
|
struct scst_user_get_cmd *cmd = vcmd->cmd;
|
|
struct scst_user_reply_cmd *reply = vcmd->reply;
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
int res = 0;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
if (cmd->tm_cmd.fn <= SCST_TARGET_RESET)
|
|
TRACE(TRACE_MGMT, "%s TM fn %d (sess_h %"PRIx64", "
|
|
"cmd_h_to_abort %d)", done ? "Done" : "Received",
|
|
cmd->tm_cmd.fn, cmd->tm_cmd.sess_h,
|
|
cmd->tm_cmd.cmd_h_to_abort);
|
|
else
|
|
TRACE_MGMT_DBG("%s TM fn %d (sess_h %"PRIx64", "
|
|
"cmd_h_to_abort %d)", done ? "Done" : "Received",
|
|
cmd->tm_cmd.fn, cmd->tm_cmd.sess_h,
|
|
cmd->tm_cmd.cmd_h_to_abort);
|
|
|
|
if (done) {
|
|
switch (cmd->tm_cmd.fn) {
|
|
case SCST_LUN_RESET:
|
|
case SCST_TARGET_RESET:
|
|
case SCST_PR_ABORT_ALL:
|
|
pthread_mutex_lock(&dev->dev_mutex);
|
|
dev->prevent_allow_medium_removal = 0;
|
|
pthread_mutex_unlock(&dev->dev_mutex);
|
|
break;
|
|
}
|
|
}
|
|
|
|
memset(reply, 0, sizeof(*reply));
|
|
reply->cmd_h = cmd->cmd_h;
|
|
reply->subcode = cmd->subcode;
|
|
reply->result = 0;
|
|
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
static int do_sess(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct scst_user_get_cmd *cmd = vcmd->cmd;
|
|
struct scst_user_reply_cmd *reply = vcmd->reply;
|
|
int res = 0;
|
|
struct vdisk_tgt_dev *tgt_dev;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
/*
|
|
* We are guaranteed to have one and only one command at this point,
|
|
* which is ATTACH_SESS/DETACH_SESS, so no protection is needed
|
|
*/
|
|
|
|
tgt_dev = find_tgt_dev(vcmd->dev, cmd->sess.sess_h);
|
|
|
|
if (cmd->subcode == SCST_USER_ATTACH_SESS) {
|
|
if (tgt_dev != NULL) {
|
|
PRINT_ERROR("Session %"PRIx64" already exists)",
|
|
cmd->sess.sess_h);
|
|
res = EEXIST;
|
|
goto reply;
|
|
}
|
|
|
|
tgt_dev = find_empty_tgt_dev(vcmd->dev);
|
|
if (tgt_dev == NULL) {
|
|
PRINT_ERROR("Too many initiators, session %"PRIx64
|
|
" refused)", cmd->sess.sess_h);
|
|
res = ENOMEM;
|
|
goto reply;
|
|
}
|
|
|
|
tgt_dev->sess_h = cmd->sess.sess_h;
|
|
|
|
PRINT_INFO("Session from initiator %s (target %s) attached "
|
|
"(LUN %"PRIx64", threads_num %d, rd_only %d, sess_h "
|
|
"%"PRIx64")", cmd->sess.initiator_name,
|
|
cmd->sess.target_name, cmd->sess.lun,
|
|
cmd->sess.threads_num, cmd->sess.rd_only,
|
|
cmd->sess.sess_h);
|
|
} else {
|
|
if (tgt_dev == NULL) {
|
|
PRINT_ERROR("Session %"PRIx64" not found)",
|
|
cmd->sess.sess_h);
|
|
res = ESRCH;
|
|
goto reply;
|
|
}
|
|
tgt_dev->sess_h = 0;
|
|
PRINT_INFO("Session detached (sess_h %"PRIx64")",
|
|
cmd->sess.sess_h);
|
|
}
|
|
|
|
reply:
|
|
memset(reply, 0, sizeof(*reply));
|
|
reply->cmd_h = cmd->cmd_h;
|
|
reply->subcode = cmd->subcode;
|
|
reply->result = res;
|
|
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
static int open_dev_fd(struct vdisk_dev *dev)
|
|
{
|
|
int res;
|
|
int open_flags = O_LARGEFILE;
|
|
|
|
if (dev->rd_only_flag)
|
|
open_flags |= O_RDONLY;
|
|
else
|
|
open_flags |= O_RDWR;
|
|
if (dev->o_direct_flag)
|
|
open_flags |= O_DIRECT;
|
|
if (dev->wt_flag)
|
|
open_flags |= O_DSYNC;
|
|
|
|
TRACE_DBG("Opening file %s, flags 0x%x", dev->file_name, open_flags);
|
|
res = open(dev->file_name, open_flags);
|
|
|
|
return res;
|
|
}
|
|
|
|
void *main_loop(void *arg)
|
|
{
|
|
int res = 0;
|
|
struct vdisk_dev *dev = (struct vdisk_dev *)arg;
|
|
struct scst_user_get_cmd cmd;
|
|
struct scst_user_reply_cmd reply;
|
|
struct vdisk_cmd vcmd = { -1, &cmd, dev, &reply, {0}};
|
|
int scst_usr_fd = dev->scst_usr_fd;
|
|
struct pollfd pl;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
vcmd.fd = open_dev_fd(dev);
|
|
if (vcmd.fd < 0) {
|
|
res = -errno;
|
|
PRINT_ERROR("Unable to open file %s (%s)", dev->file_name,
|
|
strerror(-res));
|
|
goto out;
|
|
}
|
|
|
|
memset(&pl, 0, sizeof(pl));
|
|
pl.fd = scst_usr_fd;
|
|
pl.events = POLLIN;
|
|
|
|
cmd.preply = 0;
|
|
|
|
while(1) {
|
|
#ifdef DEBUG_TM_IGNORE_ALL
|
|
if (dev->debug_tm_ignore && (random() % 50000) == 55) {
|
|
TRACE_MGMT_DBG("%s", "Ignore ALL");
|
|
dev->debug_tm_ignore_all = 1;
|
|
}
|
|
if (dev->debug_tm_ignore_all) {
|
|
/* Go Astral */
|
|
while(1) {
|
|
sleep(60);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
res = ioctl(scst_usr_fd, SCST_USER_REPLY_AND_GET_CMD, &cmd);
|
|
if (res != 0) {
|
|
res = errno;
|
|
switch(res) {
|
|
case ESRCH:
|
|
case EBUSY:
|
|
TRACE_MGMT_DBG("SCST_USER_REPLY_AND_GET_CMD returned "
|
|
"%d (%s)", res, strerror(res));
|
|
cmd.preply = 0;
|
|
case EINTR:
|
|
continue;
|
|
case EAGAIN:
|
|
TRACE_DBG("SCST_USER_REPLY_AND_GET_CMD returned "
|
|
"EAGAIN (%d)", res);
|
|
cmd.preply = 0;
|
|
if (dev->non_blocking)
|
|
break;
|
|
else
|
|
continue;
|
|
default:
|
|
PRINT_ERROR("SCST_USER_REPLY_AND_GET_CMD failed: "
|
|
"%s (%d)", strerror(res), res);
|
|
#if 1
|
|
cmd.preply = 0;
|
|
continue;
|
|
#else
|
|
goto out_close;
|
|
#endif
|
|
}
|
|
again_poll:
|
|
res = poll(&pl, 1, 2000);
|
|
if (res > 0)
|
|
continue;
|
|
else if (res == 0)
|
|
goto again_poll;
|
|
else {
|
|
res = errno;
|
|
switch(res) {
|
|
case ESRCH:
|
|
case EBUSY:
|
|
case EAGAIN:
|
|
TRACE_MGMT_DBG("poll() returned %d "
|
|
"(%s)", res, strerror(res));
|
|
case EINTR:
|
|
goto again_poll;
|
|
default:
|
|
PRINT_ERROR("poll() failed: %s", strerror(res));
|
|
#if 1
|
|
goto again_poll;
|
|
#else
|
|
goto out_close;
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
TRACE_BUFFER("Received cmd", &cmd, sizeof(cmd));
|
|
|
|
switch(cmd.subcode) {
|
|
case SCST_USER_EXEC:
|
|
if (cmd.exec_cmd.data_direction & SCST_DATA_WRITE) {
|
|
TRACE_BUFFER("Received cmd data",
|
|
(void *)(unsigned long)cmd.exec_cmd.pbuf,
|
|
cmd.exec_cmd.bufflen);
|
|
}
|
|
res = do_exec(&vcmd);
|
|
#ifdef DEBUG_TM_IGNORE
|
|
if (res == 150) {
|
|
cmd.preply = 0;
|
|
continue;
|
|
}
|
|
#endif
|
|
if (reply.exec_reply.resp_data_len != 0) {
|
|
TRACE_BUFFER("Reply data",
|
|
(void *)(unsigned long)reply.exec_reply.pbuf,
|
|
reply.exec_reply.resp_data_len);
|
|
}
|
|
break;
|
|
|
|
case SCST_USER_ALLOC_MEM:
|
|
res = do_alloc_mem(&vcmd);
|
|
break;
|
|
|
|
case SCST_USER_PARSE:
|
|
res = do_parse(&vcmd);
|
|
break;
|
|
|
|
case SCST_USER_ON_CACHED_MEM_FREE:
|
|
res = do_cached_mem_free(&vcmd);
|
|
break;
|
|
|
|
case SCST_USER_ON_FREE_CMD:
|
|
res = do_on_free_cmd(&vcmd);
|
|
break;
|
|
|
|
case SCST_USER_TASK_MGMT_RECEIVED:
|
|
res = do_tm(&vcmd, 0);
|
|
break;
|
|
|
|
case SCST_USER_TASK_MGMT_DONE:
|
|
res = do_tm(&vcmd, 1);
|
|
#if DEBUG_TM_FN_IGNORE
|
|
if (dev->debug_tm_ignore) {
|
|
sleep(15);
|
|
}
|
|
#endif
|
|
break;
|
|
|
|
case SCST_USER_ATTACH_SESS:
|
|
case SCST_USER_DETACH_SESS:
|
|
res = do_sess(&vcmd);
|
|
break;
|
|
|
|
default:
|
|
PRINT_ERROR("Unknown or wrong cmd subcode %x",
|
|
cmd.subcode);
|
|
goto out_close;
|
|
}
|
|
|
|
if (res != 0)
|
|
goto out_close;
|
|
|
|
cmd.preply = (unsigned long)&reply;
|
|
TRACE_BUFFER("Sending reply", &reply, sizeof(reply));
|
|
}
|
|
|
|
out_close:
|
|
close(vcmd.fd);
|
|
|
|
out:
|
|
PRINT_INFO("Thread %d exiting (res=%d)", gettid(), res);
|
|
|
|
TRACE_EXIT_RES(res);
|
|
return (void *)(long)res;
|
|
}
|
|
|
|
uint64_t gen_dev_id_num(const struct vdisk_dev *dev)
|
|
{
|
|
uint32_t dev_id_num;
|
|
|
|
dev_id_num = crc32buf(dev->name, strlen(dev->name)+1);
|
|
|
|
return ((uint64_t)vdisk_ID << 32) | dev_id_num;
|
|
}
|
|
|
|
static void exec_inquiry(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
int resp_len = 0;
|
|
int length = cmd->bufflen;
|
|
unsigned int i;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
uint8_t buf[INQ_BUF_SZ];
|
|
|
|
TRACE_ENTRY();
|
|
|
|
if (cmd->cdb[1] & CMDDT) {
|
|
TRACE_DBG("%s", "INQUIRY: CMDDT is unsupported");
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
buf[0] = dev->type; /* type dev */
|
|
if (buf[0] == TYPE_ROM)
|
|
buf[1] = 0x80; /* removable */
|
|
/* Vital Product */
|
|
if (cmd->cdb[1] & EVPD) {
|
|
uint64_t dev_id_num = gen_dev_id_num(dev);
|
|
|
|
if (0 == cmd->cdb[2]) { /* supported vital product data pages */
|
|
buf[3] = 3;
|
|
buf[4] = 0x0; /* this page */
|
|
buf[5] = 0x80; /* unit serial number */
|
|
buf[6] = 0x83; /* device identification */
|
|
resp_len = buf[3] + 4;
|
|
} else if (0x80 == cmd->cdb[2]) { /* unit serial number */
|
|
int usn_len = strlen(dev->usn);
|
|
buf[1] = 0x80;
|
|
buf[3] = usn_len;
|
|
strncpy((char *)&buf[4], dev->usn, usn_len);
|
|
resp_len = buf[3] + 4;
|
|
} else if (0x83 == cmd->cdb[2]) { /* device identification */
|
|
int num = 4;
|
|
char *t10_id = (char *)&buf[num + 12];
|
|
|
|
buf[1] = 0x83;
|
|
/* Two identification descriptors: */
|
|
/* T10 vendor identifier field format (faked) */
|
|
buf[num + 0] = 0x2; /* ASCII */
|
|
buf[num + 1] = 0x1; /* Vendor ID */
|
|
memcpy(&buf[num + 4], VENDOR, 8);
|
|
snprintf(t10_id, sizeof(buf) - num - 12,
|
|
"%"PRIx64"-%s", dev_id_num, dev->name);
|
|
i = strlen(t10_id) + 1;
|
|
TRACE_DBG("t10_dev_id %s", t10_id);
|
|
buf[num + 3] = 8 + i;
|
|
num += buf[num + 3];
|
|
|
|
#if 0 /* This isn't required and can be misleading, so let's disable it */
|
|
num += 4;
|
|
|
|
/* NAA IEEE registered identifier (faked) */
|
|
buf[num] = 0x1; /* binary */
|
|
buf[num + 1] = 0x3;
|
|
buf[num + 2] = 0x0;
|
|
buf[num + 3] = 0x8;
|
|
buf[num + 4] = 0x51; /* ieee company id=0x123456 (faked) */
|
|
buf[num + 5] = 0x23;
|
|
buf[num + 6] = 0x45;
|
|
buf[num + 7] = 0x60;
|
|
buf[num + 8] = (dev_id_num >> 24);
|
|
buf[num + 9] = (dev_id_num >> 16) & 0xff;
|
|
buf[num + 10] = (dev_id_num >> 8) & 0xff;
|
|
buf[num + 11] = dev_id_num & 0xff;
|
|
num = num + 12 - 4;
|
|
#endif
|
|
|
|
resp_len = num;
|
|
buf[2] = (resp_len >> 8) & 0xFF;
|
|
buf[3] = resp_len & 0xFF;
|
|
resp_len += 4;
|
|
} else {
|
|
TRACE_DBG("INQUIRY: Unsupported EVPD page %x",
|
|
cmd->cdb[2]);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
} else {
|
|
int len;
|
|
|
|
if (cmd->cdb[2] != 0) {
|
|
TRACE_DBG("INQUIRY: Unsupported page %x", cmd->cdb[2]);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
|
|
buf[2] = 6; /* Device complies to SPC-4 */
|
|
buf[3] = 0x12; /* HiSup + data in format specified in SPC */
|
|
buf[4] = 31;/* n - 4 = 35 - 4 = 31 for full 36 byte data */
|
|
buf[6] = 1; /* MultiP 1 */
|
|
buf[7] = 2; /* CMDQUE 1, BQue 0 => commands queuing supported */
|
|
|
|
/* 8 byte ASCII Vendor Identification of the target - left aligned */
|
|
memcpy(&buf[8], VENDOR, 8);
|
|
|
|
/* 16 byte ASCII Product Identification of the target - left aligned */
|
|
memset(&buf[16], ' ', 16);
|
|
len = min(strlen(dev->name), (size_t)16);
|
|
memcpy(&buf[16], dev->name, len);
|
|
|
|
/* 4 byte ASCII Product Revision Level of the target - left aligned */
|
|
memcpy(&buf[32], FIO_REV, 4);
|
|
resp_len = buf[4] + 5;
|
|
}
|
|
|
|
sBUG_ON(resp_len >= (int)sizeof(buf));
|
|
if (length > resp_len)
|
|
length = resp_len;
|
|
|
|
memcpy(address, buf, length);
|
|
|
|
if (length < reply->resp_data_len)
|
|
reply->resp_data_len = length;
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static void exec_request_sense(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
int length = cmd->bufflen, l;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
uint8_t b[SCST_STANDARD_SENSE_LEN];
|
|
|
|
TRACE_ENTRY();
|
|
|
|
l = set_sense(b, sizeof(b), SCST_LOAD_SENSE(scst_sense_no_sense));
|
|
|
|
length = min(l, length);
|
|
|
|
memcpy(address, b, length);
|
|
|
|
if (length < reply->resp_data_len)
|
|
reply->resp_data_len = length;
|
|
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* <<Following mode pages info copied from ST318451LW with some corrections>>
|
|
*
|
|
* ToDo: revise them
|
|
*/
|
|
|
|
static int err_recov_pg(unsigned char *p, int pcontrol)
|
|
{ /* Read-Write Error Recovery page for mode_sense */
|
|
const unsigned char err_recov_pg[] = {0x1, 0xa, 0xc0, 11, 240, 0, 0, 0,
|
|
5, 0, 0xff, 0xff};
|
|
|
|
memcpy(p, err_recov_pg, sizeof(err_recov_pg));
|
|
if (1 == pcontrol)
|
|
memset(p + 2, 0, sizeof(err_recov_pg) - 2);
|
|
return sizeof(err_recov_pg);
|
|
}
|
|
|
|
static int disconnect_pg(unsigned char *p, int pcontrol)
|
|
{ /* Disconnect-Reconnect page for mode_sense */
|
|
const unsigned char disconnect_pg[] = {0x2, 0xe, 128, 128, 0, 10, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0};
|
|
|
|
memcpy(p, disconnect_pg, sizeof(disconnect_pg));
|
|
if (1 == pcontrol)
|
|
memset(p + 2, 0, sizeof(disconnect_pg) - 2);
|
|
return sizeof(disconnect_pg);
|
|
}
|
|
|
|
static int rigid_geo_pg(unsigned char *p, int pcontrol,
|
|
struct vdisk_dev *dev)
|
|
{
|
|
unsigned char geo_m_pg[] = {0x04, 0x16, 0, 0, 0, DEF_HEADS, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
|
0x3a, 0x98/* 15K RPM */, 0, 0};
|
|
int32_t ncyl, n;
|
|
|
|
memcpy(p, geo_m_pg, sizeof(geo_m_pg));
|
|
ncyl = dev->nblocks / (DEF_HEADS * DEF_SECTORS);
|
|
if ((dev->nblocks % (DEF_HEADS * DEF_SECTORS)) != 0)
|
|
ncyl++;
|
|
memcpy(&n, p + 2, sizeof(uint32_t));
|
|
n = n | (htonl(ncyl) >> 8);
|
|
memcpy(p + 2, &n, sizeof(uint32_t));
|
|
if (1 == pcontrol)
|
|
memset(p + 2, 0, sizeof(geo_m_pg) - 2);
|
|
return sizeof(geo_m_pg);
|
|
}
|
|
|
|
static int format_pg(unsigned char *p, int pcontrol,
|
|
struct vdisk_dev *dev)
|
|
{ /* Format device page for mode_sense */
|
|
const unsigned char format_pg[] = {0x3, 0x16, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0, 0, 0, 0,
|
|
0, 0, 0, 0, 0x40, 0, 0, 0};
|
|
|
|
memcpy(p, format_pg, sizeof(format_pg));
|
|
p[10] = (DEF_SECTORS >> 8) & 0xff;
|
|
p[11] = DEF_SECTORS & 0xff;
|
|
p[12] = (dev->block_size >> 8) & 0xff;
|
|
p[13] = dev->block_size & 0xff;
|
|
if (1 == pcontrol)
|
|
memset(p + 2, 0, sizeof(format_pg) - 2);
|
|
return sizeof(format_pg);
|
|
}
|
|
|
|
static int caching_pg(unsigned char *p, int pcontrol,
|
|
struct vdisk_dev *dev)
|
|
{ /* Caching page for mode_sense */
|
|
const unsigned char caching_pg[] = {0x8, 18, 0x10, 0, 0xff, 0xff, 0, 0,
|
|
0xff, 0xff, 0xff, 0xff, 0x80, 0x14, 0, 0, 0, 0, 0, 0};
|
|
|
|
memcpy(p, caching_pg, sizeof(caching_pg));
|
|
p[2] |= !(dev->wt_flag) ? WCE : 0;
|
|
if (1 == pcontrol)
|
|
memset(p + 2, 0, sizeof(caching_pg) - 2);
|
|
return sizeof(caching_pg);
|
|
}
|
|
|
|
static int ctrl_m_pg(unsigned char *p, int pcontrol,
|
|
struct vdisk_dev *dev)
|
|
{ /* Control mode page for mode_sense */
|
|
const unsigned char ctrl_m_pg[] = {0xa, 0xa, 0x20, 0, 0, 0x40, 0, 0,
|
|
0, 0, 0x2, 0x4b};
|
|
|
|
memcpy(p, ctrl_m_pg, sizeof(ctrl_m_pg));
|
|
if (!dev->wt_flag && !dev->nv_cache)
|
|
p[3] |= 0x10; /* Enable unrestricted reordering */
|
|
if (1 == pcontrol)
|
|
memset(p + 2, 0, sizeof(ctrl_m_pg) - 2);
|
|
return sizeof(ctrl_m_pg);
|
|
}
|
|
|
|
static int iec_m_pg(unsigned char *p, int pcontrol)
|
|
{ /* Informational Exceptions control mode page for mode_sense */
|
|
const unsigned char iec_m_pg[] = {0x1c, 0xa, 0x08, 0, 0, 0, 0, 0,
|
|
0, 0, 0x0, 0x0};
|
|
memcpy(p, iec_m_pg, sizeof(iec_m_pg));
|
|
if (1 == pcontrol)
|
|
memset(p + 2, 0, sizeof(iec_m_pg) - 2);
|
|
return sizeof(iec_m_pg);
|
|
}
|
|
|
|
static void exec_mode_sense(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
int length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
uint8_t buf[MSENSE_BUF_SZ];
|
|
int blocksize;
|
|
uint64_t nblocks;
|
|
unsigned char dbd, type;
|
|
int pcontrol, pcode, subpcode;
|
|
unsigned char dev_spec;
|
|
int msense_6, offset = 0, len;
|
|
unsigned char *bp;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
blocksize = dev->block_size;
|
|
nblocks = dev->nblocks;
|
|
|
|
type = dev->type; /* type dev */
|
|
dbd = cmd->cdb[1] & DBD;
|
|
pcontrol = (cmd->cdb[2] & 0xc0) >> 6;
|
|
pcode = cmd->cdb[2] & 0x3f;
|
|
subpcode = cmd->cdb[3];
|
|
msense_6 = (MODE_SENSE == cmd->cdb[0]);
|
|
dev_spec = (dev->rd_only_flag ? WP : 0) | DPOFUA;
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
|
|
if (0x3 == pcontrol) {
|
|
TRACE_DBG("%s", "MODE SENSE: Saving values not supported");
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_saving_params_unsup));
|
|
goto out;
|
|
}
|
|
|
|
if (msense_6) {
|
|
buf[1] = type;
|
|
buf[2] = dev_spec;
|
|
offset = 4;
|
|
} else {
|
|
buf[2] = type;
|
|
buf[3] = dev_spec;
|
|
offset = 8;
|
|
}
|
|
|
|
if (0 != subpcode) { /* TODO: Control Extension page */
|
|
TRACE_DBG("%s", "MODE SENSE: Only subpage 0 is supported");
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
|
|
if (!dbd) {
|
|
/* Create block descriptor */
|
|
buf[offset - 1] = 0x08; /* block descriptor length */
|
|
if (nblocks >> 32) {
|
|
buf[offset + 0] = 0xFF;
|
|
buf[offset + 1] = 0xFF;
|
|
buf[offset + 2] = 0xFF;
|
|
buf[offset + 3] = 0xFF;
|
|
} else {
|
|
buf[offset + 0] = (nblocks >> (BYTE * 3)) & 0xFF;/* num blks */
|
|
buf[offset + 1] = (nblocks >> (BYTE * 2)) & 0xFF;
|
|
buf[offset + 2] = (nblocks >> (BYTE * 1)) & 0xFF;
|
|
buf[offset + 3] = (nblocks >> (BYTE * 0)) & 0xFF;
|
|
}
|
|
buf[offset + 4] = 0; /* density code */
|
|
buf[offset + 5] = (blocksize >> (BYTE * 2)) & 0xFF;/* blklen */
|
|
buf[offset + 6] = (blocksize >> (BYTE * 1)) & 0xFF;
|
|
buf[offset + 7] = (blocksize >> (BYTE * 0)) & 0xFF;
|
|
|
|
offset += 8; /* increment offset */
|
|
}
|
|
|
|
bp = buf + offset;
|
|
|
|
switch (pcode) {
|
|
case 0x1: /* Read-Write error recovery page, direct access */
|
|
len = err_recov_pg(bp, pcontrol);
|
|
break;
|
|
case 0x2: /* Disconnect-Reconnect page, all devices */
|
|
len = disconnect_pg(bp, pcontrol);
|
|
break;
|
|
case 0x3: /* Format device page, direct access */
|
|
len = format_pg(bp, pcontrol, dev);
|
|
break;
|
|
case 0x4: /* Rigid disk geometry */
|
|
len = rigid_geo_pg(bp, pcontrol, dev);
|
|
break;
|
|
case 0x8: /* Caching page, direct access */
|
|
len = caching_pg(bp, pcontrol, dev);
|
|
break;
|
|
case 0xa: /* Control Mode page, all devices */
|
|
len = ctrl_m_pg(bp, pcontrol, dev);
|
|
break;
|
|
case 0x1c: /* Informational Exceptions Mode page, all devices */
|
|
len = iec_m_pg(bp, pcontrol);
|
|
break;
|
|
case 0x3f: /* Read all Mode pages */
|
|
len = err_recov_pg(bp, pcontrol);
|
|
len += disconnect_pg(bp + len, pcontrol);
|
|
len += format_pg(bp + len, pcontrol, dev);
|
|
len += caching_pg(bp + len, pcontrol, dev);
|
|
len += ctrl_m_pg(bp + len, pcontrol, dev);
|
|
len += iec_m_pg(bp + len, pcontrol);
|
|
len += rigid_geo_pg(bp + len, pcontrol, dev);
|
|
break;
|
|
default:
|
|
TRACE_DBG("MODE SENSE: Unsupported page %x", pcode);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
|
|
offset += len;
|
|
|
|
if (msense_6)
|
|
buf[0] = offset - 1;
|
|
else {
|
|
buf[0] = ((offset - 2) >> 8) & 0xff;
|
|
buf[1] = (offset - 2) & 0xff;
|
|
}
|
|
|
|
sBUG_ON(offset >= (int)sizeof(buf));
|
|
if (offset > length)
|
|
offset = length;
|
|
|
|
memcpy(address, buf, offset);
|
|
|
|
if (offset < reply->resp_data_len)
|
|
reply->resp_data_len = offset;
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static int set_wt(struct vdisk_dev *dev, int wt)
|
|
{
|
|
int res = 0;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
if ((dev->wt_flag == wt) || dev->nullio)
|
|
goto out;
|
|
|
|
pthread_mutex_lock(&dev->dev_mutex);
|
|
dev->wt_flag = wt;
|
|
pthread_mutex_unlock(&dev->dev_mutex);
|
|
|
|
out:
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
static void exec_mode_select(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
int length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
int mselect_6, offset;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
mselect_6 = (MODE_SELECT == cmd->cdb[0]);
|
|
|
|
if (!(cmd->cdb[1] & PF) || (cmd->cdb[1] & SP)) {
|
|
PRINT_ERROR("MODE SELECT: PF and/or SP are wrongly set "
|
|
"(cdb[1]=%x)", cmd->cdb[1]);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
|
|
if (mselect_6) {
|
|
offset = 4;
|
|
} else {
|
|
offset = 8;
|
|
}
|
|
|
|
if (address[offset - 1] == 8) {
|
|
offset += 8;
|
|
} else if (address[offset - 1] != 0) {
|
|
PRINT_ERROR("%s", "MODE SELECT: Wrong parameters list "
|
|
"length");
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_parm_list));
|
|
goto out;
|
|
}
|
|
|
|
while (length > offset + 2) {
|
|
if (address[offset] & PS) {
|
|
PRINT_ERROR("%s", "MODE SELECT: Illegal PS bit");
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(
|
|
scst_sense_invalid_field_in_parm_list));
|
|
goto out;
|
|
}
|
|
if ((address[offset] & 0x3f) == 0x8) { /* Caching page */
|
|
if (address[offset + 1] != 18) {
|
|
PRINT_ERROR("%s", "MODE SELECT: Invalid "
|
|
"caching page request");
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(
|
|
scst_sense_invalid_field_in_parm_list));
|
|
goto out;
|
|
}
|
|
if (set_wt(dev,
|
|
(address[offset + 2] & WCE) ? 0 : 1) != 0) {
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
goto out;
|
|
}
|
|
break;
|
|
}
|
|
offset += address[offset + 1];
|
|
}
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static void exec_read_capacity(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
int length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
uint32_t blocksize;
|
|
uint64_t nblocks;
|
|
uint8_t buffer[8];
|
|
|
|
TRACE_ENTRY();
|
|
|
|
blocksize = dev->block_size;
|
|
nblocks = dev->nblocks;
|
|
|
|
/* last block on the dev is (nblocks-1) */
|
|
memset(buffer, 0, sizeof(buffer));
|
|
if (nblocks >> 32) {
|
|
buffer[0] = 0xFF;
|
|
buffer[1] = 0xFF;
|
|
buffer[2] = 0xFF;
|
|
buffer[3] = 0xFF;
|
|
} else {
|
|
buffer[0] = ((nblocks - 1) >> (BYTE * 3)) & 0xFF;
|
|
buffer[1] = ((nblocks - 1) >> (BYTE * 2)) & 0xFF;
|
|
buffer[2] = ((nblocks - 1) >> (BYTE * 1)) & 0xFF;
|
|
buffer[3] = ((nblocks - 1) >> (BYTE * 0)) & 0xFF;
|
|
}
|
|
buffer[4] = (blocksize >> (BYTE * 3)) & 0xFF;
|
|
buffer[5] = (blocksize >> (BYTE * 2)) & 0xFF;
|
|
buffer[6] = (blocksize >> (BYTE * 1)) & 0xFF;
|
|
buffer[7] = (blocksize >> (BYTE * 0)) & 0xFF;
|
|
|
|
length = min(length, (int)sizeof(buffer));
|
|
|
|
memcpy(address, buffer, length);
|
|
|
|
if (length < reply->resp_data_len)
|
|
reply->resp_data_len = length;
|
|
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static void exec_read_capacity16(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
int length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
uint32_t blocksize;
|
|
uint64_t nblocks;
|
|
uint8_t buffer[32];
|
|
|
|
TRACE_ENTRY();
|
|
|
|
blocksize = dev->block_size;
|
|
nblocks = dev->nblocks - 1;
|
|
|
|
memset(buffer, 0, sizeof(buffer));
|
|
|
|
buffer[0] = nblocks >> 56;
|
|
buffer[1] = (nblocks >> 48) & 0xFF;
|
|
buffer[2] = (nblocks >> 40) & 0xFF;
|
|
buffer[3] = (nblocks >> 32) & 0xFF;
|
|
buffer[4] = (nblocks >> 24) & 0xFF;
|
|
buffer[5] = (nblocks >> 16) & 0xFF;
|
|
buffer[6] = (nblocks >> 8) & 0xFF;
|
|
buffer[7] = nblocks& 0xFF;
|
|
|
|
buffer[8] = (blocksize >> (BYTE * 3)) & 0xFF;
|
|
buffer[9] = (blocksize >> (BYTE * 2)) & 0xFF;
|
|
buffer[10] = (blocksize >> (BYTE * 1)) & 0xFF;
|
|
buffer[11] = (blocksize >> (BYTE * 0)) & 0xFF;
|
|
|
|
switch (blocksize) {
|
|
case 512:
|
|
buffer[13] = 3;
|
|
break;
|
|
case 1024:
|
|
buffer[13] = 2;
|
|
break;
|
|
case 2048:
|
|
buffer[13] = 1;
|
|
break;
|
|
case 4096:
|
|
default:
|
|
buffer[13] = 0;
|
|
break;
|
|
}
|
|
|
|
length = min(length, (int)sizeof(buffer));
|
|
|
|
memcpy(address, buffer, length);
|
|
|
|
if (length < reply->resp_data_len)
|
|
reply->resp_data_len = length;
|
|
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static void exec_read_toc(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
int32_t off = 0;
|
|
int length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
uint32_t nblocks;
|
|
uint8_t buffer[4+8+8] = { 0x00, 0x0a, 0x01, 0x01, 0x00, 0x14,
|
|
0x01, 0x00, 0x00, 0x00, 0x00, 0x00 };
|
|
|
|
TRACE_ENTRY();
|
|
|
|
if (dev->type != TYPE_ROM) {
|
|
PRINT_ERROR("%s", "READ TOC for non-CDROM device");
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_opcode));
|
|
goto out;
|
|
}
|
|
|
|
if (cmd->cdb[2] & 0x0e/*Format*/) {
|
|
PRINT_ERROR("%s", "READ TOC: invalid requested data format");
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
|
|
if ((cmd->cdb[6] != 0 && (cmd->cdb[2] & 0x01)) ||
|
|
(cmd->cdb[6] > 1 && cmd->cdb[6] != 0xAA)) {
|
|
PRINT_ERROR("READ TOC: invalid requested track number %x",
|
|
cmd->cdb[6]);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_invalid_field_in_cdb));
|
|
goto out;
|
|
}
|
|
|
|
/* ToDo when you have > 8TB ROM device. */
|
|
nblocks = (uint32_t)dev->nblocks;
|
|
|
|
/* Header */
|
|
memset(buffer, 0, sizeof(buffer));
|
|
buffer[2] = 0x01; /* First Track/Session */
|
|
buffer[3] = 0x01; /* Last Track/Session */
|
|
off = 4;
|
|
if (cmd->cdb[6] <= 1)
|
|
{
|
|
/* Fistr TOC Track Descriptor */
|
|
buffer[off+1] = 0x14; /* ADDR 0x10 - Q Sub-channel encodes current position data
|
|
CONTROL 0x04 - Data track, recoreded uninterrupted */
|
|
buffer[off+2] = 0x01; /* Track Number */
|
|
off += 8;
|
|
}
|
|
if (!(cmd->cdb[2] & 0x01))
|
|
{
|
|
/* Lead-out area TOC Track Descriptor */
|
|
buffer[off+1] = 0x14;
|
|
buffer[off+2] = 0xAA; /* Track Number */
|
|
buffer[off+4] = (nblocks >> (BYTE * 3)) & 0xFF; /* Track Start Address */
|
|
buffer[off+5] = (nblocks >> (BYTE * 2)) & 0xFF;
|
|
buffer[off+6] = (nblocks >> (BYTE * 1)) & 0xFF;
|
|
buffer[off+7] = (nblocks >> (BYTE * 0)) & 0xFF;
|
|
off += 8;
|
|
}
|
|
|
|
buffer[1] = off - 2; /* Data Length */
|
|
|
|
if (off > length)
|
|
off = length;
|
|
|
|
memcpy(address, buffer, off);
|
|
|
|
if (off < reply->resp_data_len)
|
|
reply->resp_data_len = off;
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static void exec_prevent_allow_medium_removal(struct vdisk_cmd *vcmd)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
|
|
TRACE_DBG("PERSIST/PREVENT 0x%02x", cmd->cdb[4]);
|
|
|
|
pthread_mutex_lock(&dev->dev_mutex);
|
|
dev->prevent_allow_medium_removal = cmd->cdb[4] & 0x01 ? 1 : 0;
|
|
pthread_mutex_unlock(&dev->dev_mutex);
|
|
|
|
return;
|
|
}
|
|
|
|
static int exec_fsync(struct vdisk_cmd *vcmd)
|
|
{
|
|
int res = 0;
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
|
|
/* Hopefully, the compiler will generate the single comparison */
|
|
if (dev->nv_cache || dev->wt_flag || dev->rd_only_flag ||
|
|
dev->o_direct_flag || dev->nullio)
|
|
goto out;
|
|
|
|
/* ToDo: use sync_file_range() instead */
|
|
fsync(vcmd->fd);
|
|
|
|
out:
|
|
TRACE_EXIT_RES(res);
|
|
return res;
|
|
}
|
|
|
|
static void exec_read(struct vdisk_cmd *vcmd, loff_t loff)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
int length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
int fd = vcmd->fd;
|
|
loff_t err;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
TRACE_DBG("reading off %"PRId64", len %d", loff, length);
|
|
if (dev->nullio)
|
|
err = length;
|
|
else {
|
|
/* SEEK */
|
|
err = lseek64(fd, loff, 0/*SEEK_SET*/);
|
|
if (err != loff) {
|
|
PRINT_ERROR("lseek trouble %"PRId64" != %"PRId64
|
|
" (errno %d)", (uint64_t)err, (uint64_t)loff,
|
|
errno);
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
goto out;
|
|
}
|
|
/* READ */
|
|
err = read(fd, address, length);
|
|
}
|
|
|
|
if ((err < 0) || (err < length)) {
|
|
PRINT_ERROR("read() returned %"PRId64" from %d (errno %d)",
|
|
(uint64_t)err, length, errno);
|
|
if (err == -EAGAIN)
|
|
set_busy(reply);
|
|
else {
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_read_error));
|
|
}
|
|
goto out;
|
|
}
|
|
|
|
reply->resp_data_len = cmd->bufflen;
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static void exec_write(struct vdisk_cmd *vcmd, loff_t loff)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
loff_t err;
|
|
int length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t*)(unsigned long)cmd->pbuf;
|
|
int fd = vcmd->fd;
|
|
|
|
TRACE_ENTRY();
|
|
|
|
restart:
|
|
TRACE_DBG("writing off %"PRId64", len %d", loff, length);
|
|
|
|
if (dev->nullio)
|
|
err = length;
|
|
else {
|
|
/* SEEK */
|
|
err = lseek64(fd, loff, 0/*SEEK_SET*/);
|
|
if (err != loff) {
|
|
PRINT_ERROR("lseek trouble %"PRId64" != %"PRId64
|
|
" (errno %d)", (uint64_t)err, (uint64_t)loff,
|
|
errno);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
goto out;
|
|
}
|
|
|
|
/* WRITE */
|
|
err = write(fd, address, length);
|
|
}
|
|
|
|
if (err < 0) {
|
|
PRINT_ERROR("write() returned %"PRId64" from %d (errno %d, "
|
|
"cmd_h %x)", err, length, errno, vcmd->cmd->cmd_h);
|
|
if (err == -EAGAIN)
|
|
set_busy(reply);
|
|
else {
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_write_error));
|
|
}
|
|
goto out;
|
|
} else if (err < length) {
|
|
/*
|
|
* Probably that's wrong, but sometimes write() returns
|
|
* value less, than requested. Let's restart.
|
|
*/
|
|
TRACE_MGMT_DBG("write() returned %d from %d", (int)err, length);
|
|
if (err == 0) {
|
|
PRINT_INFO("Suspicious: write() returned 0 from "
|
|
"%d", length);
|
|
}
|
|
length -= err;
|
|
goto restart;
|
|
}
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|
|
|
|
static void exec_verify(struct vdisk_cmd *vcmd, loff_t loff)
|
|
{
|
|
struct vdisk_dev *dev = vcmd->dev;
|
|
struct scst_user_scsi_cmd_exec *cmd = &vcmd->cmd->exec_cmd;
|
|
struct scst_user_scsi_cmd_reply_exec *reply = &vcmd->reply->exec_reply;
|
|
loff_t err;
|
|
int64_t length = cmd->bufflen;
|
|
uint8_t *address = (uint8_t *)(unsigned long)cmd->pbuf;
|
|
int compare;
|
|
int fd = vcmd->fd;
|
|
uint8_t mem_verify[128*1024];
|
|
|
|
TRACE_ENTRY();
|
|
|
|
if (exec_fsync(vcmd) != 0)
|
|
goto out;
|
|
|
|
/*
|
|
* Until the cache is cleared prior the verifying, there is not
|
|
* much point in this code. ToDo.
|
|
*
|
|
* Nevertherless, this code is valuable if the data have not read
|
|
* from the file/disk yet.
|
|
*/
|
|
|
|
/* SEEK */
|
|
if (!dev->nullio) {
|
|
err = lseek64(fd, loff, 0/*SEEK_SET*/);
|
|
if (err != loff) {
|
|
PRINT_ERROR("lseek trouble %"PRId64" != %"PRId64
|
|
" (errno %d)", (uint64_t)err,
|
|
(uint64_t)loff, errno);
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
if ((length == 0) && (cmd->data_len != 0)) {
|
|
length = cmd->data_len;
|
|
compare = 0;
|
|
} else
|
|
compare = 1;
|
|
|
|
while (length > 0) {
|
|
int64_t len_mem = (length > (int)sizeof(mem_verify)) ?
|
|
(int)sizeof(mem_verify) : length;
|
|
TRACE_DBG("Verify: length %"PRId64" - len_mem %"PRId64,
|
|
length, len_mem);
|
|
|
|
if (!dev->nullio)
|
|
err = read(fd, (char *)mem_verify, len_mem);
|
|
else
|
|
err = len_mem;
|
|
if ((err < 0) || (err < len_mem)) {
|
|
PRINT_ERROR("read() returned %"PRId64" from %"PRId64" "
|
|
"(errno %d)", (uint64_t)err, len_mem, errno);
|
|
if (err == -EAGAIN)
|
|
set_busy(reply);
|
|
else {
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_read_error));
|
|
}
|
|
goto out;
|
|
}
|
|
if (compare && memcmp(address, mem_verify, len_mem) != 0) {
|
|
TRACE_DBG("Verify: error memcmp length %"PRId64, length);
|
|
set_cmd_error(vcmd,
|
|
SCST_LOAD_SENSE(scst_sense_miscompare_error));
|
|
goto out;
|
|
}
|
|
length -= len_mem;
|
|
address += len_mem;
|
|
}
|
|
|
|
if (length < 0) {
|
|
PRINT_ERROR("Failure: %"PRId64, length);
|
|
set_cmd_error(vcmd, SCST_LOAD_SENSE(scst_sense_hardw_error));
|
|
}
|
|
|
|
out:
|
|
TRACE_EXIT();
|
|
return;
|
|
}
|