scsi_debug.c 162.1 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8
/*
 * vvvvvvvvvvvvvvvvvvvvvvv Original vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv
 *  Copyright (C) 1992  Eric Youngdale
 *  Simulate a host adapter with 2 disks attached.  Do a lot of checking
 *  to make sure that we are not getting blocks mixed up, and PANIC if
 *  anything out of the ordinary is seen.
 * ^^^^^^^^^^^^^^^^^^^^^^^ Original ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
 *
9
 * Copyright (C) 2001 - 2016 Douglas Gilbert
L
Linus Torvalds 已提交
10
 *
11 12 13 14
 * 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 2, or (at your option)
 * any later version.
L
Linus Torvalds 已提交
15
 *
16
 *  For documentation see http://sg.danny.cz/sg/sdebug26.html
L
Linus Torvalds 已提交
17 18 19
 *
 */

20 21 22

#define pr_fmt(fmt) KBUILD_MODNAME ":%s: " fmt, __func__

L
Linus Torvalds 已提交
23 24 25 26
#include <linux/module.h>

#include <linux/kernel.h>
#include <linux/errno.h>
27
#include <linux/jiffies.h>
28
#include <linux/slab.h>
L
Linus Torvalds 已提交
29 30 31 32 33 34 35 36
#include <linux/types.h>
#include <linux/string.h>
#include <linux/genhd.h>
#include <linux/fs.h>
#include <linux/init.h>
#include <linux/proc_fs.h>
#include <linux/vmalloc.h>
#include <linux/moduleparam.h>
J
Jens Axboe 已提交
37
#include <linux/scatterlist.h>
L
Linus Torvalds 已提交
38
#include <linux/blkdev.h>
39
#include <linux/crc-t10dif.h>
40 41 42 43
#include <linux/spinlock.h>
#include <linux/interrupt.h>
#include <linux/atomic.h>
#include <linux/hrtimer.h>
D
Douglas Gilbert 已提交
44
#include <linux/uuid.h>
45 46

#include <net/checksum.h>
47

48 49
#include <asm/unaligned.h>

50 51 52
#include <scsi/scsi.h>
#include <scsi/scsi_cmnd.h>
#include <scsi/scsi_device.h>
L
Linus Torvalds 已提交
53 54
#include <scsi/scsi_host.h>
#include <scsi/scsicam.h>
55
#include <scsi/scsi_eh.h>
56
#include <scsi/scsi_tcq.h>
57
#include <scsi/scsi_dbg.h>
L
Linus Torvalds 已提交
58

59
#include "sd.h"
L
Linus Torvalds 已提交
60 61
#include "scsi_logging.h"

62
/* make sure inq_product_rev string corresponds to this version */
D
Douglas Gilbert 已提交
63 64
#define SDEBUG_VERSION "1.86"
static const char *sdebug_version_date = "20160430";
65 66

#define MY_NAME "scsi_debug"
L
Linus Torvalds 已提交
67

68
/* Additional Sense Code (ASC) */
D
Douglas Gilbert 已提交
69 70
#define NO_ADDITIONAL_SENSE 0x0
#define LOGICAL_UNIT_NOT_READY 0x4
71
#define LOGICAL_UNIT_COMMUNICATION_FAILURE 0x8
L
Linus Torvalds 已提交
72
#define UNRECOVERED_READ_ERR 0x11
D
Douglas Gilbert 已提交
73
#define PARAMETER_LIST_LENGTH_ERR 0x1a
L
Linus Torvalds 已提交
74
#define INVALID_OPCODE 0x20
75
#define LBA_OUT_OF_RANGE 0x21
L
Linus Torvalds 已提交
76
#define INVALID_FIELD_IN_CDB 0x24
D
Douglas Gilbert 已提交
77
#define INVALID_FIELD_IN_PARAM_LIST 0x26
78 79
#define UA_RESET_ASC 0x29
#define UA_CHANGED_ASC 0x2a
80 81
#define TARGET_CHANGED_ASC 0x3f
#define LUNS_CHANGED_ASCQ 0x0e
82 83
#define INSUFF_RES_ASC 0x55
#define INSUFF_RES_ASCQ 0x3
84 85 86
#define POWER_ON_RESET_ASCQ 0x0
#define BUS_RESET_ASCQ 0x2	/* scsi bus reset occurred */
#define MODE_CHANGED_ASCQ 0x1	/* mode parameters changed */
87
#define CAPACITY_CHANGED_ASCQ 0x9
L
Linus Torvalds 已提交
88
#define SAVING_PARAMS_UNSUP 0x39
89
#define TRANSPORT_PROBLEM 0x4b
D
Douglas Gilbert 已提交
90 91
#define THRESHOLD_EXCEEDED 0x5d
#define LOW_POWER_COND_ON 0x5e
92
#define MISCOMPARE_VERIFY_ASC 0x1d
93 94
#define MICROCODE_CHANGED_ASCQ 0x1	/* with TARGET_CHANGED_ASC */
#define MICROCODE_CHANGED_WO_RESET_ASCQ 0x16
L
Linus Torvalds 已提交
95

96 97 98
/* Additional Sense Code Qualifier (ASCQ) */
#define ACK_NAK_TO 0x3

L
Linus Torvalds 已提交
99 100 101 102 103 104 105
/* Default values for driver parameters */
#define DEF_NUM_HOST   1
#define DEF_NUM_TGTS   1
#define DEF_MAX_LUNS   1
/* With these defaults, this driver will make 1 host with 1 target
 * (id 0) containing 1 logical unit (lun 0). That is 1 device.
 */
106
#define DEF_ATO 1
107
#define DEF_JDELAY   1		/* if > 0 unit is a jiffy */
L
Linus Torvalds 已提交
108
#define DEF_DEV_SIZE_MB   8
109 110
#define DEF_DIF 0
#define DEF_DIX 0
L
Linus Torvalds 已提交
111
#define DEF_D_SENSE   0
112
#define DEF_EVERY_NTH   0
D
Douglas Gilbert 已提交
113
#define DEF_FAKE_RW	0
114
#define DEF_GUARD 0
115
#define DEF_HOST_LOCK 0
116 117 118
#define DEF_LBPU 0
#define DEF_LBPWS 0
#define DEF_LBPWS10 0
119
#define DEF_LBPRZ 1
120
#define DEF_LOWEST_ALIGNED 0
121
#define DEF_NDELAY   0		/* if > 0 unit is a nanosecond */
122 123 124
#define DEF_NO_LUN_0   0
#define DEF_NUM_PARTS   0
#define DEF_OPTS   0
125
#define DEF_OPT_BLKS 1024
126
#define DEF_PHYSBLK_EXP 0
D
Douglas Gilbert 已提交
127
#define DEF_PTYPE   TYPE_DISK
128
#define DEF_REMOVABLE false
129
#define DEF_SCSI_LEVEL   7    /* INQUIRY, byte2 [6->SPC-4; 7->SPC-5] */
130 131 132
#define DEF_SECTOR_SIZE 512
#define DEF_UNMAP_ALIGNMENT 0
#define DEF_UNMAP_GRANULARITY 1
133 134
#define DEF_UNMAP_MAX_BLOCKS 0xFFFFFFFF
#define DEF_UNMAP_MAX_DESC 256
135 136 137
#define DEF_VIRTUAL_GB   0
#define DEF_VPD_USE_HOSTNO 1
#define DEF_WRITESAME_LENGTH 0xFFFF
138
#define DEF_STRICT 0
139 140
#define DEF_STATISTICS false
#define DEF_SUBMIT_QUEUES 1
D
Douglas Gilbert 已提交
141
#define DEF_UUID_CTL 0
142
#define JDELAY_OVERRIDDEN -9999
L
Linus Torvalds 已提交
143

D
Douglas Gilbert 已提交
144 145
#define SDEBUG_LUN_0_VAL 0

146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167
/* bit mask values for sdebug_opts */
#define SDEBUG_OPT_NOISE		1
#define SDEBUG_OPT_MEDIUM_ERR		2
#define SDEBUG_OPT_TIMEOUT		4
#define SDEBUG_OPT_RECOVERED_ERR	8
#define SDEBUG_OPT_TRANSPORT_ERR	16
#define SDEBUG_OPT_DIF_ERR		32
#define SDEBUG_OPT_DIX_ERR		64
#define SDEBUG_OPT_MAC_TIMEOUT		128
#define SDEBUG_OPT_SHORT_TRANSFER	0x100
#define SDEBUG_OPT_Q_NOISE		0x200
#define SDEBUG_OPT_ALL_TSF		0x400
#define SDEBUG_OPT_RARE_TSF		0x800
#define SDEBUG_OPT_N_WCE		0x1000
#define SDEBUG_OPT_RESET_NOISE		0x2000
#define SDEBUG_OPT_NO_CDB_NOISE		0x4000
#define SDEBUG_OPT_ALL_NOISE (SDEBUG_OPT_NOISE | SDEBUG_OPT_Q_NOISE | \
			      SDEBUG_OPT_RESET_NOISE)
#define SDEBUG_OPT_ALL_INJECTING (SDEBUG_OPT_RECOVERED_ERR | \
				  SDEBUG_OPT_TRANSPORT_ERR | \
				  SDEBUG_OPT_DIF_ERR | SDEBUG_OPT_DIX_ERR | \
				  SDEBUG_OPT_SHORT_TRANSFER)
L
Linus Torvalds 已提交
168
/* When "every_nth" > 0 then modulo "every_nth" commands:
169
 *   - a missing response is simulated if SDEBUG_OPT_TIMEOUT is set
L
Linus Torvalds 已提交
170
 *   - a RECOVERED_ERROR is simulated on successful read and write
171
 *     commands if SDEBUG_OPT_RECOVERED_ERR is set.
172
 *   - a TRANSPORT_ERROR is simulated on successful read and write
173
 *     commands if SDEBUG_OPT_TRANSPORT_ERR is set.
L
Linus Torvalds 已提交
174 175
 *
 * When "every_nth" < 0 then after "- every_nth" commands:
176
 *   - a missing response is simulated if SDEBUG_OPT_TIMEOUT is set
L
Linus Torvalds 已提交
177
 *   - a RECOVERED_ERROR is simulated on successful read and write
178
 *     commands if SDEBUG_OPT_RECOVERED_ERR is set.
179
 *   - a TRANSPORT_ERROR is simulated on successful read and write
180 181 182 183
 *     commands if _DEBUG_OPT_TRANSPORT_ERR is set.
 * This will continue on every subsequent command until some other action
 * occurs (e.g. the user * writing a new value (other than -1 or 1) to
 * every_nth via sysfs).
L
Linus Torvalds 已提交
184 185
 */

186
/* As indicated in SAM-5 and SPC-4 Unit Attentions (UAs) are returned in
187 188 189 190 191 192
 * priority order. In the subset implemented here lower numbers have higher
 * priority. The UA numbers should be a sequence starting from 0 with
 * SDEBUG_NUM_UAS being 1 higher than the highest numbered UA. */
#define SDEBUG_UA_POR 0		/* Power on, reset, or bus device reset */
#define SDEBUG_UA_BUS_RESET 1
#define SDEBUG_UA_MODE_CHANGED 2
193
#define SDEBUG_UA_CAPACITY_CHANGED 3
194
#define SDEBUG_UA_LUNS_CHANGED 4
195 196 197
#define SDEBUG_UA_MICROCODE_CHANGED 5	/* simulate firmware change */
#define SDEBUG_UA_MICROCODE_CHANGED_WO_RESET 6
#define SDEBUG_NUM_UAS 7
198

199
/* when 1==SDEBUG_OPT_MEDIUM_ERR, a medium error is simulated at this
L
Linus Torvalds 已提交
200 201
 * sector on read commands: */
#define OPT_MEDIUM_ERR_ADDR   0x1234 /* that's sector 4660 in decimal */
202
#define OPT_MEDIUM_ERR_NUM    10     /* number of consecutive medium errs */
L
Linus Torvalds 已提交
203 204 205 206 207

/* If REPORT LUNS has luns >= 256 it can choose "flat space" (value 1)
 * or "peripheral device" addressing (value 0) */
#define SAM2_LUN_ADDRESS_METHOD 0

208 209 210 211 212 213 214 215 216
/* SDEBUG_CANQUEUE is the maximum number of commands that can be queued
 * (for response) per submit queue at one time. Can be reduced by max_queue
 * option. Command responses are not queued when jdelay=0 and ndelay=0. The
 * per-device DEF_CMD_PER_LUN can be changed via sysfs:
 * /sys/class/scsi_device/<h:c:t:l>/device/queue_depth
 * but cannot exceed SDEBUG_CANQUEUE .
 */
#define SDEBUG_CANQUEUE_WORDS  3	/* a WORD is bits in a long */
#define SDEBUG_CANQUEUE  (SDEBUG_CANQUEUE_WORDS * BITS_PER_LONG)
217 218
#define DEF_CMD_PER_LUN  255

219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237
#define F_D_IN			1
#define F_D_OUT			2
#define F_D_OUT_MAYBE		4	/* WRITE SAME, NDOB bit */
#define F_D_UNKN		8
#define F_RL_WLUN_OK		0x10
#define F_SKIP_UA		0x20
#define F_DELAY_OVERR		0x40
#define F_SA_LOW		0x80	/* cdb byte 1, bits 4 to 0 */
#define F_SA_HIGH		0x100	/* as used by variable length cdbs */
#define F_INV_OP		0x200
#define F_FAKE_RW		0x400
#define F_M_ACCESS		0x800	/* media access */

#define FF_RESPOND (F_RL_WLUN_OK | F_SKIP_UA | F_DELAY_OVERR)
#define FF_DIRECT_IO (F_M_ACCESS | F_FAKE_RW)
#define FF_SA (F_SA_HIGH | F_SA_LOW)

#define SDEBUG_MAX_PARTS 4

D
Douglas Gilbert 已提交
238
#define SDEBUG_MAX_CMD_LEN 32
239 240 241 242 243 244 245


struct sdebug_dev_info {
	struct list_head dev_list;
	unsigned int channel;
	unsigned int target;
	u64 lun;
D
Douglas Gilbert 已提交
246
	uuid_be lu_name;
247 248 249
	struct sdebug_host_info *sdbg_host;
	unsigned long uas_bm[1];
	atomic_t num_in_q;
250
	atomic_t stopped;
251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266
	bool used;
};

struct sdebug_host_info {
	struct list_head host_list;
	struct Scsi_Host *shost;
	struct device dev;
	struct list_head dev_info_list;
};

#define to_sdebug_host(d)	\
	container_of(d, struct sdebug_host_info, dev)

struct sdebug_defer {
	struct hrtimer hrt;
	struct execute_work ew;
267 268 269
	int sqa_idx;	/* index of sdebug_queue array */
	int qc_idx;	/* index of sdebug_queued_cmd array within sqa_idx */
	int issuing_cpu;
270 271 272
};

struct sdebug_queued_cmd {
273 274 275
	/* corresponding bit set in in_use_bm[] in owning struct sdebug_queue
	 * instance indicates this slot is in use.
	 */
276 277
	struct sdebug_defer *sd_dp;
	struct scsi_cmnd *a_cmnd;
278 279 280 281 282
	unsigned int inj_recovered:1;
	unsigned int inj_transport:1;
	unsigned int inj_dif:1;
	unsigned int inj_dix:1;
	unsigned int inj_short:1;
283 284
};

285 286 287 288 289
struct sdebug_queue {
	struct sdebug_queued_cmd qc_arr[SDEBUG_CANQUEUE];
	unsigned long in_use_bm[SDEBUG_CANQUEUE_WORDS];
	spinlock_t qc_lock;
	atomic_t blocked;	/* to temporarily stop more being queued */
290 291
};

292 293 294 295 296
static atomic_t sdebug_cmnd_count;   /* number of incoming commands */
static atomic_t sdebug_completions;  /* count of deferred completions */
static atomic_t sdebug_miss_cpus;    /* submission + completion cpus differ */
static atomic_t sdebug_a_tsf;	     /* 'almost task set full' counter */

297
struct opcode_info_t {
D
Douglas Gilbert 已提交
298 299
	u8 num_attached;	/* 0 if this is it (i.e. a leaf); use 0xff */
				/* for terminating element */
300 301 302 303 304 305 306 307 308 309
	u8 opcode;		/* if num_attached > 0, preferred */
	u16 sa;			/* service action */
	u32 flags;		/* OR-ed set of SDEB_F_* */
	int (*pfp)(struct scsi_cmnd *, struct sdebug_dev_info *);
	const struct opcode_info_t *arrp;  /* num_attached elements or NULL */
	u8 len_mask[16];	/* len=len_mask[0], then mask for cdb[1]... */
				/* ignore cdb bytes after position 15 */
};

/* SCSI opcodes (first byte of cdb) of interest mapped onto these indexes */
310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343
enum sdeb_opcode_index {
	SDEB_I_INVALID_OPCODE =	0,
	SDEB_I_INQUIRY = 1,
	SDEB_I_REPORT_LUNS = 2,
	SDEB_I_REQUEST_SENSE = 3,
	SDEB_I_TEST_UNIT_READY = 4,
	SDEB_I_MODE_SENSE = 5,		/* 6, 10 */
	SDEB_I_MODE_SELECT = 6,		/* 6, 10 */
	SDEB_I_LOG_SENSE = 7,
	SDEB_I_READ_CAPACITY = 8,	/* 10; 16 is in SA_IN(16) */
	SDEB_I_READ = 9,		/* 6, 10, 12, 16 */
	SDEB_I_WRITE = 10,		/* 6, 10, 12, 16 */
	SDEB_I_START_STOP = 11,
	SDEB_I_SERV_ACT_IN = 12,	/* 12, 16 */
	SDEB_I_SERV_ACT_OUT = 13,	/* 12, 16 */
	SDEB_I_MAINT_IN = 14,
	SDEB_I_MAINT_OUT = 15,
	SDEB_I_VERIFY = 16,		/* 10 only */
	SDEB_I_VARIABLE_LEN = 17,
	SDEB_I_RESERVE = 18,		/* 6, 10 */
	SDEB_I_RELEASE = 19,		/* 6, 10 */
	SDEB_I_ALLOW_REMOVAL = 20,	/* PREVENT ALLOW MEDIUM REMOVAL */
	SDEB_I_REZERO_UNIT = 21,	/* REWIND in SSC */
	SDEB_I_ATA_PT = 22,		/* 12, 16 */
	SDEB_I_SEND_DIAG = 23,
	SDEB_I_UNMAP = 24,
	SDEB_I_XDWRITEREAD = 25,	/* 10 only */
	SDEB_I_WRITE_BUFFER = 26,
	SDEB_I_WRITE_SAME = 27,		/* 10, 16 */
	SDEB_I_SYNC_CACHE = 28,		/* 10 only */
	SDEB_I_COMP_WRITE = 29,
	SDEB_I_LAST_ELEMENT = 30,	/* keep this last */
};

344

345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364
static const unsigned char opcode_ind_arr[256] = {
/* 0x0; 0x0->0x1f: 6 byte cdbs */
	SDEB_I_TEST_UNIT_READY, SDEB_I_REZERO_UNIT, 0, SDEB_I_REQUEST_SENSE,
	    0, 0, 0, 0,
	SDEB_I_READ, 0, SDEB_I_WRITE, 0, 0, 0, 0, 0,
	0, 0, SDEB_I_INQUIRY, 0, 0, SDEB_I_MODE_SELECT, SDEB_I_RESERVE,
	    SDEB_I_RELEASE,
	0, 0, SDEB_I_MODE_SENSE, SDEB_I_START_STOP, 0, SDEB_I_SEND_DIAG,
	    SDEB_I_ALLOW_REMOVAL, 0,
/* 0x20; 0x20->0x3f: 10 byte cdbs */
	0, 0, 0, 0, 0, SDEB_I_READ_CAPACITY, 0, 0,
	SDEB_I_READ, 0, SDEB_I_WRITE, 0, 0, 0, 0, SDEB_I_VERIFY,
	0, 0, 0, 0, 0, SDEB_I_SYNC_CACHE, 0, 0,
	0, 0, 0, SDEB_I_WRITE_BUFFER, 0, 0, 0, 0,
/* 0x40; 0x40->0x5f: 10 byte cdbs */
	0, SDEB_I_WRITE_SAME, SDEB_I_UNMAP, 0, 0, 0, 0, 0,
	0, 0, 0, 0, 0, SDEB_I_LOG_SENSE, 0, 0,
	0, 0, 0, SDEB_I_XDWRITEREAD, 0, SDEB_I_MODE_SELECT, SDEB_I_RESERVE,
	    SDEB_I_RELEASE,
	0, 0, SDEB_I_MODE_SENSE, 0, 0, 0, 0, 0,
365
/* 0x60; 0x60->0x7d are reserved, 0x7e is "extended cdb" */
366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401
	0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
	0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
	0, SDEB_I_VARIABLE_LEN,
/* 0x80; 0x80->0x9f: 16 byte cdbs */
	0, 0, 0, 0, 0, SDEB_I_ATA_PT, 0, 0,
	SDEB_I_READ, SDEB_I_COMP_WRITE, SDEB_I_WRITE, 0, 0, 0, 0, 0,
	0, 0, 0, SDEB_I_WRITE_SAME, 0, 0, 0, 0,
	0, 0, 0, 0, 0, 0, SDEB_I_SERV_ACT_IN, SDEB_I_SERV_ACT_OUT,
/* 0xa0; 0xa0->0xbf: 12 byte cdbs */
	SDEB_I_REPORT_LUNS, SDEB_I_ATA_PT, 0, SDEB_I_MAINT_IN,
	     SDEB_I_MAINT_OUT, 0, 0, 0,
	SDEB_I_READ, SDEB_I_SERV_ACT_OUT, SDEB_I_WRITE, SDEB_I_SERV_ACT_IN,
	     0, 0, 0, 0,
	0, 0, 0, 0, 0, 0, 0, 0,
	0, 0, 0, 0, 0, 0, 0, 0,
/* 0xc0; 0xc0->0xff: vendor specific */
	0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
	0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
	0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
	0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
};

static int resp_inquiry(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_report_luns(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_requests(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_mode_sense(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_mode_select(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_log_sense(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_readcap(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_read_dt0(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_write_dt0(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_start_stop(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_readcap16(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_get_lba_status(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_report_tgtpgs(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_unmap(struct scsi_cmnd *, struct sdebug_dev_info *);
402 403
static int resp_rsup_opcodes(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_rsup_tmfs(struct scsi_cmnd *, struct sdebug_dev_info *);
404 405 406
static int resp_write_same_10(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_write_same_16(struct scsi_cmnd *, struct sdebug_dev_info *);
static int resp_xdwriteread_10(struct scsi_cmnd *, struct sdebug_dev_info *);
407
static int resp_comp_write(struct scsi_cmnd *, struct sdebug_dev_info *);
408
static int resp_write_buffer(struct scsi_cmnd *, struct sdebug_dev_info *);
409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454

static const struct opcode_info_t msense_iarr[1] = {
	{0, 0x1a, 0, F_D_IN, NULL, NULL,
	    {6,  0xe8, 0xff, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
};

static const struct opcode_info_t mselect_iarr[1] = {
	{0, 0x15, 0, F_D_OUT, NULL, NULL,
	    {6,  0xf1, 0, 0, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
};

static const struct opcode_info_t read_iarr[3] = {
	{0, 0x28, 0, F_D_IN | FF_DIRECT_IO, resp_read_dt0, NULL,/* READ(10) */
	    {10,  0xff, 0xff, 0xff, 0xff, 0xff, 0x1f, 0xff, 0xff, 0xc7, 0, 0,
	     0, 0, 0, 0} },
	{0, 0x8, 0, F_D_IN | FF_DIRECT_IO, resp_read_dt0, NULL, /* READ(6) */
	    {6,  0xff, 0xff, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0xa8, 0, F_D_IN | FF_DIRECT_IO, resp_read_dt0, NULL,/* READ(12) */
	    {12,  0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x9f,
	     0xc7, 0, 0, 0, 0} },
};

static const struct opcode_info_t write_iarr[3] = {
	{0, 0x2a, 0, F_D_OUT | FF_DIRECT_IO, resp_write_dt0, NULL,   /* 10 */
	    {10,  0xfb, 0xff, 0xff, 0xff, 0xff, 0x1f, 0xff, 0xff, 0xc7, 0, 0,
	     0, 0, 0, 0} },
	{0, 0xa, 0, F_D_OUT | FF_DIRECT_IO, resp_write_dt0, NULL,    /* 6 */
	    {6,  0xff, 0xff, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0xaa, 0, F_D_OUT | FF_DIRECT_IO, resp_write_dt0, NULL,   /* 12 */
	    {12,  0xfb, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x9f,
	     0xc7, 0, 0, 0, 0} },
};

static const struct opcode_info_t sa_in_iarr[1] = {
	{0, 0x9e, 0x12, F_SA_LOW | F_D_IN, resp_get_lba_status, NULL,
	    {16,  0x12, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
	     0xff, 0xff, 0xff, 0, 0xc7} },
};

static const struct opcode_info_t vl_iarr[1] = {	/* VARIABLE LENGTH */
	{0, 0x7f, 0xb, F_SA_HIGH | F_D_OUT | FF_DIRECT_IO, resp_write_dt0,
	    NULL, {32,  0xc7, 0, 0, 0, 0, 0x1f, 0x18, 0x0, 0xb, 0xfa,
		   0, 0xff, 0xff, 0xff, 0xff} },	/* WRITE(32) */
};

static const struct opcode_info_t maint_in_iarr[2] = {
455
	{0, 0xa3, 0xc, F_SA_LOW | F_D_IN, resp_rsup_opcodes, NULL,
456 457
	    {12,  0xc, 0x87, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0,
	     0xc7, 0, 0, 0, 0} },
458
	{0, 0xa3, 0xd, F_SA_LOW | F_D_IN, resp_rsup_tmfs, NULL,
459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525
	    {12,  0xd, 0x80, 0, 0, 0, 0xff, 0xff, 0xff, 0xff, 0, 0xc7, 0, 0,
	     0, 0} },
};

static const struct opcode_info_t write_same_iarr[1] = {
	{0, 0x93, 0, F_D_OUT_MAYBE | FF_DIRECT_IO, resp_write_same_16, NULL,
	    {16,  0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
	     0xff, 0xff, 0xff, 0x1f, 0xc7} },
};

static const struct opcode_info_t reserve_iarr[1] = {
	{0, 0x16, 0, F_D_OUT, NULL, NULL,	/* RESERVE(6) */
	    {6,  0x1f, 0xff, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
};

static const struct opcode_info_t release_iarr[1] = {
	{0, 0x17, 0, F_D_OUT, NULL, NULL,	/* RELEASE(6) */
	    {6,  0x1f, 0xff, 0, 0, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
};


/* This array is accessed via SDEB_I_* values. Make sure all are mapped,
 * plus the terminating elements for logic that scans this table such as
 * REPORT SUPPORTED OPERATION CODES. */
static const struct opcode_info_t opcode_info_arr[SDEB_I_LAST_ELEMENT + 1] = {
/* 0 */
	{0, 0, 0, F_INV_OP | FF_RESPOND, NULL, NULL,
	    {0,  0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0x12, 0, FF_RESPOND | F_D_IN, resp_inquiry, NULL,
	    {6,  0xe3, 0xff, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0xa0, 0, FF_RESPOND | F_D_IN, resp_report_luns, NULL,
	    {12,  0xe3, 0xff, 0, 0, 0, 0xff, 0xff, 0xff, 0xff, 0, 0xc7, 0, 0,
	     0, 0} },
	{0, 0x3, 0, FF_RESPOND | F_D_IN, resp_requests, NULL,
	    {6,  0xe1, 0, 0, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0x0, 0, F_M_ACCESS | F_RL_WLUN_OK, NULL, NULL,/* TEST UNIT READY */
	    {6,  0, 0, 0, 0, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{1, 0x5a, 0, F_D_IN, resp_mode_sense, msense_iarr,
	    {10,  0xf8, 0xff, 0xff, 0, 0, 0, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0,
	     0} },
	{1, 0x55, 0, F_D_OUT, resp_mode_select, mselect_iarr,
	    {10,  0xf1, 0, 0, 0, 0, 0, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0} },
	{0, 0x4d, 0, F_D_IN, resp_log_sense, NULL,
	    {10,  0xe3, 0xff, 0xff, 0, 0xff, 0xff, 0xff, 0xff, 0xc7, 0, 0, 0,
	     0, 0, 0} },
	{0, 0x25, 0, F_D_IN, resp_readcap, NULL,
	    {10,  0xe1, 0xff, 0xff, 0xff, 0xff, 0, 0, 0x1, 0xc7, 0, 0, 0, 0,
	     0, 0} },
	{3, 0x88, 0, F_D_IN | FF_DIRECT_IO, resp_read_dt0, read_iarr,
	    {16,  0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
	     0xff, 0xff, 0xff, 0x9f, 0xc7} },		/* READ(16) */
/* 10 */
	{3, 0x8a, 0, F_D_OUT | FF_DIRECT_IO, resp_write_dt0, write_iarr,
	    {16,  0xfa, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
	     0xff, 0xff, 0xff, 0x9f, 0xc7} },		/* WRITE(16) */
	{0, 0x1b, 0, 0, resp_start_stop, NULL,		/* START STOP UNIT */
	    {6,  0x1, 0, 0xf, 0xf7, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{1, 0x9e, 0x10, F_SA_LOW | F_D_IN, resp_readcap16, sa_in_iarr,
	    {16,  0x10, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
	     0xff, 0xff, 0xff, 0x1, 0xc7} },	/* READ CAPACITY(16) */
	{0, 0, 0, F_INV_OP | FF_RESPOND, NULL, NULL, /* SA OUT */
	    {0,  0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{2, 0xa3, 0xa, F_SA_LOW | F_D_IN, resp_report_tgtpgs, maint_in_iarr,
	    {12,  0xea, 0, 0, 0, 0, 0xff, 0xff, 0xff, 0xff, 0, 0xc7, 0, 0, 0,
	     0} },
	{0, 0, 0, F_INV_OP | FF_RESPOND, NULL, NULL, /* MAINT OUT */
	    {0,  0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
526 527 528
	{0, 0x2f, 0, F_D_OUT_MAYBE | FF_DIRECT_IO, NULL, NULL, /* VERIFY(10) */
	    {10,  0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xc7,
	     0, 0, 0, 0, 0, 0} },
529 530 531 532 533 534 535 536 537 538
	{1, 0x7f, 0x9, F_SA_HIGH | F_D_IN | FF_DIRECT_IO, resp_read_dt0,
	    vl_iarr, {32,  0xc7, 0, 0, 0, 0, 0x1f, 0x18, 0x0, 0x9, 0xfe, 0,
		      0xff, 0xff, 0xff, 0xff} },/* VARIABLE LENGTH, READ(32) */
	{1, 0x56, 0, F_D_OUT, NULL, reserve_iarr, /* RESERVE(10) */
	    {10,  0xff, 0xff, 0xff, 0, 0, 0, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0,
	     0} },
	{1, 0x57, 0, F_D_OUT, NULL, release_iarr, /* RELEASE(10) */
	    {10,  0x13, 0xff, 0xff, 0, 0, 0, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0,
	     0} },
/* 20 */
539 540
	{0, 0x1e, 0, 0, NULL, NULL, /* ALLOW REMOVAL */
	    {6,  0, 0, 0, 0x3, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
541 542 543 544 545 546 547 548 549 550 551
	{0, 0x1, 0, 0, resp_start_stop, NULL, /* REWIND ?? */
	    {6,  0x1, 0, 0, 0, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0, 0, F_INV_OP | FF_RESPOND, NULL, NULL, /* ATA_PT */
	    {0,  0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0x1d, F_D_OUT, 0, NULL, NULL,	/* SEND DIAGNOSTIC */
	    {6,  0xf7, 0, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
	{0, 0x42, 0, F_D_OUT | FF_DIRECT_IO, resp_unmap, NULL, /* UNMAP */
	    {10,  0x1, 0, 0, 0, 0, 0x1f, 0xff, 0xff, 0xc7, 0, 0, 0, 0, 0, 0} },
	{0, 0x53, 0, F_D_IN | F_D_OUT | FF_DIRECT_IO, resp_xdwriteread_10,
	    NULL, {10,  0xff, 0xff, 0xff, 0xff, 0xff, 0x1f, 0xff, 0xff, 0xc7,
		   0, 0, 0, 0, 0, 0} },
552 553 554
	{0, 0x3b, 0, F_D_OUT_MAYBE, resp_write_buffer, NULL,
	    {10,  0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xc7, 0, 0,
	     0, 0, 0, 0} },			/* WRITE_BUFFER */
555 556 557 558 559 560
	{1, 0x41, 0, F_D_OUT_MAYBE | FF_DIRECT_IO, resp_write_same_10,
	    write_same_iarr, {10,  0xff, 0xff, 0xff, 0xff, 0xff, 0x1f, 0xff,
			      0xff, 0xc7, 0, 0, 0, 0, 0, 0} },
	{0, 0x35, 0, F_DELAY_OVERR | FF_DIRECT_IO, NULL, NULL, /* SYNC_CACHE */
	    {10,  0x7, 0xff, 0xff, 0xff, 0xff, 0x1f, 0xff, 0xff, 0xc7, 0, 0,
	     0, 0, 0, 0} },
561
	{0, 0x89, 0, F_D_OUT | FF_DIRECT_IO, resp_comp_write, NULL,
562 563 564 565 566 567 568 569
	    {16,  0xf8, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0, 0,
	     0, 0xff, 0x1f, 0xc7} },		/* COMPARE AND WRITE */

/* 30 */
	{0xff, 0, 0, 0, NULL, NULL,		/* terminating element */
	    {0,  0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0} },
};

570 571
static int sdebug_add_host = DEF_NUM_HOST;
static int sdebug_ato = DEF_ATO;
572
static int sdebug_jdelay = DEF_JDELAY;	/* if > 0 then unit is jiffies */
573 574 575 576 577 578 579 580 581
static int sdebug_dev_size_mb = DEF_DEV_SIZE_MB;
static int sdebug_dif = DEF_DIF;
static int sdebug_dix = DEF_DIX;
static int sdebug_dsense = DEF_D_SENSE;
static int sdebug_every_nth = DEF_EVERY_NTH;
static int sdebug_fake_rw = DEF_FAKE_RW;
static unsigned int sdebug_guard = DEF_GUARD;
static int sdebug_lowest_aligned = DEF_LOWEST_ALIGNED;
static int sdebug_max_luns = DEF_MAX_LUNS;
582
static int sdebug_max_queue = SDEBUG_CANQUEUE;	/* per submit queue */
583
static atomic_t retired_max_queue;	/* if > 0 then was prior max_queue */
584
static int sdebug_ndelay = DEF_NDELAY;	/* if > 0 then unit is nanoseconds */
585 586 587 588 589 590 591
static int sdebug_no_lun_0 = DEF_NO_LUN_0;
static int sdebug_no_uld;
static int sdebug_num_parts = DEF_NUM_PARTS;
static int sdebug_num_tgts = DEF_NUM_TGTS; /* targets per host */
static int sdebug_opt_blks = DEF_OPT_BLKS;
static int sdebug_opts = DEF_OPTS;
static int sdebug_physblk_exp = DEF_PHYSBLK_EXP;
D
Douglas Gilbert 已提交
592
static int sdebug_ptype = DEF_PTYPE; /* SCSI peripheral device type */
593 594 595 596 597 598 599 600 601 602 603 604 605
static int sdebug_scsi_level = DEF_SCSI_LEVEL;
static int sdebug_sector_size = DEF_SECTOR_SIZE;
static int sdebug_virtual_gb = DEF_VIRTUAL_GB;
static int sdebug_vpd_use_hostno = DEF_VPD_USE_HOSTNO;
static unsigned int sdebug_lbpu = DEF_LBPU;
static unsigned int sdebug_lbpws = DEF_LBPWS;
static unsigned int sdebug_lbpws10 = DEF_LBPWS10;
static unsigned int sdebug_lbprz = DEF_LBPRZ;
static unsigned int sdebug_unmap_alignment = DEF_UNMAP_ALIGNMENT;
static unsigned int sdebug_unmap_granularity = DEF_UNMAP_GRANULARITY;
static unsigned int sdebug_unmap_max_blocks = DEF_UNMAP_MAX_BLOCKS;
static unsigned int sdebug_unmap_max_desc = DEF_UNMAP_MAX_DESC;
static unsigned int sdebug_write_same_length = DEF_WRITESAME_LENGTH;
D
Douglas Gilbert 已提交
606
static int sdebug_uuid_ctl = DEF_UUID_CTL;
607 608 609 610
static bool sdebug_removable = DEF_REMOVABLE;
static bool sdebug_clustering;
static bool sdebug_host_lock = DEF_HOST_LOCK;
static bool sdebug_strict = DEF_STRICT;
611
static bool sdebug_any_injecting_opt;
612
static bool sdebug_verbose;
613
static bool have_dif_prot;
614 615
static bool sdebug_statistics = DEF_STATISTICS;
static bool sdebug_mq_active;
L
Linus Torvalds 已提交
616

D
Douglas Gilbert 已提交
617
static unsigned int sdebug_store_sectors;
L
Linus Torvalds 已提交
618 619 620 621 622 623 624 625 626 627 628
static sector_t sdebug_capacity;	/* in sectors */

/* old BIOS stuff, kernel may get rid of them but some mode sense pages
   may still need them */
static int sdebug_heads;		/* heads per disk */
static int sdebug_cylinders_per;	/* cylinders per surface */
static int sdebug_sectors_per;		/* sectors per cylinder */

static LIST_HEAD(sdebug_host_list);
static DEFINE_SPINLOCK(sdebug_host_list_lock);

629
static unsigned char *fake_storep;	/* ramdisk storage */
630
static struct sd_dif_tuple *dif_storep;	/* protection info */
631
static void *map_storep;		/* provisioning map */
L
Linus Torvalds 已提交
632

633
static unsigned long map_size;
634 635 636 637 638
static int num_aborts;
static int num_dev_resets;
static int num_target_resets;
static int num_bus_resets;
static int num_host_resets;
639 640 641
static int dix_writes;
static int dix_reads;
static int dif_errors;
L
Linus Torvalds 已提交
642

643 644
static int submit_queues = DEF_SUBMIT_QUEUES;  /* > 1 for multi-queue (mq) */
static struct sdebug_queue *sdebug_q_arr;  /* ptr to array of submit queues */
645

L
Linus Torvalds 已提交
646 647
static DEFINE_RWLOCK(atomic_rw);

648 649
static char sdebug_proc_name[] = MY_NAME;
static const char *my_name = MY_NAME;
L
Linus Torvalds 已提交
650 651 652 653 654 655 656 657 658 659 660

static struct bus_type pseudo_lld_bus;

static struct device_driver sdebug_driverfs_driver = {
	.name 		= sdebug_proc_name,
	.bus		= &pseudo_lld_bus,
};

static const int check_condition_result =
		(DRIVER_SENSE << 24) | SAM_STAT_CHECK_CONDITION;

661 662 663
static const int illegal_condition_result =
	(DRIVER_SENSE << 24) | (DID_ABORT << 16) | SAM_STAT_CHECK_CONDITION;

664 665 666
static const int device_qfull_result =
	(DID_OK << 16) | (COMMAND_COMPLETE << 8) | SAM_STAT_TASK_SET_FULL;

667

668 669 670 671 672
/* Only do the extra work involved in logical block provisioning if one or
 * more of the lbpu, lbpws or lbpws10 parameters are given and we are doing
 * real reads and writes (i.e. not skipping them for speed).
 */
static inline bool scsi_debug_lbp(void)
673 674 675 676
{
	return 0 == sdebug_fake_rw &&
		(sdebug_lbpu || sdebug_lbpws || sdebug_lbpws10);
}
D
Douglas Gilbert 已提交
677

678 679 680 681
static void *fake_store(unsigned long long lba)
{
	lba = do_div(lba, sdebug_store_sectors);

682
	return fake_storep + lba * sdebug_sector_size;
683 684 685 686
}

static struct sd_dif_tuple *dif_store(sector_t sector)
{
687
	sector = sector_div(sector, sdebug_store_sectors);
688 689 690 691

	return dif_storep + sector;
}

692 693 694 695 696 697 698 699 700
static void sdebug_max_tgts_luns(void)
{
	struct sdebug_host_info *sdbg_host;
	struct Scsi_Host *hpnt;

	spin_lock(&sdebug_host_list_lock);
	list_for_each_entry(sdbg_host, &sdebug_host_list, host_list) {
		hpnt = sdbg_host->shost;
		if ((hpnt->this_id >= 0) &&
701 702
		    (sdebug_num_tgts > hpnt->this_id))
			hpnt->max_id = sdebug_num_tgts + 1;
703
		else
704 705
			hpnt->max_id = sdebug_num_tgts;
		/* sdebug_max_luns; */
706
		hpnt->max_lun = SCSI_W_LUN_REPORT_LUNS + 1;
707 708 709 710
	}
	spin_unlock(&sdebug_host_list_lock);
}

711 712 713
enum sdeb_cmd_data {SDEB_IN_DATA = 0, SDEB_IN_CDB = 1};

/* Set in_bit to -1 to indicate no bit position of invalid field */
714 715 716
static void mk_sense_invalid_fld(struct scsi_cmnd *scp,
				 enum sdeb_cmd_data c_d,
				 int in_byte, int in_bit)
717 718 719 720 721 722 723 724 725 726 727 728 729
{
	unsigned char *sbuff;
	u8 sks[4];
	int sl, asc;

	sbuff = scp->sense_buffer;
	if (!sbuff) {
		sdev_printk(KERN_ERR, scp->device,
			    "%s: sense_buffer is NULL\n", __func__);
		return;
	}
	asc = c_d ? INVALID_FIELD_IN_CDB : INVALID_FIELD_IN_PARAM_LIST;
	memset(sbuff, 0, SCSI_SENSE_BUFFERSIZE);
730
	scsi_build_sense_buffer(sdebug_dsense, sbuff, ILLEGAL_REQUEST, asc, 0);
731 732 733 734 735 736 737 738 739
	memset(sks, 0, sizeof(sks));
	sks[0] = 0x80;
	if (c_d)
		sks[0] |= 0x40;
	if (in_bit >= 0) {
		sks[0] |= 0x8;
		sks[0] |= 0x7 & in_bit;
	}
	put_unaligned_be16(in_byte, sks + 1);
740
	if (sdebug_dsense) {
741 742 743 744 745 746 747
		sl = sbuff[7] + 8;
		sbuff[7] = sl;
		sbuff[sl] = 0x2;
		sbuff[sl + 1] = 0x6;
		memcpy(sbuff + sl + 4, sks, 3);
	} else
		memcpy(sbuff + 15, sks, 3);
748
	if (sdebug_verbose)
749 750 751 752 753
		sdev_printk(KERN_INFO, scp->device, "%s:  [sense_key,asc,ascq"
			    "]: [0x5,0x%x,0x0] %c byte=%d, bit=%d\n",
			    my_name, asc, c_d ? 'C' : 'D', in_byte, in_bit);
}

754
static void mk_sense_buffer(struct scsi_cmnd *scp, int key, int asc, int asq)
755 756 757
{
	unsigned char *sbuff;

758 759 760 761 762 763 764
	sbuff = scp->sense_buffer;
	if (!sbuff) {
		sdev_printk(KERN_ERR, scp->device,
			    "%s: sense_buffer is NULL\n", __func__);
		return;
	}
	memset(sbuff, 0, SCSI_SENSE_BUFFERSIZE);
765

766
	scsi_build_sense_buffer(sdebug_dsense, sbuff, key, asc, asq);
767

768
	if (sdebug_verbose)
769 770 771
		sdev_printk(KERN_INFO, scp->device,
			    "%s:  [sense_key,asc,ascq]: [0x%x,0x%x,0x%x]\n",
			    my_name, key, asc, asq);
772
}
L
Linus Torvalds 已提交
773

774
static void mk_sense_invalid_opcode(struct scsi_cmnd *scp)
775 776 777 778
{
	mk_sense_buffer(scp, ILLEGAL_REQUEST, INVALID_OPCODE, 0);
}

L
Linus Torvalds 已提交
779 780
static int scsi_debug_ioctl(struct scsi_device *dev, int cmd, void __user *arg)
{
781
	if (sdebug_verbose) {
782 783 784 785 786 787 788 789 790 791
		if (0x1261 == cmd)
			sdev_printk(KERN_INFO, dev,
				    "%s: BLKFLSBUF [0x1261]\n", __func__);
		else if (0x5331 == cmd)
			sdev_printk(KERN_INFO, dev,
				    "%s: CDROM_GET_CAPABILITY [0x5331]\n",
				    __func__);
		else
			sdev_printk(KERN_INFO, dev, "%s: cmd=0x%x\n",
				    __func__, cmd);
L
Linus Torvalds 已提交
792 793 794 795 796
	}
	return -EINVAL;
	/* return -ENOTTY; // correct return but upsets fdisk */
}

797 798 799 800 801 802 803 804 805 806 807 808 809 810 811 812
static void clear_luns_changed_on_target(struct sdebug_dev_info *devip)
{
	struct sdebug_host_info *sdhp;
	struct sdebug_dev_info *dp;

	spin_lock(&sdebug_host_list_lock);
	list_for_each_entry(sdhp, &sdebug_host_list, host_list) {
		list_for_each_entry(dp, &sdhp->dev_info_list, dev_list) {
			if ((devip->sdbg_host == dp->sdbg_host) &&
			    (devip->target == dp->target))
				clear_bit(SDEBUG_UA_LUNS_CHANGED, dp->uas_bm);
		}
	}
	spin_unlock(&sdebug_host_list_lock);
}

813
static int make_ua(struct scsi_cmnd *scp, struct sdebug_dev_info *devip)
L
Linus Torvalds 已提交
814
{
815 816 817 818 819 820 821 822
	int k;

	k = find_first_bit(devip->uas_bm, SDEBUG_NUM_UAS);
	if (k != SDEBUG_NUM_UAS) {
		const char *cp = NULL;

		switch (k) {
		case SDEBUG_UA_POR:
823 824
			mk_sense_buffer(scp, UNIT_ATTENTION, UA_RESET_ASC,
					POWER_ON_RESET_ASCQ);
825
			if (sdebug_verbose)
826 827 828
				cp = "power on reset";
			break;
		case SDEBUG_UA_BUS_RESET:
829 830
			mk_sense_buffer(scp, UNIT_ATTENTION, UA_RESET_ASC,
					BUS_RESET_ASCQ);
831
			if (sdebug_verbose)
832 833 834
				cp = "bus reset";
			break;
		case SDEBUG_UA_MODE_CHANGED:
835 836
			mk_sense_buffer(scp, UNIT_ATTENTION, UA_CHANGED_ASC,
					MODE_CHANGED_ASCQ);
837
			if (sdebug_verbose)
838 839
				cp = "mode parameters changed";
			break;
840
		case SDEBUG_UA_CAPACITY_CHANGED:
841 842
			mk_sense_buffer(scp, UNIT_ATTENTION, UA_CHANGED_ASC,
					CAPACITY_CHANGED_ASCQ);
843
			if (sdebug_verbose)
844
				cp = "capacity data changed";
845
			break;
846
		case SDEBUG_UA_MICROCODE_CHANGED:
847
			mk_sense_buffer(scp, UNIT_ATTENTION,
D
Douglas Gilbert 已提交
848 849
					TARGET_CHANGED_ASC,
					MICROCODE_CHANGED_ASCQ);
850
			if (sdebug_verbose)
851 852 853
				cp = "microcode has been changed";
			break;
		case SDEBUG_UA_MICROCODE_CHANGED_WO_RESET:
854
			mk_sense_buffer(scp, UNIT_ATTENTION,
855 856
					TARGET_CHANGED_ASC,
					MICROCODE_CHANGED_WO_RESET_ASCQ);
857
			if (sdebug_verbose)
858 859
				cp = "microcode has been changed without reset";
			break;
860 861 862 863 864 865
		case SDEBUG_UA_LUNS_CHANGED:
			/*
			 * SPC-3 behavior is to report a UNIT ATTENTION with
			 * ASC/ASCQ REPORTED LUNS DATA HAS CHANGED on every LUN
			 * on the target, until a REPORT LUNS command is
			 * received.  SPC-4 behavior is to report it only once.
866
			 * NOTE:  sdebug_scsi_level does not use the same
867 868
			 * values as struct scsi_device->scsi_level.
			 */
869
			if (sdebug_scsi_level >= 6)	/* SPC-4 and above */
870
				clear_luns_changed_on_target(devip);
871
			mk_sense_buffer(scp, UNIT_ATTENTION,
872 873
					TARGET_CHANGED_ASC,
					LUNS_CHANGED_ASCQ);
874
			if (sdebug_verbose)
875 876
				cp = "reported luns data has changed";
			break;
877
		default:
878 879
			pr_warn("unexpected unit attention code=%d\n", k);
			if (sdebug_verbose)
880 881 882 883
				cp = "unknown";
			break;
		}
		clear_bit(k, devip->uas_bm);
884
		if (sdebug_verbose)
885
			sdev_printk(KERN_INFO, scp->device,
886 887
				   "%s reports: Unit attention: %s\n",
				   my_name, cp);
L
Linus Torvalds 已提交
888 889 890 891 892
		return check_condition_result;
	}
	return 0;
}

893
/* Build SCSI "data-in" buffer. Returns 0 if ok else (DID_ERROR << 16). */
894
static int fill_from_dev_buffer(struct scsi_cmnd *scp, unsigned char *arr,
L
Linus Torvalds 已提交
895 896
				int arr_len)
{
897
	int act_len;
898
	struct scsi_data_buffer *sdb = scsi_in(scp);
L
Linus Torvalds 已提交
899

900
	if (!sdb->length)
L
Linus Torvalds 已提交
901
		return 0;
902
	if (!(scsi_bidi_cmnd(scp) || scp->sc_data_direction == DMA_FROM_DEVICE))
903
		return DID_ERROR << 16;
904 905 906

	act_len = sg_copy_from_buffer(sdb->table.sgl, sdb->table.nents,
				      arr, arr_len);
907
	sdb->resid = scsi_bufflen(scp) - act_len;
908

L
Linus Torvalds 已提交
909 910 911
	return 0;
}

912 913 914 915 916 917 918 919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936 937 938 939 940
/* Partial build of SCSI "data-in" buffer. Returns 0 if ok else
 * (DID_ERROR << 16). Can write to offset in data-in buffer. If multiple
 * calls, not required to write in ascending offset order. Assumes resid
 * set to scsi_bufflen() prior to any calls.
 */
static int p_fill_from_dev_buffer(struct scsi_cmnd *scp, const void *arr,
				  int arr_len, unsigned int off_dst)
{
	int act_len, n;
	struct scsi_data_buffer *sdb = scsi_in(scp);
	off_t skip = off_dst;

	if (sdb->length <= off_dst)
		return 0;
	if (!(scsi_bidi_cmnd(scp) || scp->sc_data_direction == DMA_FROM_DEVICE))
		return DID_ERROR << 16;

	act_len = sg_pcopy_from_buffer(sdb->table.sgl, sdb->table.nents,
				       arr, arr_len, skip);
	pr_debug("%s: off_dst=%u, scsi_bufflen=%u, act_len=%u, resid=%d\n",
		 __func__, off_dst, scsi_bufflen(scp), act_len, sdb->resid);
	n = (int)scsi_bufflen(scp) - ((int)off_dst + act_len);
	sdb->resid = min(sdb->resid, n);
	return 0;
}

/* Fetches from SCSI "data-out" buffer. Returns number of bytes fetched into
 * 'arr' or -1 if error.
 */
941 942
static int fetch_to_dev_buffer(struct scsi_cmnd *scp, unsigned char *arr,
			       int arr_len)
L
Linus Torvalds 已提交
943
{
944
	if (!scsi_bufflen(scp))
L
Linus Torvalds 已提交
945
		return 0;
946
	if (!(scsi_bidi_cmnd(scp) || scp->sc_data_direction == DMA_TO_DEVICE))
L
Linus Torvalds 已提交
947
		return -1;
948 949

	return scsi_sg_copy_to_buffer(scp, arr, arr_len);
L
Linus Torvalds 已提交
950 951 952 953 954
}


static const char * inq_vendor_id = "Linux   ";
static const char * inq_product_id = "scsi_debug      ";
955
static const char *inq_product_rev = "0186";	/* version less '.' */
956 957 958 959
/* Use some locally assigned NAAs for SAS addresses. */
static const u64 naa3_comp_a = 0x3222222000000000ULL;
static const u64 naa3_comp_b = 0x3333333000000000ULL;
static const u64 naa3_comp_c = 0x3111111000000000ULL;
L
Linus Torvalds 已提交
960

961
/* Device identification VPD page. Returns number of bytes placed in arr */
962 963
static int inquiry_vpd_83(unsigned char *arr, int port_group_id,
			  int target_dev_id, int dev_id_num,
D
Douglas Gilbert 已提交
964 965
			  const char *dev_id_str, int dev_id_str_len,
			  const uuid_be *lu_name)
L
Linus Torvalds 已提交
966
{
D
Douglas Gilbert 已提交
967 968
	int num, port_a;
	char b[32];
L
Linus Torvalds 已提交
969

D
Douglas Gilbert 已提交
970
	port_a = target_dev_id + 1;
L
Linus Torvalds 已提交
971 972 973 974 975 976 977 978 979 980
	/* T10 vendor identifier field format (faked) */
	arr[0] = 0x2;	/* ASCII */
	arr[1] = 0x1;
	arr[2] = 0x0;
	memcpy(&arr[4], inq_vendor_id, 8);
	memcpy(&arr[12], inq_product_id, 16);
	memcpy(&arr[28], dev_id_str, dev_id_str_len);
	num = 8 + 16 + dev_id_str_len;
	arr[3] = num;
	num += 4;
D
Douglas Gilbert 已提交
981
	if (dev_id_num >= 0) {
D
Douglas Gilbert 已提交
982 983 984 985 986 987 988 989 990 991 992
		if (sdebug_uuid_ctl) {
			/* Locally assigned UUID */
			arr[num++] = 0x1;  /* binary (not necessarily sas) */
			arr[num++] = 0xa;  /* PIV=0, lu, naa */
			arr[num++] = 0x0;
			arr[num++] = 0x12;
			arr[num++] = 0x10; /* uuid type=1, locally assigned */
			arr[num++] = 0x0;
			memcpy(arr + num, lu_name, 16);
			num += 16;
		} else {
993
			/* NAA-3, Logical unit identifier (binary) */
D
Douglas Gilbert 已提交
994 995 996 997
			arr[num++] = 0x1;  /* binary (not necessarily sas) */
			arr[num++] = 0x3;  /* PIV=0, lu, naa */
			arr[num++] = 0x0;
			arr[num++] = 0x8;
998
			put_unaligned_be64(naa3_comp_b + dev_id_num, arr + num);
D
Douglas Gilbert 已提交
999 1000
			num += 8;
		}
D
Douglas Gilbert 已提交
1001 1002 1003 1004 1005 1006 1007 1008 1009 1010
		/* Target relative port number */
		arr[num++] = 0x61;	/* proto=sas, binary */
		arr[num++] = 0x94;	/* PIV=1, target port, rel port */
		arr[num++] = 0x0;	/* reserved */
		arr[num++] = 0x4;	/* length */
		arr[num++] = 0x0;	/* reserved */
		arr[num++] = 0x0;	/* reserved */
		arr[num++] = 0x0;
		arr[num++] = 0x1;	/* relative port A */
	}
1011
	/* NAA-3, Target port identifier */
D
Douglas Gilbert 已提交
1012 1013 1014 1015
	arr[num++] = 0x61;	/* proto=sas, binary */
	arr[num++] = 0x93;	/* piv=1, target port, naa */
	arr[num++] = 0x0;
	arr[num++] = 0x8;
1016
	put_unaligned_be64(naa3_comp_a + port_a, arr + num);
1017
	num += 8;
1018
	/* NAA-3, Target port group identifier */
1019 1020 1021 1022 1023 1024
	arr[num++] = 0x61;	/* proto=sas, binary */
	arr[num++] = 0x95;	/* piv=1, target port group id */
	arr[num++] = 0x0;
	arr[num++] = 0x4;
	arr[num++] = 0;
	arr[num++] = 0;
1025 1026
	put_unaligned_be16(port_group_id, arr + num);
	num += 2;
1027
	/* NAA-3, Target device identifier */
D
Douglas Gilbert 已提交
1028 1029 1030 1031
	arr[num++] = 0x61;	/* proto=sas, binary */
	arr[num++] = 0xa3;	/* piv=1, target device, naa */
	arr[num++] = 0x0;
	arr[num++] = 0x8;
1032
	put_unaligned_be64(naa3_comp_a + target_dev_id, arr + num);
1033
	num += 8;
D
Douglas Gilbert 已提交
1034 1035 1036 1037 1038
	/* SCSI name string: Target device identifier */
	arr[num++] = 0x63;	/* proto=sas, UTF-8 */
	arr[num++] = 0xa8;	/* piv=1, target device, SCSI name string */
	arr[num++] = 0x0;
	arr[num++] = 24;
1039
	memcpy(arr + num, "naa.32222220", 12);
D
Douglas Gilbert 已提交
1040 1041 1042 1043 1044 1045 1046 1047 1048 1049 1050 1051 1052 1053 1054
	num += 12;
	snprintf(b, sizeof(b), "%08X", target_dev_id);
	memcpy(arr + num, b, 8);
	num += 8;
	memset(arr + num, 0, 4);
	num += 4;
	return num;
}

static unsigned char vpd84_data[] = {
/* from 4th byte */ 0x22,0x22,0x22,0x0,0xbb,0x0,
    0x22,0x22,0x22,0x0,0xbb,0x1,
    0x22,0x22,0x22,0x0,0xbb,0x2,
};

1055
/*  Software interface identification VPD page */
1056
static int inquiry_vpd_84(unsigned char *arr)
D
Douglas Gilbert 已提交
1057 1058 1059 1060 1061
{
	memcpy(arr, vpd84_data, sizeof(vpd84_data));
	return sizeof(vpd84_data);
}

1062
/* Management network addresses VPD page */
1063
static int inquiry_vpd_85(unsigned char *arr)
D
Douglas Gilbert 已提交
1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097
{
	int num = 0;
	const char * na1 = "https://www.kernel.org/config";
	const char * na2 = "http://www.kernel.org/log";
	int plen, olen;

	arr[num++] = 0x1;	/* lu, storage config */
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x0;
	olen = strlen(na1);
	plen = olen + 1;
	if (plen % 4)
		plen = ((plen / 4) + 1) * 4;
	arr[num++] = plen;	/* length, null termianted, padded */
	memcpy(arr + num, na1, olen);
	memset(arr + num + olen, 0, plen - olen);
	num += plen;

	arr[num++] = 0x4;	/* lu, logging */
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x0;
	olen = strlen(na2);
	plen = olen + 1;
	if (plen % 4)
		plen = ((plen / 4) + 1) * 4;
	arr[num++] = plen;	/* length, null terminated, padded */
	memcpy(arr + num, na2, olen);
	memset(arr + num + olen, 0, plen - olen);
	num += plen;

	return num;
}

/* SCSI ports VPD page */
1098
static int inquiry_vpd_88(unsigned char *arr, int target_dev_id)
D
Douglas Gilbert 已提交
1099 1100 1101 1102 1103 1104 1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117
{
	int num = 0;
	int port_a, port_b;

	port_a = target_dev_id + 1;
	port_b = port_a + 1;
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x0;
	arr[num++] = 0x1;	/* relative port 1 (primary) */
	memset(arr + num, 0, 6);
	num += 6;
	arr[num++] = 0x0;
	arr[num++] = 12;	/* length tp descriptor */
	/* naa-5 target port identifier (A) */
	arr[num++] = 0x61;	/* proto=sas, binary */
	arr[num++] = 0x93;	/* PIV=1, target port, NAA */
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x8;	/* length */
1118
	put_unaligned_be64(naa3_comp_a + port_a, arr + num);
1119
	num += 8;
D
Douglas Gilbert 已提交
1120 1121 1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x0;
	arr[num++] = 0x2;	/* relative port 2 (secondary) */
	memset(arr + num, 0, 6);
	num += 6;
	arr[num++] = 0x0;
	arr[num++] = 12;	/* length tp descriptor */
	/* naa-5 target port identifier (B) */
	arr[num++] = 0x61;	/* proto=sas, binary */
	arr[num++] = 0x93;	/* PIV=1, target port, NAA */
	arr[num++] = 0x0;	/* reserved */
	arr[num++] = 0x8;	/* length */
1133
	put_unaligned_be64(naa3_comp_a + port_b, arr + num);
1134
	num += 8;
D
Douglas Gilbert 已提交
1135 1136 1137 1138 1139 1140 1141 1142 1143 1144 1145 1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159 1160 1161 1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179 1180 1181 1182 1183

	return num;
}


static unsigned char vpd89_data[] = {
/* from 4th byte */ 0,0,0,0,
'l','i','n','u','x',' ',' ',' ',
'S','A','T',' ','s','c','s','i','_','d','e','b','u','g',' ',' ',
'1','2','3','4',
0x34,0,0,0,1,0,0,0,0,0,0,0,1,0,0,0,0,0,0,0,
0xec,0,0,0,
0x5a,0xc,0xff,0x3f,0x37,0xc8,0x10,0,0,0,0,0,0x3f,0,0,0,
0,0,0,0,0x58,0x58,0x58,0x58,0x58,0x58,0x58,0x58,0x20,0x20,0x20,0x20,
0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0,0,0,0x40,0x4,0,0x2e,0x33,
0x38,0x31,0x20,0x20,0x20,0x20,0x54,0x53,0x38,0x33,0x30,0x30,0x33,0x31,
0x53,0x41,
0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,
0x20,0x20,
0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,0x20,
0x10,0x80,
0,0,0,0x2f,0,0,0,0x2,0,0x2,0x7,0,0xff,0xff,0x1,0,
0x3f,0,0xc1,0xff,0x3e,0,0x10,0x1,0xb0,0xf8,0x50,0x9,0,0,0x7,0,
0x3,0,0x78,0,0x78,0,0xf0,0,0x78,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0x2,0,0,0,0,0,0,0,
0x7e,0,0x1b,0,0x6b,0x34,0x1,0x7d,0x3,0x40,0x69,0x34,0x1,0x3c,0x3,0x40,
0x7f,0x40,0,0,0,0,0xfe,0xfe,0,0,0,0,0,0xfe,0,0,
0,0,0,0,0,0,0,0,0xb0,0xf8,0x50,0x9,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0x1,0,0xb0,0xf8,0x50,0x9,0xb0,0xf8,0x50,0x9,0x20,0x20,0x2,0,0xb6,0x42,
0,0x80,0x8a,0,0x6,0x3c,0xa,0x3c,0xff,0xff,0xc6,0x7,0,0x1,0,0x8,
0xf0,0xf,0,0x10,0x2,0,0x30,0,0,0,0,0,0,0,0x6,0xfe,
0,0,0x2,0,0x50,0,0x8a,0,0x4f,0x95,0,0,0x21,0,0xb,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
0,0,0,0,0,0,0,0,0,0,0,0,0,0,0xa5,0x51,
};

1184
/* ATA Information VPD page */
1185
static int inquiry_vpd_89(unsigned char *arr)
D
Douglas Gilbert 已提交
1186 1187 1188 1189 1190 1191 1192
{
	memcpy(arr, vpd89_data, sizeof(vpd89_data));
	return sizeof(vpd89_data);
}


static unsigned char vpdb0_data[] = {
1193 1194 1195 1196
	/* from 4th byte */ 0,0,0,4, 0,0,0x4,0, 0,0,0,64,
	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
	0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,
D
Douglas Gilbert 已提交
1197 1198
};

1199
/* Block limits VPD page (SBC-3) */
1200
static int inquiry_vpd_b0(unsigned char *arr)
D
Douglas Gilbert 已提交
1201
{
1202 1203
	unsigned int gran;

D
Douglas Gilbert 已提交
1204
	memcpy(arr, vpdb0_data, sizeof(vpdb0_data));
1205 1206

	/* Optimal transfer length granularity */
1207 1208
	gran = 1 << sdebug_physblk_exp;
	put_unaligned_be16(gran, arr + 2);
1209 1210

	/* Maximum Transfer Length */
1211 1212
	if (sdebug_store_sectors > 0x400)
		put_unaligned_be32(sdebug_store_sectors, arr + 4);
1213

1214
	/* Optimal Transfer Length */
1215
	put_unaligned_be32(sdebug_opt_blks, &arr[8]);
1216

1217
	if (sdebug_lbpu) {
1218
		/* Maximum Unmap LBA Count */
1219
		put_unaligned_be32(sdebug_unmap_max_blocks, &arr[16]);
1220 1221

		/* Maximum Unmap Block Descriptor Count */
1222
		put_unaligned_be32(sdebug_unmap_max_desc, &arr[20]);
1223 1224
	}

1225
	/* Unmap Granularity Alignment */
1226 1227
	if (sdebug_unmap_alignment) {
		put_unaligned_be32(sdebug_unmap_alignment, &arr[28]);
1228 1229 1230
		arr[28] |= 0x80; /* UGAVALID */
	}

1231
	/* Optimal Unmap Granularity */
1232
	put_unaligned_be32(sdebug_unmap_granularity, &arr[24]);
1233

1234
	/* Maximum WRITE SAME Length */
1235
	put_unaligned_be64(sdebug_write_same_length, &arr[32]);
1236 1237

	return 0x3c; /* Mandatory page length for Logical Block Provisioning */
1238

D
Douglas Gilbert 已提交
1239
	return sizeof(vpdb0_data);
L
Linus Torvalds 已提交
1240 1241
}

1242
/* Block device characteristics VPD page (SBC-3) */
1243
static int inquiry_vpd_b1(unsigned char *arr)
1244 1245 1246
{
	memset(arr, 0, 0x3c);
	arr[0] = 0;
1247 1248 1249
	arr[1] = 1;	/* non rotating medium (e.g. solid state) */
	arr[2] = 0;
	arr[3] = 5;	/* less than 1.8" */
1250 1251 1252

	return 0x3c;
}
L
Linus Torvalds 已提交
1253

1254 1255
/* Logical block provisioning VPD page (SBC-4) */
static int inquiry_vpd_b2(unsigned char *arr)
1256
{
1257
	memset(arr, 0, 0x4);
1258
	arr[0] = 0;			/* threshold exponent */
1259
	if (sdebug_lbpu)
1260
		arr[1] = 1 << 7;
1261
	if (sdebug_lbpws)
1262
		arr[1] |= 1 << 6;
1263
	if (sdebug_lbpws10)
1264
		arr[1] |= 1 << 5;
1265 1266 1267 1268 1269
	if (sdebug_lbprz && scsi_debug_lbp())
		arr[1] |= (sdebug_lbprz & 0x7) << 2;  /* sbc4r07 and later */
	/* anc_sup=0; dp=0 (no provisioning group descriptor) */
	/* minimum_percentage=0; provisioning_type=0 (unknown) */
	/* threshold_percentage=0 */
1270
	return 0x4;
1271 1272
}

L
Linus Torvalds 已提交
1273
#define SDEBUG_LONG_INQ_SZ 96
D
Douglas Gilbert 已提交
1274
#define SDEBUG_MAX_INQ_ARR_SZ 584
L
Linus Torvalds 已提交
1275

1276
static int resp_inquiry(struct scsi_cmnd *scp, struct sdebug_dev_info *devip)
L
Linus Torvalds 已提交
1277 1278
{
	unsigned char pq_pdt;
1279
	unsigned char * arr;
1280
	unsigned char *cmd = scp->cmnd;
1281
	int alloc_len, n, ret;
1282
	bool have_wlun, is_disk;
L
Linus Torvalds 已提交
1283

1284
	alloc_len = get_unaligned_be16(cmd + 3);
1285 1286 1287
	arr = kzalloc(SDEBUG_MAX_INQ_ARR_SZ, GFP_ATOMIC);
	if (! arr)
		return DID_REQUEUE << 16;
1288
	is_disk = (sdebug_ptype == TYPE_DISK);
D
Douglas Gilbert 已提交
1289
	have_wlun = scsi_is_wlun(scp->device->lun);
1290
	if (have_wlun)
D
Douglas Gilbert 已提交
1291 1292 1293
		pq_pdt = TYPE_WLUN;	/* present, wlun */
	else if (sdebug_no_lun_0 && (devip->lun == SDEBUG_LUN_0_VAL))
		pq_pdt = 0x7f;	/* not present, PQ=3, PDT=0x1f */
D
Douglas Gilbert 已提交
1294
	else
1295
		pq_pdt = (sdebug_ptype & 0x1f);
L
Linus Torvalds 已提交
1296 1297
	arr[0] = pq_pdt;
	if (0x2 & cmd[1]) {  /* CMDDT bit set */
1298
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 1, 1);
1299
		kfree(arr);
L
Linus Torvalds 已提交
1300 1301
		return check_condition_result;
	} else if (0x1 & cmd[1]) {  /* EVPD bit set */
1302
		int lu_id_num, port_group_id, target_dev_id, len;
D
Douglas Gilbert 已提交
1303 1304
		char lu_id_str[6];
		int host_no = devip->sdbg_host->shost->host_no;
L
Linus Torvalds 已提交
1305
		
1306 1307
		port_group_id = (((host_no + 1) & 0x7f) << 8) +
		    (devip->channel & 0x7f);
D
Douglas Gilbert 已提交
1308
		if (sdebug_vpd_use_hostno == 0)
D
Douglas Gilbert 已提交
1309
			host_no = 0;
1310
		lu_id_num = have_wlun ? -1 : (((host_no + 1) * 2000) +
D
Douglas Gilbert 已提交
1311 1312 1313 1314
			    (devip->target * 1000) + devip->lun);
		target_dev_id = ((host_no + 1) * 2000) +
				 (devip->target * 1000) - 3;
		len = scnprintf(lu_id_str, 6, "%d", lu_id_num);
L
Linus Torvalds 已提交
1315
		if (0 == cmd[2]) { /* supported vital product data pages */
D
Douglas Gilbert 已提交
1316 1317 1318 1319 1320 1321 1322 1323 1324 1325
			arr[1] = cmd[2];	/*sanity */
			n = 4;
			arr[n++] = 0x0;   /* this page */
			arr[n++] = 0x80;  /* unit serial number */
			arr[n++] = 0x83;  /* device identification */
			arr[n++] = 0x84;  /* software interface ident. */
			arr[n++] = 0x85;  /* management network addresses */
			arr[n++] = 0x86;  /* extended inquiry */
			arr[n++] = 0x87;  /* mode page policy */
			arr[n++] = 0x88;  /* SCSI ports */
1326 1327 1328 1329 1330 1331
			if (is_disk) {	  /* SBC only */
				arr[n++] = 0x89;  /* ATA information */
				arr[n++] = 0xb0;  /* Block limits */
				arr[n++] = 0xb1;  /* Block characteristics */
				arr[n++] = 0xb2;  /* Logical Block Prov */
			}
D
Douglas Gilbert 已提交
1332
			arr[3] = n - 4;	  /* number of supported VPD pages */
L
Linus Torvalds 已提交
1333
		} else if (0x80 == cmd[2]) { /* unit serial number */
D
Douglas Gilbert 已提交
1334
			arr[1] = cmd[2];	/*sanity */
L
Linus Torvalds 已提交
1335
			arr[3] = len;
D
Douglas Gilbert 已提交
1336
			memcpy(&arr[4], lu_id_str, len);
L
Linus Torvalds 已提交
1337
		} else if (0x83 == cmd[2]) { /* device identification */
D
Douglas Gilbert 已提交
1338
			arr[1] = cmd[2];	/*sanity */
1339 1340
			arr[3] = inquiry_vpd_83(&arr[4], port_group_id,
						target_dev_id, lu_id_num,
D
Douglas Gilbert 已提交
1341 1342
						lu_id_str, len,
						&devip->lu_name);
D
Douglas Gilbert 已提交
1343 1344
		} else if (0x84 == cmd[2]) { /* Software interface ident. */
			arr[1] = cmd[2];	/*sanity */
1345
			arr[3] = inquiry_vpd_84(&arr[4]);
D
Douglas Gilbert 已提交
1346 1347
		} else if (0x85 == cmd[2]) { /* Management network addresses */
			arr[1] = cmd[2];	/*sanity */
1348
			arr[3] = inquiry_vpd_85(&arr[4]);
D
Douglas Gilbert 已提交
1349 1350 1351
		} else if (0x86 == cmd[2]) { /* extended inquiry */
			arr[1] = cmd[2];	/*sanity */
			arr[3] = 0x3c;	/* number of following entries */
1352
			if (sdebug_dif == SD_DIF_TYPE3_PROTECTION)
1353
				arr[4] = 0x4;	/* SPT: GRD_CHK:1 */
1354
			else if (have_dif_prot)
1355 1356 1357
				arr[4] = 0x5;   /* SPT: GRD_CHK:1, REF_CHK:1 */
			else
				arr[4] = 0x0;   /* no protection stuff */
D
Douglas Gilbert 已提交
1358 1359 1360 1361 1362 1363 1364 1365 1366 1367
			arr[5] = 0x7;   /* head of q, ordered + simple q's */
		} else if (0x87 == cmd[2]) { /* mode page policy */
			arr[1] = cmd[2];	/*sanity */
			arr[3] = 0x8;	/* number of following entries */
			arr[4] = 0x2;	/* disconnect-reconnect mp */
			arr[6] = 0x80;	/* mlus, shared */
			arr[8] = 0x18;	 /* protocol specific lu */
			arr[10] = 0x82;	 /* mlus, per initiator port */
		} else if (0x88 == cmd[2]) { /* SCSI Ports */
			arr[1] = cmd[2];	/*sanity */
1368 1369
			arr[3] = inquiry_vpd_88(&arr[4], target_dev_id);
		} else if (is_disk && 0x89 == cmd[2]) { /* ATA information */
D
Douglas Gilbert 已提交
1370
			arr[1] = cmd[2];        /*sanity */
1371
			n = inquiry_vpd_89(&arr[4]);
1372
			put_unaligned_be16(n, arr + 2);
1373
		} else if (is_disk && 0xb0 == cmd[2]) { /* Block limits */
D
Douglas Gilbert 已提交
1374
			arr[1] = cmd[2];        /*sanity */
1375 1376
			arr[3] = inquiry_vpd_b0(&arr[4]);
		} else if (is_disk && 0xb1 == cmd[2]) { /* Block char. */
1377
			arr[1] = cmd[2];        /*sanity */
1378 1379
			arr[3] = inquiry_vpd_b1(&arr[4]);
		} else if (is_disk && 0xb2 == cmd[2]) { /* LB Prov. */
1380
			arr[1] = cmd[2];        /*sanity */
1381
			arr[3] = inquiry_vpd_b2(&arr[4]);
L
Linus Torvalds 已提交
1382
		} else {
1383
			mk_sense_invalid_fld(scp, SDEB_IN_CDB, 2, -1);
1384
			kfree(arr);
L
Linus Torvalds 已提交
1385 1386
			return check_condition_result;
		}
1387
		len = min(get_unaligned_be16(arr + 2) + 4, alloc_len);
1388
		ret = fill_from_dev_buffer(scp, arr,
D
Douglas Gilbert 已提交
1389
			    min(len, SDEBUG_MAX_INQ_ARR_SZ));
1390 1391
		kfree(arr);
		return ret;
L
Linus Torvalds 已提交
1392 1393
	}
	/* drops through here for a standard inquiry */
1394 1395
	arr[1] = sdebug_removable ? 0x80 : 0;	/* Removable disk */
	arr[2] = sdebug_scsi_level;
L
Linus Torvalds 已提交
1396 1397
	arr[3] = 2;    /* response_data_format==2 */
	arr[4] = SDEBUG_LONG_INQ_SZ - 5;
1398
	arr[5] = (int)have_dif_prot;	/* PROTECT bit */
D
Douglas Gilbert 已提交
1399
	if (sdebug_vpd_use_hostno == 0)
1400
		arr[5] = 0x10; /* claim: implicit TGPS */
D
Douglas Gilbert 已提交
1401
	arr[6] = 0x10; /* claim: MultiP */
L
Linus Torvalds 已提交
1402
	/* arr[6] |= 0x40; ... claim: EncServ (enclosure services) */
D
Douglas Gilbert 已提交
1403
	arr[7] = 0xa; /* claim: LINKED + CMDQUE */
L
Linus Torvalds 已提交
1404 1405 1406 1407
	memcpy(&arr[8], inq_vendor_id, 8);
	memcpy(&arr[16], inq_product_id, 16);
	memcpy(&arr[32], inq_product_rev, 4);
	/* version descriptors (2 bytes each) follow */
1408 1409
	put_unaligned_be16(0xc0, arr + 58);   /* SAM-6 no version claimed */
	put_unaligned_be16(0x5c0, arr + 60);  /* SPC-5 no version claimed */
D
Douglas Gilbert 已提交
1410
	n = 62;
1411 1412 1413 1414 1415 1416
	if (is_disk) {		/* SBC-4 no version claimed */
		put_unaligned_be16(0x600, arr + n);
		n += 2;
	} else if (sdebug_ptype == TYPE_TAPE) {	/* SSC-4 rev 3 */
		put_unaligned_be16(0x525, arr + n);
		n += 2;
L
Linus Torvalds 已提交
1417
	}
1418
	put_unaligned_be16(0x2100, arr + n);	/* SPL-4 no version claimed */
1419
	ret = fill_from_dev_buffer(scp, arr,
L
Linus Torvalds 已提交
1420
			    min(alloc_len, SDEBUG_LONG_INQ_SZ));
1421 1422
	kfree(arr);
	return ret;
L
Linus Torvalds 已提交
1423 1424
}

1425 1426 1427
static unsigned char iec_m_pg[] = {0x1c, 0xa, 0x08, 0, 0, 0, 0, 0,
				   0, 0, 0x0, 0x0};

L
Linus Torvalds 已提交
1428 1429 1430 1431
static int resp_requests(struct scsi_cmnd * scp,
			 struct sdebug_dev_info * devip)
{
	unsigned char * sbuff;
1432
	unsigned char *cmd = scp->cmnd;
1433
	unsigned char arr[SCSI_SENSE_BUFFERSIZE];
1434
	bool dsense;
L
Linus Torvalds 已提交
1435 1436
	int len = 18;

D
Douglas Gilbert 已提交
1437
	memset(arr, 0, sizeof(arr));
1438
	dsense = !!(cmd[1] & 1);
1439
	sbuff = scp->sense_buffer;
D
Douglas Gilbert 已提交
1440
	if ((iec_m_pg[2] & 0x4) && (6 == (iec_m_pg[3] & 0xf))) {
1441
		if (dsense) {
D
Douglas Gilbert 已提交
1442 1443 1444 1445
			arr[0] = 0x72;
			arr[1] = 0x0;		/* NO_SENSE in sense_key */
			arr[2] = THRESHOLD_EXCEEDED;
			arr[3] = 0xff;		/* TEST set and MRIE==6 */
1446
			len = 8;
D
Douglas Gilbert 已提交
1447 1448 1449 1450 1451 1452 1453 1454
		} else {
			arr[0] = 0x70;
			arr[2] = 0x0;		/* NO_SENSE in sense_key */
			arr[7] = 0xa;   	/* 18 byte sense buffer */
			arr[12] = THRESHOLD_EXCEEDED;
			arr[13] = 0xff;		/* TEST set and MRIE==6 */
		}
	} else {
1455
		memcpy(arr, sbuff, SCSI_SENSE_BUFFERSIZE);
1456
		if (arr[0] >= 0x70 && dsense == sdebug_dsense)
1457 1458 1459 1460 1461 1462 1463 1464 1465 1466 1467 1468 1469
			;	/* have sense and formats match */
		else if (arr[0] <= 0x70) {
			if (dsense) {
				memset(arr, 0, 8);
				arr[0] = 0x72;
				len = 8;
			} else {
				memset(arr, 0, 18);
				arr[0] = 0x70;
				arr[7] = 0xa;
			}
		} else if (dsense) {
			memset(arr, 0, 8);
D
Douglas Gilbert 已提交
1470 1471 1472 1473 1474
			arr[0] = 0x72;
			arr[1] = sbuff[2];     /* sense key */
			arr[2] = sbuff[12];    /* asc */
			arr[3] = sbuff[13];    /* ascq */
			len = 8;
1475 1476 1477 1478 1479 1480 1481
		} else {
			memset(arr, 0, 18);
			arr[0] = 0x70;
			arr[2] = sbuff[1];
			arr[7] = 0xa;
			arr[12] = sbuff[1];
			arr[13] = sbuff[3];
D
Douglas Gilbert 已提交
1482
		}
1483

D
Douglas Gilbert 已提交
1484
	}
1485
	mk_sense_buffer(scp, 0, NO_ADDITIONAL_SENSE, 0);
L
Linus Torvalds 已提交
1486 1487 1488
	return fill_from_dev_buffer(scp, arr, len);
}

D
Douglas Gilbert 已提交
1489 1490 1491
static int resp_start_stop(struct scsi_cmnd * scp,
			   struct sdebug_dev_info * devip)
{
1492
	unsigned char *cmd = scp->cmnd;
1493
	int power_cond, stop;
D
Douglas Gilbert 已提交
1494 1495 1496

	power_cond = (cmd[4] & 0xf0) >> 4;
	if (power_cond) {
1497
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 4, 7);
D
Douglas Gilbert 已提交
1498 1499
		return check_condition_result;
	}
1500 1501
	stop = !(cmd[4] & 1);
	atomic_xchg(&devip->stopped, stop);
D
Douglas Gilbert 已提交
1502 1503 1504
	return 0;
}

1505 1506
static sector_t get_sdebug_capacity(void)
{
1507 1508 1509 1510 1511
	static const unsigned int gibibyte = 1073741824;

	if (sdebug_virtual_gb > 0)
		return (sector_t)sdebug_virtual_gb *
			(gibibyte / sdebug_sector_size);
1512 1513 1514 1515
	else
		return sdebug_store_sectors;
}

L
Linus Torvalds 已提交
1516 1517 1518 1519 1520
#define SDEBUG_READCAP_ARR_SZ 8
static int resp_readcap(struct scsi_cmnd * scp,
			struct sdebug_dev_info * devip)
{
	unsigned char arr[SDEBUG_READCAP_ARR_SZ];
D
Douglas Gilbert 已提交
1521
	unsigned int capac;
L
Linus Torvalds 已提交
1522

D
Douglas Gilbert 已提交
1523
	/* following just in case virtual_gb changed */
1524
	sdebug_capacity = get_sdebug_capacity();
L
Linus Torvalds 已提交
1525
	memset(arr, 0, SDEBUG_READCAP_ARR_SZ);
D
Douglas Gilbert 已提交
1526 1527
	if (sdebug_capacity < 0xffffffff) {
		capac = (unsigned int)sdebug_capacity - 1;
1528 1529 1530 1531
		put_unaligned_be32(capac, arr + 0);
	} else
		put_unaligned_be32(0xffffffff, arr + 0);
	put_unaligned_be16(sdebug_sector_size, arr + 6);
L
Linus Torvalds 已提交
1532 1533 1534
	return fill_from_dev_buffer(scp, arr, SDEBUG_READCAP_ARR_SZ);
}

D
Douglas Gilbert 已提交
1535 1536 1537 1538
#define SDEBUG_READCAP16_ARR_SZ 32
static int resp_readcap16(struct scsi_cmnd * scp,
			  struct sdebug_dev_info * devip)
{
1539
	unsigned char *cmd = scp->cmnd;
D
Douglas Gilbert 已提交
1540
	unsigned char arr[SDEBUG_READCAP16_ARR_SZ];
1541
	int alloc_len;
D
Douglas Gilbert 已提交
1542

1543
	alloc_len = get_unaligned_be32(cmd + 10);
D
Douglas Gilbert 已提交
1544
	/* following just in case virtual_gb changed */
1545
	sdebug_capacity = get_sdebug_capacity();
D
Douglas Gilbert 已提交
1546
	memset(arr, 0, SDEBUG_READCAP16_ARR_SZ);
1547 1548 1549 1550
	put_unaligned_be64((u64)(sdebug_capacity - 1), arr + 0);
	put_unaligned_be32(sdebug_sector_size, arr + 8);
	arr[13] = sdebug_physblk_exp & 0xf;
	arr[14] = (sdebug_lowest_aligned >> 8) & 0x3f;
1551

1552
	if (scsi_debug_lbp()) {
1553
		arr[14] |= 0x80; /* LBPME */
1554 1555 1556 1557 1558 1559
		/* from sbc4r07, this LBPRZ field is 1 bit, but the LBPRZ in
		 * the LB Provisioning VPD page is 3 bits. Note that lbprz=2
		 * in the wider field maps to 0 in this field.
		 */
		if (sdebug_lbprz & 1)	/* precisely what the draft requires */
			arr[14] |= 0x40;
1560
	}
1561

1562
	arr[15] = sdebug_lowest_aligned & 0xff;
1563

1564
	if (have_dif_prot) {
1565
		arr[12] = (sdebug_dif - 1) << 1; /* P_TYPE */
1566 1567 1568
		arr[12] |= 1; /* PROT_EN */
	}

D
Douglas Gilbert 已提交
1569 1570 1571 1572
	return fill_from_dev_buffer(scp, arr,
				    min(alloc_len, SDEBUG_READCAP16_ARR_SZ));
}

1573 1574 1575 1576 1577
#define SDEBUG_MAX_TGTPGS_ARR_SZ 1412

static int resp_report_tgtpgs(struct scsi_cmnd * scp,
			      struct sdebug_dev_info * devip)
{
1578
	unsigned char *cmd = scp->cmnd;
1579 1580 1581 1582 1583
	unsigned char * arr;
	int host_no = devip->sdbg_host->shost->host_no;
	int n, ret, alen, rlen;
	int port_group_a, port_group_b, port_a, port_b;

1584
	alen = get_unaligned_be32(cmd + 6);
1585 1586 1587
	arr = kzalloc(SDEBUG_MAX_TGTPGS_ARR_SZ, GFP_ATOMIC);
	if (! arr)
		return DID_REQUEUE << 16;
1588 1589 1590 1591 1592 1593 1594 1595 1596
	/*
	 * EVPD page 0x88 states we have two ports, one
	 * real and a fake port with no device connected.
	 * So we create two port groups with one port each
	 * and set the group with port B to unavailable.
	 */
	port_a = 0x1; /* relative port A */
	port_b = 0x2; /* relative port B */
	port_group_a = (((host_no + 1) & 0x7f) << 8) +
1597
			(devip->channel & 0x7f);
1598
	port_group_b = (((host_no + 1) & 0x7f) << 8) +
1599
			(devip->channel & 0x7f) + 0x80;
1600 1601 1602 1603 1604

	/*
	 * The asymmetric access state is cycled according to the host_id.
	 */
	n = 4;
D
Douglas Gilbert 已提交
1605
	if (sdebug_vpd_use_hostno == 0) {
1606 1607
		arr[n++] = host_no % 3; /* Asymm access state */
		arr[n++] = 0x0F; /* claim: all states are supported */
1608
	} else {
1609 1610
		arr[n++] = 0x0; /* Active/Optimized path */
		arr[n++] = 0x01; /* only support active/optimized paths */
1611
	}
1612 1613
	put_unaligned_be16(port_group_a, arr + n);
	n += 2;
1614 1615 1616 1617 1618 1619
	arr[n++] = 0;    /* Reserved */
	arr[n++] = 0;    /* Status code */
	arr[n++] = 0;    /* Vendor unique */
	arr[n++] = 0x1;  /* One port per group */
	arr[n++] = 0;    /* Reserved */
	arr[n++] = 0;    /* Reserved */
1620 1621
	put_unaligned_be16(port_a, arr + n);
	n += 2;
1622 1623
	arr[n++] = 3;    /* Port unavailable */
	arr[n++] = 0x08; /* claim: only unavailalbe paths are supported */
1624 1625
	put_unaligned_be16(port_group_b, arr + n);
	n += 2;
1626 1627 1628 1629 1630 1631
	arr[n++] = 0;    /* Reserved */
	arr[n++] = 0;    /* Status code */
	arr[n++] = 0;    /* Vendor unique */
	arr[n++] = 0x1;  /* One port per group */
	arr[n++] = 0;    /* Reserved */
	arr[n++] = 0;    /* Reserved */
1632 1633
	put_unaligned_be16(port_b, arr + n);
	n += 2;
1634 1635

	rlen = n - 4;
1636
	put_unaligned_be32(rlen, arr + 0);
1637 1638 1639 1640 1641 1642 1643 1644 1645 1646 1647 1648 1649 1650

	/*
	 * Return the smallest value of either
	 * - The allocated length
	 * - The constructed command length
	 * - The maximum array size
	 */
	rlen = min(alen,n);
	ret = fill_from_dev_buffer(scp, arr,
				   min(rlen, SDEBUG_MAX_TGTPGS_ARR_SZ));
	kfree(arr);
	return ret;
}

1651 1652
static int resp_rsup_opcodes(struct scsi_cmnd *scp,
			     struct sdebug_dev_info *devip)
1653 1654 1655 1656 1657 1658 1659 1660 1661 1662 1663 1664 1665 1666 1667 1668
{
	bool rctd;
	u8 reporting_opts, req_opcode, sdeb_i, supp;
	u16 req_sa, u;
	u32 alloc_len, a_len;
	int k, offset, len, errsts, count, bump, na;
	const struct opcode_info_t *oip;
	const struct opcode_info_t *r_oip;
	u8 *arr;
	u8 *cmd = scp->cmnd;

	rctd = !!(cmd[2] & 0x80);
	reporting_opts = cmd[2] & 0x7;
	req_opcode = cmd[3];
	req_sa = get_unaligned_be16(cmd + 4);
	alloc_len = get_unaligned_be32(cmd + 6);
1669
	if (alloc_len < 4 || alloc_len > 0xffff) {
1670 1671 1672 1673 1674 1675 1676
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 6, -1);
		return check_condition_result;
	}
	if (alloc_len > 8192)
		a_len = 8192;
	else
		a_len = alloc_len;
1677
	arr = kzalloc((a_len < 256) ? 320 : a_len + 64, GFP_ATOMIC);
1678 1679 1680 1681 1682 1683 1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694 1695 1696 1697 1698 1699 1700 1701 1702 1703 1704 1705 1706 1707 1708 1709 1710 1711 1712 1713 1714 1715 1716 1717 1718 1719 1720 1721 1722 1723 1724 1725 1726 1727 1728 1729 1730 1731 1732 1733 1734 1735 1736 1737 1738 1739 1740 1741 1742 1743 1744 1745 1746 1747 1748 1749 1750 1751 1752 1753 1754 1755 1756 1757 1758 1759 1760 1761 1762 1763 1764 1765 1766 1767 1768 1769 1770 1771 1772 1773 1774 1775 1776 1777 1778 1779 1780 1781 1782 1783 1784 1785 1786 1787 1788 1789 1790 1791 1792 1793 1794 1795 1796 1797 1798 1799 1800 1801
	if (NULL == arr) {
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INSUFF_RES_ASC,
				INSUFF_RES_ASCQ);
		return check_condition_result;
	}
	switch (reporting_opts) {
	case 0:	/* all commands */
		/* count number of commands */
		for (count = 0, oip = opcode_info_arr;
		     oip->num_attached != 0xff; ++oip) {
			if (F_INV_OP & oip->flags)
				continue;
			count += (oip->num_attached + 1);
		}
		bump = rctd ? 20 : 8;
		put_unaligned_be32(count * bump, arr);
		for (offset = 4, oip = opcode_info_arr;
		     oip->num_attached != 0xff && offset < a_len; ++oip) {
			if (F_INV_OP & oip->flags)
				continue;
			na = oip->num_attached;
			arr[offset] = oip->opcode;
			put_unaligned_be16(oip->sa, arr + offset + 2);
			if (rctd)
				arr[offset + 5] |= 0x2;
			if (FF_SA & oip->flags)
				arr[offset + 5] |= 0x1;
			put_unaligned_be16(oip->len_mask[0], arr + offset + 6);
			if (rctd)
				put_unaligned_be16(0xa, arr + offset + 8);
			r_oip = oip;
			for (k = 0, oip = oip->arrp; k < na; ++k, ++oip) {
				if (F_INV_OP & oip->flags)
					continue;
				offset += bump;
				arr[offset] = oip->opcode;
				put_unaligned_be16(oip->sa, arr + offset + 2);
				if (rctd)
					arr[offset + 5] |= 0x2;
				if (FF_SA & oip->flags)
					arr[offset + 5] |= 0x1;
				put_unaligned_be16(oip->len_mask[0],
						   arr + offset + 6);
				if (rctd)
					put_unaligned_be16(0xa,
							   arr + offset + 8);
			}
			oip = r_oip;
			offset += bump;
		}
		break;
	case 1:	/* one command: opcode only */
	case 2:	/* one command: opcode plus service action */
	case 3:	/* one command: if sa==0 then opcode only else opcode+sa */
		sdeb_i = opcode_ind_arr[req_opcode];
		oip = &opcode_info_arr[sdeb_i];
		if (F_INV_OP & oip->flags) {
			supp = 1;
			offset = 4;
		} else {
			if (1 == reporting_opts) {
				if (FF_SA & oip->flags) {
					mk_sense_invalid_fld(scp, SDEB_IN_CDB,
							     2, 2);
					kfree(arr);
					return check_condition_result;
				}
				req_sa = 0;
			} else if (2 == reporting_opts &&
				   0 == (FF_SA & oip->flags)) {
				mk_sense_invalid_fld(scp, SDEB_IN_CDB, 4, -1);
				kfree(arr);	/* point at requested sa */
				return check_condition_result;
			}
			if (0 == (FF_SA & oip->flags) &&
			    req_opcode == oip->opcode)
				supp = 3;
			else if (0 == (FF_SA & oip->flags)) {
				na = oip->num_attached;
				for (k = 0, oip = oip->arrp; k < na;
				     ++k, ++oip) {
					if (req_opcode == oip->opcode)
						break;
				}
				supp = (k >= na) ? 1 : 3;
			} else if (req_sa != oip->sa) {
				na = oip->num_attached;
				for (k = 0, oip = oip->arrp; k < na;
				     ++k, ++oip) {
					if (req_sa == oip->sa)
						break;
				}
				supp = (k >= na) ? 1 : 3;
			} else
				supp = 3;
			if (3 == supp) {
				u = oip->len_mask[0];
				put_unaligned_be16(u, arr + 2);
				arr[4] = oip->opcode;
				for (k = 1; k < u; ++k)
					arr[4 + k] = (k < 16) ?
						 oip->len_mask[k] : 0xff;
				offset = 4 + u;
			} else
				offset = 4;
		}
		arr[1] = (rctd ? 0x80 : 0) | supp;
		if (rctd) {
			put_unaligned_be16(0xa, arr + offset);
			offset += 12;
		}
		break;
	default:
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 2, 2);
		kfree(arr);
		return check_condition_result;
	}
	offset = (offset < a_len) ? offset : a_len;
	len = (offset < alloc_len) ? offset : alloc_len;
	errsts = fill_from_dev_buffer(scp, arr, len);
	kfree(arr);
	return errsts;
}

1802 1803
static int resp_rsup_tmfs(struct scsi_cmnd *scp,
			  struct sdebug_dev_info *devip)
1804 1805 1806 1807 1808 1809 1810 1811 1812 1813 1814 1815 1816 1817 1818 1819 1820 1821 1822 1823 1824 1825 1826 1827 1828
{
	bool repd;
	u32 alloc_len, len;
	u8 arr[16];
	u8 *cmd = scp->cmnd;

	memset(arr, 0, sizeof(arr));
	repd = !!(cmd[2] & 0x80);
	alloc_len = get_unaligned_be32(cmd + 6);
	if (alloc_len < 4) {
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 6, -1);
		return check_condition_result;
	}
	arr[0] = 0xc8;		/* ATS | ATSS | LURS */
	arr[1] = 0x1;		/* ITNRS */
	if (repd) {
		arr[3] = 0xc;
		len = 16;
	} else
		len = 4;

	len = (len < alloc_len) ? len : alloc_len;
	return fill_from_dev_buffer(scp, arr, len);
}

L
Linus Torvalds 已提交
1829 1830 1831 1832 1833 1834 1835 1836 1837 1838 1839 1840 1841 1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853 1854
/* <<Following mode page info copied from ST318451LW>> */

static int resp_err_recov_pg(unsigned char * p, int pcontrol, int target)
{	/* Read-Write Error Recovery page for mode_sense */
	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 resp_disconnect_pg(unsigned char * p, int pcontrol, int target)
{ 	/* Disconnect-Reconnect page for mode_sense */
	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 resp_format_pg(unsigned char * p, int pcontrol, int target)
{       /* Format device page for mode_sense */
1855 1856 1857 1858 1859
	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));
1860 1861 1862
	put_unaligned_be16(sdebug_sectors_per, p + 10);
	put_unaligned_be16(sdebug_sector_size, p + 12);
	if (sdebug_removable)
1863 1864 1865 1866
		p[20] |= 0x20; /* should agree with INQUIRY */
	if (1 == pcontrol)
		memset(p + 2, 0, sizeof(format_pg) - 2);
	return sizeof(format_pg);
L
Linus Torvalds 已提交
1867 1868
}

1869 1870 1871 1872
static unsigned char caching_pg[] = {0x8, 18, 0x14, 0, 0xff, 0xff, 0, 0,
				     0xff, 0xff, 0xff, 0xff, 0x80, 0x14, 0, 0,
				     0, 0, 0, 0};

L
Linus Torvalds 已提交
1873 1874
static int resp_caching_pg(unsigned char * p, int pcontrol, int target)
{ 	/* Caching page for mode_sense */
1875 1876 1877
	unsigned char ch_caching_pg[] = {/* 0x8, 18, */ 0x4, 0, 0, 0, 0, 0,
		0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0};
	unsigned char d_caching_pg[] = {0x8, 18, 0x14, 0, 0xff, 0xff, 0, 0,
L
Linus Torvalds 已提交
1878 1879
		0xff, 0xff, 0xff, 0xff, 0x80, 0x14, 0, 0,     0, 0, 0, 0};

1880
	if (SDEBUG_OPT_N_WCE & sdebug_opts)
1881
		caching_pg[2] &= ~0x4;	/* set WCE=0 (default WCE=1) */
L
Linus Torvalds 已提交
1882 1883
	memcpy(p, caching_pg, sizeof(caching_pg));
	if (1 == pcontrol)
1884 1885 1886
		memcpy(p + 2, ch_caching_pg, sizeof(ch_caching_pg));
	else if (2 == pcontrol)
		memcpy(p, d_caching_pg, sizeof(d_caching_pg));
L
Linus Torvalds 已提交
1887 1888 1889
	return sizeof(caching_pg);
}

1890 1891 1892
static unsigned char ctrl_m_pg[] = {0xa, 10, 2, 0, 0, 0, 0, 0,
				    0, 0, 0x2, 0x4b};

L
Linus Torvalds 已提交
1893 1894
static int resp_ctrl_m_pg(unsigned char * p, int pcontrol, int target)
{ 	/* Control mode page for mode_sense */
D
Douglas Gilbert 已提交
1895 1896 1897
	unsigned char ch_ctrl_m_pg[] = {/* 0xa, 10, */ 0x6, 0, 0, 0, 0, 0,
				        0, 0, 0, 0};
	unsigned char d_ctrl_m_pg[] = {0xa, 10, 2, 0, 0, 0, 0, 0,
L
Linus Torvalds 已提交
1898 1899
				     0, 0, 0x2, 0x4b};

1900
	if (sdebug_dsense)
L
Linus Torvalds 已提交
1901
		ctrl_m_pg[2] |= 0x4;
D
Douglas Gilbert 已提交
1902 1903
	else
		ctrl_m_pg[2] &= ~0x4;
1904

1905
	if (sdebug_ato)
1906 1907
		ctrl_m_pg[5] |= 0x80; /* ATO=1 */

L
Linus Torvalds 已提交
1908 1909
	memcpy(p, ctrl_m_pg, sizeof(ctrl_m_pg));
	if (1 == pcontrol)
D
Douglas Gilbert 已提交
1910 1911 1912
		memcpy(p + 2, ch_ctrl_m_pg, sizeof(ch_ctrl_m_pg));
	else if (2 == pcontrol)
		memcpy(p, d_ctrl_m_pg, sizeof(d_ctrl_m_pg));
L
Linus Torvalds 已提交
1913 1914 1915
	return sizeof(ctrl_m_pg);
}

D
Douglas Gilbert 已提交
1916

L
Linus Torvalds 已提交
1917 1918
static int resp_iec_m_pg(unsigned char * p, int pcontrol, int target)
{	/* Informational Exceptions control mode page for mode_sense */
D
Douglas Gilbert 已提交
1919 1920 1921 1922 1923
	unsigned char ch_iec_m_pg[] = {/* 0x1c, 0xa, */ 0x4, 0xf, 0, 0, 0, 0,
				       0, 0, 0x0, 0x0};
	unsigned char d_iec_m_pg[] = {0x1c, 0xa, 0x08, 0, 0, 0, 0, 0,
				      0, 0, 0x0, 0x0};

L
Linus Torvalds 已提交
1924 1925
	memcpy(p, iec_m_pg, sizeof(iec_m_pg));
	if (1 == pcontrol)
D
Douglas Gilbert 已提交
1926 1927 1928
		memcpy(p + 2, ch_iec_m_pg, sizeof(ch_iec_m_pg));
	else if (2 == pcontrol)
		memcpy(p, d_iec_m_pg, sizeof(d_iec_m_pg));
L
Linus Torvalds 已提交
1929 1930 1931
	return sizeof(iec_m_pg);
}

D
Douglas Gilbert 已提交
1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943 1944 1945 1946 1947 1948
static int resp_sas_sf_m_pg(unsigned char * p, int pcontrol, int target)
{	/* SAS SSP mode page - short format for mode_sense */
	unsigned char sas_sf_m_pg[] = {0x19, 0x6,
		0x6, 0x0, 0x7, 0xd0, 0x0, 0x0};

	memcpy(p, sas_sf_m_pg, sizeof(sas_sf_m_pg));
	if (1 == pcontrol)
		memset(p + 2, 0, sizeof(sas_sf_m_pg) - 2);
	return sizeof(sas_sf_m_pg);
}


static int resp_sas_pcd_m_spg(unsigned char * p, int pcontrol, int target,
			      int target_dev_id)
{	/* SAS phy control and discover mode page for mode_sense */
	unsigned char sas_pcd_m_pg[] = {0x59, 0x1, 0, 0x64, 0, 0x6, 0, 2,
		    0, 0, 0, 0, 0x10, 0x9, 0x8, 0x0,
1949 1950
		    0, 0, 0, 0, 0, 0, 0, 0,	/* insert SAS addr */
		    0, 0, 0, 0, 0, 0, 0, 0,	/* insert SAS addr */
D
Douglas Gilbert 已提交
1951 1952 1953 1954
		    0x2, 0, 0, 0, 0, 0, 0, 0,
		    0x88, 0x99, 0, 0, 0, 0, 0, 0,
		    0, 0, 0, 0, 0, 0, 0, 0,
		    0, 1, 0, 0, 0x10, 0x9, 0x8, 0x0,
1955 1956
		    0, 0, 0, 0, 0, 0, 0, 0,	/* insert SAS addr */
		    0, 0, 0, 0, 0, 0, 0, 0,	/* insert SAS addr */
D
Douglas Gilbert 已提交
1957 1958 1959 1960 1961 1962
		    0x3, 0, 0, 0, 0, 0, 0, 0,
		    0x88, 0x99, 0, 0, 0, 0, 0, 0,
		    0, 0, 0, 0, 0, 0, 0, 0,
		};
	int port_a, port_b;

1963 1964 1965 1966
	put_unaligned_be64(naa3_comp_a, sas_pcd_m_pg + 16);
	put_unaligned_be64(naa3_comp_c + 1, sas_pcd_m_pg + 24);
	put_unaligned_be64(naa3_comp_a, sas_pcd_m_pg + 64);
	put_unaligned_be64(naa3_comp_c + 1, sas_pcd_m_pg + 72);
D
Douglas Gilbert 已提交
1967 1968 1969
	port_a = target_dev_id + 1;
	port_b = port_a + 1;
	memcpy(p, sas_pcd_m_pg, sizeof(sas_pcd_m_pg));
1970 1971
	put_unaligned_be32(port_a, p + 20);
	put_unaligned_be32(port_b, p + 48 + 20);
D
Douglas Gilbert 已提交
1972 1973 1974 1975 1976 1977 1978 1979 1980 1981 1982 1983 1984 1985 1986 1987 1988
	if (1 == pcontrol)
		memset(p + 4, 0, sizeof(sas_pcd_m_pg) - 4);
	return sizeof(sas_pcd_m_pg);
}

static int resp_sas_sha_m_spg(unsigned char * p, int pcontrol)
{	/* SAS SSP shared protocol specific port mode subpage */
	unsigned char sas_sha_m_pg[] = {0x59, 0x2, 0, 0xc, 0, 0x6, 0x10, 0,
		    0, 0, 0, 0, 0, 0, 0, 0,
		};

	memcpy(p, sas_sha_m_pg, sizeof(sas_sha_m_pg));
	if (1 == pcontrol)
		memset(p + 4, 0, sizeof(sas_sha_m_pg) - 4);
	return sizeof(sas_sha_m_pg);
}

L
Linus Torvalds 已提交
1989 1990
#define SDEBUG_MAX_MSENSE_SZ 256

1991 1992
static int resp_mode_sense(struct scsi_cmnd *scp,
			   struct sdebug_dev_info *devip)
L
Linus Torvalds 已提交
1993
{
D
Douglas Gilbert 已提交
1994
	int pcontrol, pcode, subpcode, bd_len;
L
Linus Torvalds 已提交
1995
	unsigned char dev_spec;
1996
	int alloc_len, offset, len, target_dev_id;
1997
	int target = scp->device->id;
L
Linus Torvalds 已提交
1998 1999
	unsigned char * ap;
	unsigned char arr[SDEBUG_MAX_MSENSE_SZ];
2000
	unsigned char *cmd = scp->cmnd;
2001
	bool dbd, llbaa, msense_6, is_disk, bad_pcode;
L
Linus Torvalds 已提交
2002

2003
	dbd = !!(cmd[1] & 0x8);		/* disable block descriptors */
L
Linus Torvalds 已提交
2004 2005 2006 2007
	pcontrol = (cmd[2] & 0xc0) >> 6;
	pcode = cmd[2] & 0x3f;
	subpcode = cmd[3];
	msense_6 = (MODE_SENSE == cmd[0]);
2008 2009 2010
	llbaa = msense_6 ? false : !!(cmd[1] & 0x10);
	is_disk = (sdebug_ptype == TYPE_DISK);
	if (is_disk && !dbd)
D
Douglas Gilbert 已提交
2011 2012 2013
		bd_len = llbaa ? 16 : 8;
	else
		bd_len = 0;
2014
	alloc_len = msense_6 ? cmd[4] : get_unaligned_be16(cmd + 7);
L
Linus Torvalds 已提交
2015 2016
	memset(arr, 0, SDEBUG_MAX_MSENSE_SZ);
	if (0x3 == pcontrol) {  /* Saving values not supported */
2017
		mk_sense_buffer(scp, ILLEGAL_REQUEST, SAVING_PARAMS_UNSUP, 0);
L
Linus Torvalds 已提交
2018 2019
		return check_condition_result;
	}
D
Douglas Gilbert 已提交
2020 2021
	target_dev_id = ((devip->sdbg_host->shost->host_no + 1) * 2000) +
			(devip->target * 1000) - 3;
D
Douglas Gilbert 已提交
2022
	/* for disks set DPOFUA bit and clear write protect (WP) bit */
2023
	if (is_disk)
D
Douglas Gilbert 已提交
2024
		dev_spec = 0x10;	/* =0x90 if WP=1 implies read-only */
D
Douglas Gilbert 已提交
2025 2026
	else
		dev_spec = 0x0;
L
Linus Torvalds 已提交
2027 2028
	if (msense_6) {
		arr[2] = dev_spec;
D
Douglas Gilbert 已提交
2029
		arr[3] = bd_len;
L
Linus Torvalds 已提交
2030 2031 2032
		offset = 4;
	} else {
		arr[3] = dev_spec;
D
Douglas Gilbert 已提交
2033 2034 2035
		if (16 == bd_len)
			arr[4] = 0x1;	/* set LONGLBA bit */
		arr[7] = bd_len;	/* assume 255 or less */
L
Linus Torvalds 已提交
2036 2037 2038
		offset = 8;
	}
	ap = arr + offset;
2039 2040 2041
	if ((bd_len > 0) && (!sdebug_capacity))
		sdebug_capacity = get_sdebug_capacity();

D
Douglas Gilbert 已提交
2042
	if (8 == bd_len) {
2043 2044 2045 2046 2047
		if (sdebug_capacity > 0xfffffffe)
			put_unaligned_be32(0xffffffff, ap + 0);
		else
			put_unaligned_be32(sdebug_capacity, ap + 0);
		put_unaligned_be16(sdebug_sector_size, ap + 6);
D
Douglas Gilbert 已提交
2048 2049 2050
		offset += bd_len;
		ap = arr + offset;
	} else if (16 == bd_len) {
2051 2052
		put_unaligned_be64((u64)sdebug_capacity, ap + 0);
		put_unaligned_be32(sdebug_sector_size, ap + 12);
D
Douglas Gilbert 已提交
2053 2054 2055
		offset += bd_len;
		ap = arr + offset;
	}
L
Linus Torvalds 已提交
2056

D
Douglas Gilbert 已提交
2057 2058
	if ((subpcode > 0x0) && (subpcode < 0xff) && (0x19 != pcode)) {
		/* TODO: Control Extension page */
2059
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 3, -1);
L
Linus Torvalds 已提交
2060 2061
		return check_condition_result;
	}
2062 2063
	bad_pcode = false;

L
Linus Torvalds 已提交
2064 2065 2066 2067 2068 2069 2070 2071 2072 2073
	switch (pcode) {
	case 0x1:	/* Read-Write error recovery page, direct access */
		len = resp_err_recov_pg(ap, pcontrol, target);
		offset += len;
		break;
	case 0x2:	/* Disconnect-Reconnect page, all devices */
		len = resp_disconnect_pg(ap, pcontrol, target);
		offset += len;
		break;
        case 0x3:       /* Format device page, direct access */
2074 2075 2076 2077 2078
		if (is_disk) {
			len = resp_format_pg(ap, pcontrol, target);
			offset += len;
		} else
			bad_pcode = true;
L
Linus Torvalds 已提交
2079 2080
                break;
	case 0x8:	/* Caching page, direct access */
2081 2082 2083 2084 2085
		if (is_disk) {
			len = resp_caching_pg(ap, pcontrol, target);
			offset += len;
		} else
			bad_pcode = true;
L
Linus Torvalds 已提交
2086 2087 2088 2089 2090
		break;
	case 0xa:	/* Control Mode page, all devices */
		len = resp_ctrl_m_pg(ap, pcontrol, target);
		offset += len;
		break;
D
Douglas Gilbert 已提交
2091 2092
	case 0x19:	/* if spc==1 then sas phy, control+discover */
		if ((subpcode > 0x2) && (subpcode < 0xff)) {
2093
			mk_sense_invalid_fld(scp, SDEB_IN_CDB, 3, -1);
D
Douglas Gilbert 已提交
2094 2095 2096 2097 2098 2099 2100 2101 2102 2103 2104 2105
			return check_condition_result;
	        }
		len = 0;
		if ((0x0 == subpcode) || (0xff == subpcode))
			len += resp_sas_sf_m_pg(ap + len, pcontrol, target);
		if ((0x1 == subpcode) || (0xff == subpcode))
			len += resp_sas_pcd_m_spg(ap + len, pcontrol, target,
						  target_dev_id);
		if ((0x2 == subpcode) || (0xff == subpcode))
			len += resp_sas_sha_m_spg(ap + len, pcontrol);
		offset += len;
		break;
L
Linus Torvalds 已提交
2106 2107 2108 2109 2110
	case 0x1c:	/* Informational Exceptions Mode page, all devices */
		len = resp_iec_m_pg(ap, pcontrol, target);
		offset += len;
		break;
	case 0x3f:	/* Read all Mode pages */
D
Douglas Gilbert 已提交
2111 2112 2113
		if ((0 == subpcode) || (0xff == subpcode)) {
			len = resp_err_recov_pg(ap, pcontrol, target);
			len += resp_disconnect_pg(ap + len, pcontrol, target);
2114 2115 2116 2117 2118 2119
			if (is_disk) {
				len += resp_format_pg(ap + len, pcontrol,
						      target);
				len += resp_caching_pg(ap + len, pcontrol,
						       target);
			}
D
Douglas Gilbert 已提交
2120 2121 2122 2123 2124 2125 2126 2127
			len += resp_ctrl_m_pg(ap + len, pcontrol, target);
			len += resp_sas_sf_m_pg(ap + len, pcontrol, target);
			if (0xff == subpcode) {
				len += resp_sas_pcd_m_spg(ap + len, pcontrol,
						  target, target_dev_id);
				len += resp_sas_sha_m_spg(ap + len, pcontrol);
			}
			len += resp_iec_m_pg(ap + len, pcontrol, target);
2128
			offset += len;
D
Douglas Gilbert 已提交
2129
		} else {
2130
			mk_sense_invalid_fld(scp, SDEB_IN_CDB, 3, -1);
D
Douglas Gilbert 已提交
2131 2132
			return check_condition_result;
                }
L
Linus Torvalds 已提交
2133 2134
		break;
	default:
2135 2136 2137 2138
		bad_pcode = true;
		break;
	}
	if (bad_pcode) {
2139
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 2, 5);
L
Linus Torvalds 已提交
2140 2141 2142 2143
		return check_condition_result;
	}
	if (msense_6)
		arr[0] = offset - 1;
2144 2145
	else
		put_unaligned_be16((offset - 2), arr + 0);
L
Linus Torvalds 已提交
2146 2147 2148
	return fill_from_dev_buffer(scp, arr, min(alloc_len, offset));
}

D
Douglas Gilbert 已提交
2149 2150
#define SDEBUG_MAX_MSELECT_SZ 512

2151 2152
static int resp_mode_select(struct scsi_cmnd *scp,
			    struct sdebug_dev_info *devip)
D
Douglas Gilbert 已提交
2153 2154
{
	int pf, sp, ps, md_len, bd_len, off, spf, pg_len;
2155
	int param_len, res, mpage;
D
Douglas Gilbert 已提交
2156
	unsigned char arr[SDEBUG_MAX_MSELECT_SZ];
2157
	unsigned char *cmd = scp->cmnd;
2158
	int mselect6 = (MODE_SELECT == cmd[0]);
D
Douglas Gilbert 已提交
2159 2160 2161 2162

	memset(arr, 0, sizeof(arr));
	pf = cmd[1] & 0x10;
	sp = cmd[1] & 0x1;
2163
	param_len = mselect6 ? cmd[4] : get_unaligned_be16(cmd + 7);
D
Douglas Gilbert 已提交
2164
	if ((0 == pf) || sp || (param_len > SDEBUG_MAX_MSELECT_SZ)) {
2165
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, mselect6 ? 4 : 7, -1);
D
Douglas Gilbert 已提交
2166 2167 2168 2169
		return check_condition_result;
	}
        res = fetch_to_dev_buffer(scp, arr, param_len);
        if (-1 == res)
2170 2171
		return DID_ERROR << 16;
	else if (sdebug_verbose && (res < param_len))
2172 2173 2174
		sdev_printk(KERN_INFO, scp->device,
			    "%s: cdb indicated=%d, IO sent=%d bytes\n",
			    __func__, param_len, res);
2175 2176
	md_len = mselect6 ? (arr[0] + 1) : (get_unaligned_be16(arr + 0) + 2);
	bd_len = mselect6 ? arr[3] : get_unaligned_be16(arr + 6);
D
Douglas Gilbert 已提交
2177
	if (md_len > 2) {
2178
		mk_sense_invalid_fld(scp, SDEB_IN_DATA, 0, -1);
D
Douglas Gilbert 已提交
2179 2180 2181 2182 2183 2184
		return check_condition_result;
	}
	off = bd_len + (mselect6 ? 4 : 8);
	mpage = arr[off] & 0x3f;
	ps = !!(arr[off] & 0x80);
	if (ps) {
2185
		mk_sense_invalid_fld(scp, SDEB_IN_DATA, off, 7);
D
Douglas Gilbert 已提交
2186 2187 2188
		return check_condition_result;
	}
	spf = !!(arr[off] & 0x40);
2189
	pg_len = spf ? (get_unaligned_be16(arr + off + 2) + 4) :
D
Douglas Gilbert 已提交
2190 2191
		       (arr[off + 1] + 2);
	if ((pg_len + off) > param_len) {
2192
		mk_sense_buffer(scp, ILLEGAL_REQUEST,
D
Douglas Gilbert 已提交
2193 2194 2195 2196
				PARAMETER_LIST_LENGTH_ERR, 0);
		return check_condition_result;
	}
	switch (mpage) {
2197 2198 2199 2200 2201 2202 2203
	case 0x8:      /* Caching Mode page */
		if (caching_pg[1] == arr[off + 1]) {
			memcpy(caching_pg + 2, arr + off + 2,
			       sizeof(caching_pg) - 2);
			goto set_mode_changed_ua;
		}
		break;
D
Douglas Gilbert 已提交
2204 2205 2206 2207
	case 0xa:      /* Control Mode page */
		if (ctrl_m_pg[1] == arr[off + 1]) {
			memcpy(ctrl_m_pg + 2, arr + off + 2,
			       sizeof(ctrl_m_pg) - 2);
2208
			sdebug_dsense = !!(ctrl_m_pg[2] & 0x4);
2209
			goto set_mode_changed_ua;
D
Douglas Gilbert 已提交
2210 2211 2212 2213 2214 2215
		}
		break;
	case 0x1c:      /* Informational Exceptions Mode page */
		if (iec_m_pg[1] == arr[off + 1]) {
			memcpy(iec_m_pg + 2, arr + off + 2,
			       sizeof(iec_m_pg) - 2);
2216
			goto set_mode_changed_ua;
D
Douglas Gilbert 已提交
2217 2218 2219 2220 2221
		}
		break;
	default:
		break;
	}
2222
	mk_sense_invalid_fld(scp, SDEB_IN_DATA, off, 5);
D
Douglas Gilbert 已提交
2223
	return check_condition_result;
2224 2225 2226
set_mode_changed_ua:
	set_bit(SDEBUG_UA_MODE_CHANGED, devip->uas_bm);
	return 0;
D
Douglas Gilbert 已提交
2227 2228 2229 2230 2231 2232 2233 2234 2235 2236 2237 2238 2239 2240 2241 2242 2243 2244 2245 2246 2247 2248 2249 2250 2251 2252 2253 2254 2255 2256
}

static int resp_temp_l_pg(unsigned char * arr)
{
	unsigned char temp_l_pg[] = {0x0, 0x0, 0x3, 0x2, 0x0, 38,
				     0x0, 0x1, 0x3, 0x2, 0x0, 65,
		};

        memcpy(arr, temp_l_pg, sizeof(temp_l_pg));
        return sizeof(temp_l_pg);
}

static int resp_ie_l_pg(unsigned char * arr)
{
	unsigned char ie_l_pg[] = {0x0, 0x0, 0x3, 0x3, 0x0, 0x0, 38,
		};

        memcpy(arr, ie_l_pg, sizeof(ie_l_pg));
	if (iec_m_pg[2] & 0x4) {	/* TEST bit set */
		arr[4] = THRESHOLD_EXCEEDED;
		arr[5] = 0xff;
	}
        return sizeof(ie_l_pg);
}

#define SDEBUG_MAX_LSENSE_SZ 512

static int resp_log_sense(struct scsi_cmnd * scp,
                          struct sdebug_dev_info * devip)
{
2257
	int ppc, sp, pcontrol, pcode, subpcode, alloc_len, len, n;
D
Douglas Gilbert 已提交
2258
	unsigned char arr[SDEBUG_MAX_LSENSE_SZ];
2259
	unsigned char *cmd = scp->cmnd;
D
Douglas Gilbert 已提交
2260 2261 2262 2263 2264

	memset(arr, 0, sizeof(arr));
	ppc = cmd[1] & 0x2;
	sp = cmd[1] & 0x1;
	if (ppc || sp) {
2265
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 1, ppc ? 1 : 0);
D
Douglas Gilbert 已提交
2266 2267 2268 2269
		return check_condition_result;
	}
	pcontrol = (cmd[2] & 0xc0) >> 6;
	pcode = cmd[2] & 0x3f;
D
Douglas Gilbert 已提交
2270
	subpcode = cmd[3] & 0xff;
2271
	alloc_len = get_unaligned_be16(cmd + 7);
D
Douglas Gilbert 已提交
2272
	arr[0] = pcode;
D
Douglas Gilbert 已提交
2273 2274 2275 2276 2277 2278 2279 2280 2281 2282 2283 2284 2285 2286 2287 2288
	if (0 == subpcode) {
		switch (pcode) {
		case 0x0:	/* Supported log pages log page */
			n = 4;
			arr[n++] = 0x0;		/* this page */
			arr[n++] = 0xd;		/* Temperature */
			arr[n++] = 0x2f;	/* Informational exceptions */
			arr[3] = n - 4;
			break;
		case 0xd:	/* Temperature log page */
			arr[3] = resp_temp_l_pg(arr + 4);
			break;
		case 0x2f:	/* Informational exceptions log page */
			arr[3] = resp_ie_l_pg(arr + 4);
			break;
		default:
2289
			mk_sense_invalid_fld(scp, SDEB_IN_CDB, 2, 5);
D
Douglas Gilbert 已提交
2290 2291 2292 2293 2294 2295 2296 2297 2298 2299 2300 2301 2302 2303 2304 2305 2306 2307 2308 2309 2310 2311 2312 2313 2314 2315 2316 2317 2318 2319 2320
			return check_condition_result;
		}
	} else if (0xff == subpcode) {
		arr[0] |= 0x40;
		arr[1] = subpcode;
		switch (pcode) {
		case 0x0:	/* Supported log pages and subpages log page */
			n = 4;
			arr[n++] = 0x0;
			arr[n++] = 0x0;		/* 0,0 page */
			arr[n++] = 0x0;
			arr[n++] = 0xff;	/* this page */
			arr[n++] = 0xd;
			arr[n++] = 0x0;		/* Temperature */
			arr[n++] = 0x2f;
			arr[n++] = 0x0;	/* Informational exceptions */
			arr[3] = n - 4;
			break;
		case 0xd:	/* Temperature subpages */
			n = 4;
			arr[n++] = 0xd;
			arr[n++] = 0x0;		/* Temperature */
			arr[3] = n - 4;
			break;
		case 0x2f:	/* Informational exceptions subpages */
			n = 4;
			arr[n++] = 0x2f;
			arr[n++] = 0x0;		/* Informational exceptions */
			arr[3] = n - 4;
			break;
		default:
2321
			mk_sense_invalid_fld(scp, SDEB_IN_CDB, 2, 5);
D
Douglas Gilbert 已提交
2322 2323 2324
			return check_condition_result;
		}
	} else {
2325
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 3, -1);
D
Douglas Gilbert 已提交
2326 2327
		return check_condition_result;
	}
2328
	len = min(get_unaligned_be16(arr + 2) + 4, alloc_len);
D
Douglas Gilbert 已提交
2329 2330 2331 2332
	return fill_from_dev_buffer(scp, arr,
		    min(len, SDEBUG_MAX_INQ_ARR_SZ));
}

2333
static int check_device_access_params(struct scsi_cmnd *scp,
2334
				      unsigned long long lba, unsigned int num)
L
Linus Torvalds 已提交
2335
{
D
Douglas Gilbert 已提交
2336
	if (lba + num > sdebug_capacity) {
2337
		mk_sense_buffer(scp, ILLEGAL_REQUEST, LBA_OUT_OF_RANGE, 0);
L
Linus Torvalds 已提交
2338 2339
		return check_condition_result;
	}
D
Douglas Gilbert 已提交
2340 2341
	/* transfer length excessive (tie in to block limits VPD page) */
	if (num > sdebug_store_sectors) {
2342
		/* needs work to find which cdb byte 'num' comes from */
2343
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INVALID_FIELD_IN_CDB, 0);
D
Douglas Gilbert 已提交
2344 2345
		return check_condition_result;
	}
2346 2347 2348
	return 0;
}

2349
/* Returns number of bytes copied or -1 if error. */
2350 2351
static int do_device_access(struct scsi_cmnd *scmd, u64 lba, u32 num,
			    bool do_write)
2352 2353
{
	int ret;
2354
	u64 block, rest = 0;
2355 2356 2357
	struct scsi_data_buffer *sdb;
	enum dma_data_direction dir;

2358
	if (do_write) {
2359 2360 2361 2362 2363 2364
		sdb = scsi_out(scmd);
		dir = DMA_TO_DEVICE;
	} else {
		sdb = scsi_in(scmd);
		dir = DMA_FROM_DEVICE;
	}
2365

2366 2367 2368 2369
	if (!sdb->length)
		return 0;
	if (!(scsi_bidi_cmnd(scmd) || scmd->sc_data_direction == dir))
		return -1;
2370 2371 2372 2373 2374

	block = do_div(lba, sdebug_store_sectors);
	if (block + num > sdebug_store_sectors)
		rest = block + num - sdebug_store_sectors;

2375
	ret = sg_copy_buffer(sdb->table.sgl, sdb->table.nents,
2376 2377 2378
		   fake_storep + (block * sdebug_sector_size),
		   (num - rest) * sdebug_sector_size, 0, do_write);
	if (ret != (num - rest) * sdebug_sector_size)
2379 2380 2381
		return ret;

	if (rest) {
2382
		ret += sg_copy_buffer(sdb->table.sgl, sdb->table.nents,
2383 2384
			    fake_storep, rest * sdebug_sector_size,
			    (num - rest) * sdebug_sector_size, do_write);
2385
	}
2386 2387 2388 2389

	return ret;
}

2390 2391 2392
/* If fake_store(lba,num) compares equal to arr(num), then copy top half of
 * arr into fake_store(lba,num) and return true. If comparison fails then
 * return false. */
2393
static bool comp_write_worker(u64 lba, u32 num, const u8 *arr)
2394 2395 2396 2397
{
	bool res;
	u64 block, rest = 0;
	u32 store_blks = sdebug_store_sectors;
2398
	u32 lb_size = sdebug_sector_size;
2399 2400 2401 2402 2403 2404 2405 2406 2407 2408 2409 2410 2411 2412 2413 2414 2415 2416 2417 2418 2419 2420

	block = do_div(lba, store_blks);
	if (block + num > store_blks)
		rest = block + num - store_blks;

	res = !memcmp(fake_storep + (block * lb_size), arr,
		      (num - rest) * lb_size);
	if (!res)
		return res;
	if (rest)
		res = memcmp(fake_storep, arr + ((num - rest) * lb_size),
			     rest * lb_size);
	if (!res)
		return res;
	arr += num * lb_size;
	memcpy(fake_storep + (block * lb_size), arr, (num - rest) * lb_size);
	if (rest)
		memcpy(fake_storep, arr + ((num - rest) * lb_size),
		       rest * lb_size);
	return res;
}

2421
static __be16 dif_compute_csum(const void *buf, int len)
2422
{
2423
	__be16 csum;
2424

2425
	if (sdebug_guard)
2426 2427
		csum = (__force __be16)ip_compute_csum(buf, len);
	else
2428
		csum = cpu_to_be16(crc_t10dif(buf, len));
2429

2430 2431 2432 2433 2434 2435
	return csum;
}

static int dif_verify(struct sd_dif_tuple *sdt, const void *data,
		      sector_t sector, u32 ei_lba)
{
2436
	__be16 csum = dif_compute_csum(data, sdebug_sector_size);
2437 2438

	if (sdt->guard_tag != csum) {
2439
		pr_err("GUARD check failed on sector %lu rcvd 0x%04x, data 0x%04x\n",
2440 2441 2442 2443 2444
			(unsigned long)sector,
			be16_to_cpu(sdt->guard_tag),
			be16_to_cpu(csum));
		return 0x01;
	}
2445
	if (sdebug_dif == SD_DIF_TYPE1_PROTECTION &&
2446
	    be32_to_cpu(sdt->ref_tag) != (sector & 0xffffffff)) {
2447 2448
		pr_err("REF check failed on sector %lu\n",
			(unsigned long)sector);
2449 2450
		return 0x03;
	}
2451
	if (sdebug_dif == SD_DIF_TYPE2_PROTECTION &&
2452
	    be32_to_cpu(sdt->ref_tag) != ei_lba) {
2453 2454
		pr_err("REF check failed on sector %lu\n",
			(unsigned long)sector);
2455 2456 2457 2458 2459
		return 0x03;
	}
	return 0;
}

2460
static void dif_copy_prot(struct scsi_cmnd *SCpnt, sector_t sector,
2461
			  unsigned int sectors, bool read)
2462
{
2463
	size_t resid;
2464
	void *paddr;
2465
	const void *dif_store_end = dif_storep + sdebug_store_sectors;
2466
	struct sg_mapping_iter miter;
2467

2468 2469
	/* Bytes of protection data to copy into sgl */
	resid = sectors * sizeof(*dif_storep);
2470

2471 2472 2473 2474 2475 2476
	sg_miter_start(&miter, scsi_prot_sglist(SCpnt),
			scsi_prot_sg_count(SCpnt), SG_MITER_ATOMIC |
			(read ? SG_MITER_TO_SG : SG_MITER_FROM_SG));

	while (sg_miter_next(&miter) && resid > 0) {
		size_t len = min(miter.length, resid);
2477
		void *start = dif_store(sector);
2478
		size_t rest = 0;
2479 2480 2481

		if (dif_store_end < start + len)
			rest = start + len - dif_store_end;
2482

2483
		paddr = miter.addr;
2484

2485 2486 2487 2488 2489 2490 2491 2492 2493 2494 2495
		if (read)
			memcpy(paddr, start, len - rest);
		else
			memcpy(start, paddr, len - rest);

		if (rest) {
			if (read)
				memcpy(paddr + len - rest, dif_storep, rest);
			else
				memcpy(dif_storep, paddr + len - rest, rest);
		}
2496

2497
		sector += len / sizeof(*dif_storep);
2498 2499
		resid -= len;
	}
2500
	sg_miter_stop(&miter);
2501 2502 2503 2504 2505 2506 2507 2508 2509
}

static int prot_verify_read(struct scsi_cmnd *SCpnt, sector_t start_sec,
			    unsigned int sectors, u32 ei_lba)
{
	unsigned int i;
	struct sd_dif_tuple *sdt;
	sector_t sector;

2510
	for (i = 0; i < sectors; i++, ei_lba++) {
2511 2512 2513 2514 2515
		int ret;

		sector = start_sec + i;
		sdt = dif_store(sector);

2516
		if (sdt->app_tag == cpu_to_be16(0xffff))
2517 2518 2519 2520 2521 2522 2523 2524
			continue;

		ret = dif_verify(sdt, fake_store(sector), sector, ei_lba);
		if (ret) {
			dif_errors++;
			return ret;
		}
	}
2525

2526
	dif_copy_prot(SCpnt, start_sec, sectors, true);
2527 2528 2529 2530 2531
	dix_reads++;

	return 0;
}

2532
static int resp_read_dt0(struct scsi_cmnd *scp, struct sdebug_dev_info *devip)
2533
{
2534
	u8 *cmd = scp->cmnd;
2535
	struct sdebug_queued_cmd *sqcp;
2536 2537 2538
	u64 lba;
	u32 num;
	u32 ei_lba;
2539 2540
	unsigned long iflags;
	int ret;
2541
	bool check_prot;
2542

2543 2544 2545 2546 2547 2548 2549 2550 2551 2552 2553 2554 2555 2556 2557 2558 2559 2560 2561 2562 2563 2564 2565 2566 2567 2568 2569 2570 2571 2572 2573 2574 2575 2576 2577 2578 2579 2580 2581
	switch (cmd[0]) {
	case READ_16:
		ei_lba = 0;
		lba = get_unaligned_be64(cmd + 2);
		num = get_unaligned_be32(cmd + 10);
		check_prot = true;
		break;
	case READ_10:
		ei_lba = 0;
		lba = get_unaligned_be32(cmd + 2);
		num = get_unaligned_be16(cmd + 7);
		check_prot = true;
		break;
	case READ_6:
		ei_lba = 0;
		lba = (u32)cmd[3] | (u32)cmd[2] << 8 |
		      (u32)(cmd[1] & 0x1f) << 16;
		num = (0 == cmd[4]) ? 256 : cmd[4];
		check_prot = true;
		break;
	case READ_12:
		ei_lba = 0;
		lba = get_unaligned_be32(cmd + 2);
		num = get_unaligned_be32(cmd + 6);
		check_prot = true;
		break;
	case XDWRITEREAD_10:
		ei_lba = 0;
		lba = get_unaligned_be32(cmd + 2);
		num = get_unaligned_be16(cmd + 7);
		check_prot = false;
		break;
	default:	/* assume READ(32) */
		lba = get_unaligned_be64(cmd + 12);
		ei_lba = get_unaligned_be32(cmd + 20);
		num = get_unaligned_be32(cmd + 28);
		check_prot = false;
		break;
	}
2582
	if (unlikely(have_dif_prot && check_prot)) {
2583
		if (sdebug_dif == SD_DIF_TYPE2_PROTECTION &&
2584 2585 2586 2587
		    (cmd[1] & 0xe0)) {
			mk_sense_invalid_opcode(scp);
			return check_condition_result;
		}
2588 2589
		if ((sdebug_dif == SD_DIF_TYPE1_PROTECTION ||
		     sdebug_dif == SD_DIF_TYPE3_PROTECTION) &&
2590 2591 2592 2593
		    (cmd[1] & 0xe0) == 0)
			sdev_printk(KERN_ERR, scp->device, "Unprotected RD "
				    "to DIF device\n");
	}
2594
	if (unlikely(sdebug_any_injecting_opt)) {
2595
		sqcp = (struct sdebug_queued_cmd *)scp->host_scribble;
2596

2597 2598 2599 2600 2601 2602
		if (sqcp) {
			if (sqcp->inj_short)
				num /= 2;
		}
	} else
		sqcp = NULL;
2603 2604

	/* inline check_device_access_params() */
2605
	if (unlikely(lba + num > sdebug_capacity)) {
2606 2607 2608 2609
		mk_sense_buffer(scp, ILLEGAL_REQUEST, LBA_OUT_OF_RANGE, 0);
		return check_condition_result;
	}
	/* transfer length excessive (tie in to block limits VPD page) */
2610
	if (unlikely(num > sdebug_store_sectors)) {
2611 2612 2613 2614
		/* needs work to find which cdb byte 'num' comes from */
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INVALID_FIELD_IN_CDB, 0);
		return check_condition_result;
	}
2615

2616 2617 2618
	if (unlikely((SDEBUG_OPT_MEDIUM_ERR & sdebug_opts) &&
		     (lba <= (OPT_MEDIUM_ERR_ADDR + OPT_MEDIUM_ERR_NUM - 1)) &&
		     ((lba + num) > OPT_MEDIUM_ERR_ADDR))) {
D
Douglas Gilbert 已提交
2619
		/* claim unrecoverable read error */
2620
		mk_sense_buffer(scp, MEDIUM_ERROR, UNRECOVERED_READ_ERR, 0);
D
Douglas Gilbert 已提交
2621
		/* set info field and valid bit for fixed descriptor */
2622 2623
		if (0x70 == (scp->sense_buffer[0] & 0x7f)) {
			scp->sense_buffer[0] |= 0x80;	/* Valid bit */
2624 2625
			ret = (lba < OPT_MEDIUM_ERR_ADDR)
			      ? OPT_MEDIUM_ERR_ADDR : (int)lba;
2626
			put_unaligned_be32(ret, scp->sense_buffer + 3);
D
Douglas Gilbert 已提交
2627
		}
2628
		scsi_set_resid(scp, scsi_bufflen(scp));
L
Linus Torvalds 已提交
2629 2630
		return check_condition_result;
	}
2631

2632 2633
	read_lock_irqsave(&atomic_rw, iflags);

2634
	/* DIX + T10 DIF */
2635
	if (unlikely(sdebug_dix && scsi_prot_sg_count(scp))) {
2636
		int prot_ret = prot_verify_read(scp, lba, num, ei_lba);
2637 2638

		if (prot_ret) {
2639
			read_unlock_irqrestore(&atomic_rw, iflags);
2640
			mk_sense_buffer(scp, ABORTED_COMMAND, 0x10, prot_ret);
2641 2642 2643 2644
			return illegal_condition_result;
		}
	}

2645
	ret = do_device_access(scp, lba, num, false);
L
Linus Torvalds 已提交
2646
	read_unlock_irqrestore(&atomic_rw, iflags);
2647
	if (unlikely(ret == -1))
2648 2649
		return DID_ERROR << 16;

2650
	scsi_in(scp)->resid = scsi_bufflen(scp) - ret;
2651

2652 2653
	if (unlikely(sqcp)) {
		if (sqcp->inj_recovered) {
2654 2655 2656
			mk_sense_buffer(scp, RECOVERED_ERROR,
					THRESHOLD_EXCEEDED, 0);
			return check_condition_result;
2657
		} else if (sqcp->inj_transport) {
2658 2659 2660
			mk_sense_buffer(scp, ABORTED_COMMAND,
					TRANSPORT_PROBLEM, ACK_NAK_TO);
			return check_condition_result;
2661
		} else if (sqcp->inj_dif) {
2662 2663 2664
			/* Logical block guard check failed */
			mk_sense_buffer(scp, ABORTED_COMMAND, 0x10, 1);
			return illegal_condition_result;
2665
		} else if (sqcp->inj_dix) {
2666 2667 2668 2669
			mk_sense_buffer(scp, ILLEGAL_REQUEST, 0x10, 1);
			return illegal_condition_result;
		}
	}
2670
	return 0;
L
Linus Torvalds 已提交
2671 2672
}

2673
static void dump_sector(unsigned char *buf, int len)
2674
{
2675
	int i, j, n;
2676

2677
	pr_err(">>> Sector Dump <<<\n");
2678
	for (i = 0 ; i < len ; i += 16) {
2679
		char b[128];
2680

2681
		for (j = 0, n = 0; j < 16; j++) {
2682
			unsigned char c = buf[i+j];
2683

2684
			if (c >= 0x20 && c < 0x7e)
2685 2686
				n += scnprintf(b + n, sizeof(b) - n,
					       " %c ", buf[i+j]);
2687
			else
2688 2689
				n += scnprintf(b + n, sizeof(b) - n,
					       "%02x ", buf[i+j]);
2690
		}
2691
		pr_err("%04d: %s\n", i, b);
2692 2693 2694 2695
	}
}

static int prot_verify_write(struct scsi_cmnd *SCpnt, sector_t start_sec,
2696
			     unsigned int sectors, u32 ei_lba)
2697
{
2698
	int ret;
2699
	struct sd_dif_tuple *sdt;
2700
	void *daddr;
2701
	sector_t sector = start_sec;
2702
	int ppage_offset;
2703 2704 2705
	int dpage_offset;
	struct sg_mapping_iter diter;
	struct sg_mapping_iter piter;
2706 2707 2708 2709

	BUG_ON(scsi_sg_count(SCpnt) == 0);
	BUG_ON(scsi_prot_sg_count(SCpnt) == 0);

2710 2711 2712 2713 2714 2715 2716 2717 2718 2719 2720 2721 2722
	sg_miter_start(&piter, scsi_prot_sglist(SCpnt),
			scsi_prot_sg_count(SCpnt),
			SG_MITER_ATOMIC | SG_MITER_FROM_SG);
	sg_miter_start(&diter, scsi_sglist(SCpnt), scsi_sg_count(SCpnt),
			SG_MITER_ATOMIC | SG_MITER_FROM_SG);

	/* For each protection page */
	while (sg_miter_next(&piter)) {
		dpage_offset = 0;
		if (WARN_ON(!sg_miter_next(&diter))) {
			ret = 0x01;
			goto out;
		}
2723

2724 2725
		for (ppage_offset = 0; ppage_offset < piter.length;
		     ppage_offset += sizeof(struct sd_dif_tuple)) {
2726
			/* If we're at the end of the current
2727
			 * data page advance to the next one
2728
			 */
2729 2730 2731 2732 2733 2734
			if (dpage_offset >= diter.length) {
				if (WARN_ON(!sg_miter_next(&diter))) {
					ret = 0x01;
					goto out;
				}
				dpage_offset = 0;
2735 2736
			}

2737 2738
			sdt = piter.addr + ppage_offset;
			daddr = diter.addr + dpage_offset;
2739

2740
			ret = dif_verify(sdt, daddr, sector, ei_lba);
2741
			if (ret) {
2742
				dump_sector(daddr, sdebug_sector_size);
2743 2744 2745
				goto out;
			}

2746
			sector++;
2747
			ei_lba++;
2748
			dpage_offset += sdebug_sector_size;
2749
		}
2750 2751
		diter.consumed = dpage_offset;
		sg_miter_stop(&diter);
2752
	}
2753
	sg_miter_stop(&piter);
2754

2755
	dif_copy_prot(SCpnt, start_sec, sectors, false);
2756 2757 2758 2759 2760 2761
	dix_writes++;

	return 0;

out:
	dif_errors++;
2762 2763
	sg_miter_stop(&diter);
	sg_miter_stop(&piter);
2764 2765 2766
	return ret;
}

2767 2768
static unsigned long lba_to_map_index(sector_t lba)
{
2769 2770 2771
	if (sdebug_unmap_alignment)
		lba += sdebug_unmap_granularity - sdebug_unmap_alignment;
	sector_div(lba, sdebug_unmap_granularity);
2772 2773 2774 2775
	return lba;
}

static sector_t map_index_to_lba(unsigned long index)
2776
{
2777
	sector_t lba = index * sdebug_unmap_granularity;
2778

2779 2780
	if (sdebug_unmap_alignment)
		lba -= sdebug_unmap_granularity - sdebug_unmap_alignment;
2781
	return lba;
2782
}
2783

2784 2785 2786 2787 2788 2789
static unsigned int map_state(sector_t lba, unsigned int *num)
{
	sector_t end;
	unsigned int mapped;
	unsigned long index;
	unsigned long next;
2790

2791 2792
	index = lba_to_map_index(lba);
	mapped = test_bit(index, map_storep);
2793 2794

	if (mapped)
2795
		next = find_next_zero_bit(map_storep, map_size, index);
2796
	else
2797
		next = find_next_bit(map_storep, map_size, index);
2798

2799
	end = min_t(sector_t, sdebug_store_sectors,  map_index_to_lba(next));
2800 2801 2802 2803 2804 2805 2806 2807 2808
	*num = end - lba;
	return mapped;
}

static void map_region(sector_t lba, unsigned int len)
{
	sector_t end = lba + len;

	while (lba < end) {
2809
		unsigned long index = lba_to_map_index(lba);
2810

2811 2812
		if (index < map_size)
			set_bit(index, map_storep);
2813

2814
		lba = map_index_to_lba(index + 1);
2815 2816 2817 2818 2819 2820 2821 2822
	}
}

static void unmap_region(sector_t lba, unsigned int len)
{
	sector_t end = lba + len;

	while (lba < end) {
2823
		unsigned long index = lba_to_map_index(lba);
2824

2825
		if (lba == map_index_to_lba(index) &&
2826
		    lba + sdebug_unmap_granularity <= end &&
2827 2828
		    index < map_size) {
			clear_bit(index, map_storep);
2829
			if (sdebug_lbprz) {  /* for LBPRZ=2 return 0xff_s */
2830
				memset(fake_storep +
2831 2832
				       lba * sdebug_sector_size,
				       (sdebug_lbprz & 1) ? 0 : 0xff,
2833 2834
				       sdebug_sector_size *
				       sdebug_unmap_granularity);
2835
			}
2836 2837 2838
			if (dif_storep) {
				memset(dif_storep + lba, 0xff,
				       sizeof(*dif_storep) *
2839
				       sdebug_unmap_granularity);
2840
			}
2841
		}
2842
		lba = map_index_to_lba(index + 1);
2843 2844 2845
	}
}

2846
static int resp_write_dt0(struct scsi_cmnd *scp, struct sdebug_dev_info *devip)
L
Linus Torvalds 已提交
2847
{
2848 2849 2850 2851
	u8 *cmd = scp->cmnd;
	u64 lba;
	u32 num;
	u32 ei_lba;
L
Linus Torvalds 已提交
2852
	unsigned long iflags;
2853
	int ret;
2854
	bool check_prot;
L
Linus Torvalds 已提交
2855

2856 2857 2858 2859 2860 2861 2862 2863 2864 2865 2866 2867 2868 2869 2870 2871 2872 2873 2874 2875 2876 2877 2878 2879 2880 2881 2882 2883 2884 2885 2886 2887 2888 2889 2890 2891 2892 2893 2894
	switch (cmd[0]) {
	case WRITE_16:
		ei_lba = 0;
		lba = get_unaligned_be64(cmd + 2);
		num = get_unaligned_be32(cmd + 10);
		check_prot = true;
		break;
	case WRITE_10:
		ei_lba = 0;
		lba = get_unaligned_be32(cmd + 2);
		num = get_unaligned_be16(cmd + 7);
		check_prot = true;
		break;
	case WRITE_6:
		ei_lba = 0;
		lba = (u32)cmd[3] | (u32)cmd[2] << 8 |
		      (u32)(cmd[1] & 0x1f) << 16;
		num = (0 == cmd[4]) ? 256 : cmd[4];
		check_prot = true;
		break;
	case WRITE_12:
		ei_lba = 0;
		lba = get_unaligned_be32(cmd + 2);
		num = get_unaligned_be32(cmd + 6);
		check_prot = true;
		break;
	case 0x53:	/* XDWRITEREAD(10) */
		ei_lba = 0;
		lba = get_unaligned_be32(cmd + 2);
		num = get_unaligned_be16(cmd + 7);
		check_prot = false;
		break;
	default:	/* assume WRITE(32) */
		lba = get_unaligned_be64(cmd + 12);
		ei_lba = get_unaligned_be32(cmd + 20);
		num = get_unaligned_be32(cmd + 28);
		check_prot = false;
		break;
	}
2895
	if (unlikely(have_dif_prot && check_prot)) {
2896
		if (sdebug_dif == SD_DIF_TYPE2_PROTECTION &&
2897 2898 2899 2900
		    (cmd[1] & 0xe0)) {
			mk_sense_invalid_opcode(scp);
			return check_condition_result;
		}
2901 2902
		if ((sdebug_dif == SD_DIF_TYPE1_PROTECTION ||
		     sdebug_dif == SD_DIF_TYPE3_PROTECTION) &&
2903 2904 2905 2906 2907 2908
		    (cmd[1] & 0xe0) == 0)
			sdev_printk(KERN_ERR, scp->device, "Unprotected WR "
				    "to DIF device\n");
	}

	/* inline check_device_access_params() */
2909
	if (unlikely(lba + num > sdebug_capacity)) {
2910 2911 2912 2913
		mk_sense_buffer(scp, ILLEGAL_REQUEST, LBA_OUT_OF_RANGE, 0);
		return check_condition_result;
	}
	/* transfer length excessive (tie in to block limits VPD page) */
2914
	if (unlikely(num > sdebug_store_sectors)) {
2915 2916 2917 2918
		/* needs work to find which cdb byte 'num' comes from */
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INVALID_FIELD_IN_CDB, 0);
		return check_condition_result;
	}
L
Linus Torvalds 已提交
2919

2920 2921
	write_lock_irqsave(&atomic_rw, iflags);

2922
	/* DIX + T10 DIF */
2923
	if (unlikely(sdebug_dix && scsi_prot_sg_count(scp))) {
2924
		int prot_ret = prot_verify_write(scp, lba, num, ei_lba);
2925 2926

		if (prot_ret) {
2927
			write_unlock_irqrestore(&atomic_rw, iflags);
2928
			mk_sense_buffer(scp, ILLEGAL_REQUEST, 0x10, prot_ret);
2929 2930 2931 2932
			return illegal_condition_result;
		}
	}

2933
	ret = do_device_access(scp, lba, num, true);
2934
	if (unlikely(scsi_debug_lbp()))
2935
		map_region(lba, num);
L
Linus Torvalds 已提交
2936
	write_unlock_irqrestore(&atomic_rw, iflags);
2937
	if (unlikely(-1 == ret))
2938
		return DID_ERROR << 16;
2939 2940
	else if (unlikely(sdebug_verbose &&
			  (ret < (num * sdebug_sector_size))))
2941
		sdev_printk(KERN_INFO, scp->device,
2942
			    "%s: write: cdb indicated=%u, IO sent=%d bytes\n",
2943
			    my_name, num * sdebug_sector_size, ret);
2944

2945
	if (unlikely(sdebug_any_injecting_opt)) {
2946 2947
		struct sdebug_queued_cmd *sqcp =
				(struct sdebug_queued_cmd *)scp->host_scribble;
2948

2949 2950 2951 2952 2953 2954 2955 2956 2957 2958 2959 2960 2961
		if (sqcp) {
			if (sqcp->inj_recovered) {
				mk_sense_buffer(scp, RECOVERED_ERROR,
						THRESHOLD_EXCEEDED, 0);
				return check_condition_result;
			} else if (sqcp->inj_dif) {
				/* Logical block guard check failed */
				mk_sense_buffer(scp, ABORTED_COMMAND, 0x10, 1);
				return illegal_condition_result;
			} else if (sqcp->inj_dix) {
				mk_sense_buffer(scp, ILLEGAL_REQUEST, 0x10, 1);
				return illegal_condition_result;
			}
2962 2963
		}
	}
2964 2965 2966
	return 0;
}

2967 2968
static int resp_write_same(struct scsi_cmnd *scp, u64 lba, u32 num,
			   u32 ei_lba, bool unmap, bool ndob)
2969 2970 2971 2972
{
	unsigned long iflags;
	unsigned long long i;
	int ret;
2973
	u64 lba_off;
2974

2975
	ret = check_device_access_params(scp, lba, num);
2976 2977 2978 2979 2980
	if (ret)
		return ret;

	write_lock_irqsave(&atomic_rw, iflags);

2981
	if (unmap && scsi_debug_lbp()) {
2982 2983 2984 2985
		unmap_region(lba, num);
		goto out;
	}

2986
	lba_off = lba * sdebug_sector_size;
2987 2988
	/* if ndob then zero 1 logical block, else fetch 1 logical block */
	if (ndob) {
2989
		memset(fake_storep + lba_off, 0, sdebug_sector_size);
2990 2991
		ret = 0;
	} else
2992 2993
		ret = fetch_to_dev_buffer(scp, fake_storep + lba_off,
					  sdebug_sector_size);
2994 2995 2996

	if (-1 == ret) {
		write_unlock_irqrestore(&atomic_rw, iflags);
2997 2998
		return DID_ERROR << 16;
	} else if (sdebug_verbose && (ret < (num * sdebug_sector_size)))
2999
		sdev_printk(KERN_INFO, scp->device,
3000 3001
			    "%s: %s: cdb indicated=%u, IO sent=%d bytes\n",
			    my_name, "write same",
3002
			    num * sdebug_sector_size, ret);
3003 3004 3005

	/* Copy first sector to remaining blocks */
	for (i = 1 ; i < num ; i++)
3006 3007 3008
		memcpy(fake_storep + ((lba + i) * sdebug_sector_size),
		       fake_storep + lba_off,
		       sdebug_sector_size);
3009

3010
	if (scsi_debug_lbp())
3011 3012 3013 3014
		map_region(lba, num);
out:
	write_unlock_irqrestore(&atomic_rw, iflags);

L
Linus Torvalds 已提交
3015 3016 3017
	return 0;
}

3018 3019
static int resp_write_same_10(struct scsi_cmnd *scp,
			      struct sdebug_dev_info *devip)
3020 3021 3022 3023 3024 3025 3026 3027
{
	u8 *cmd = scp->cmnd;
	u32 lba;
	u16 num;
	u32 ei_lba = 0;
	bool unmap = false;

	if (cmd[1] & 0x8) {
3028
		if (sdebug_lbpws10 == 0) {
3029 3030 3031 3032 3033 3034 3035
			mk_sense_invalid_fld(scp, SDEB_IN_CDB, 1, 3);
			return check_condition_result;
		} else
			unmap = true;
	}
	lba = get_unaligned_be32(cmd + 2);
	num = get_unaligned_be16(cmd + 7);
3036
	if (num > sdebug_write_same_length) {
3037 3038 3039 3040 3041 3042
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 7, -1);
		return check_condition_result;
	}
	return resp_write_same(scp, lba, num, ei_lba, unmap, false);
}

3043 3044
static int resp_write_same_16(struct scsi_cmnd *scp,
			      struct sdebug_dev_info *devip)
3045 3046 3047 3048 3049 3050 3051 3052 3053
{
	u8 *cmd = scp->cmnd;
	u64 lba;
	u32 num;
	u32 ei_lba = 0;
	bool unmap = false;
	bool ndob = false;

	if (cmd[1] & 0x8) {	/* UNMAP */
3054
		if (sdebug_lbpws == 0) {
3055 3056 3057 3058 3059 3060 3061 3062 3063
			mk_sense_invalid_fld(scp, SDEB_IN_CDB, 1, 3);
			return check_condition_result;
		} else
			unmap = true;
	}
	if (cmd[1] & 0x1)  /* NDOB (no data-out buffer, assumes zeroes) */
		ndob = true;
	lba = get_unaligned_be64(cmd + 2);
	num = get_unaligned_be32(cmd + 10);
3064
	if (num > sdebug_write_same_length) {
3065 3066 3067 3068 3069 3070
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 10, -1);
		return check_condition_result;
	}
	return resp_write_same(scp, lba, num, ei_lba, unmap, ndob);
}

3071 3072 3073
/* Note the mode field is in the same position as the (lower) service action
 * field. For the Report supported operation codes command, SPC-4 suggests
 * each mode of this command should be reported separately; for future. */
3074 3075
static int resp_write_buffer(struct scsi_cmnd *scp,
			     struct sdebug_dev_info *devip)
3076 3077 3078 3079 3080 3081 3082 3083 3084 3085 3086 3087 3088 3089 3090 3091 3092 3093 3094 3095 3096 3097 3098 3099 3100 3101 3102 3103 3104 3105 3106 3107 3108 3109 3110 3111 3112 3113 3114 3115 3116 3117 3118 3119
{
	u8 *cmd = scp->cmnd;
	struct scsi_device *sdp = scp->device;
	struct sdebug_dev_info *dp;
	u8 mode;

	mode = cmd[1] & 0x1f;
	switch (mode) {
	case 0x4:	/* download microcode (MC) and activate (ACT) */
		/* set UAs on this device only */
		set_bit(SDEBUG_UA_BUS_RESET, devip->uas_bm);
		set_bit(SDEBUG_UA_MICROCODE_CHANGED, devip->uas_bm);
		break;
	case 0x5:	/* download MC, save and ACT */
		set_bit(SDEBUG_UA_MICROCODE_CHANGED_WO_RESET, devip->uas_bm);
		break;
	case 0x6:	/* download MC with offsets and ACT */
		/* set UAs on most devices (LUs) in this target */
		list_for_each_entry(dp,
				    &devip->sdbg_host->dev_info_list,
				    dev_list)
			if (dp->target == sdp->id) {
				set_bit(SDEBUG_UA_BUS_RESET, dp->uas_bm);
				if (devip != dp)
					set_bit(SDEBUG_UA_MICROCODE_CHANGED,
						dp->uas_bm);
			}
		break;
	case 0x7:	/* download MC with offsets, save, and ACT */
		/* set UA on all devices (LUs) in this target */
		list_for_each_entry(dp,
				    &devip->sdbg_host->dev_info_list,
				    dev_list)
			if (dp->target == sdp->id)
				set_bit(SDEBUG_UA_MICROCODE_CHANGED_WO_RESET,
					dp->uas_bm);
		break;
	default:
		/* do nothing for this command for other mode values */
		break;
	}
	return 0;
}

3120 3121
static int resp_comp_write(struct scsi_cmnd *scp,
			   struct sdebug_dev_info *devip)
3122 3123 3124 3125 3126 3127
{
	u8 *cmd = scp->cmnd;
	u8 *arr;
	u8 *fake_storep_hold;
	u64 lba;
	u32 dnum;
3128
	u32 lb_size = sdebug_sector_size;
3129 3130 3131
	u8 num;
	unsigned long iflags;
	int ret;
3132
	int retval = 0;
3133

3134
	lba = get_unaligned_be64(cmd + 2);
3135 3136 3137
	num = cmd[13];		/* 1 to a maximum of 255 logical blocks */
	if (0 == num)
		return 0;	/* degenerate case, not an error */
3138
	if (sdebug_dif == SD_DIF_TYPE2_PROTECTION &&
3139 3140 3141 3142
	    (cmd[1] & 0xe0)) {
		mk_sense_invalid_opcode(scp);
		return check_condition_result;
	}
3143 3144
	if ((sdebug_dif == SD_DIF_TYPE1_PROTECTION ||
	     sdebug_dif == SD_DIF_TYPE3_PROTECTION) &&
3145 3146 3147 3148 3149 3150 3151 3152 3153 3154 3155 3156 3157 3158 3159
	    (cmd[1] & 0xe0) == 0)
		sdev_printk(KERN_ERR, scp->device, "Unprotected WR "
			    "to DIF device\n");

	/* inline check_device_access_params() */
	if (lba + num > sdebug_capacity) {
		mk_sense_buffer(scp, ILLEGAL_REQUEST, LBA_OUT_OF_RANGE, 0);
		return check_condition_result;
	}
	/* transfer length excessive (tie in to block limits VPD page) */
	if (num > sdebug_store_sectors) {
		/* needs work to find which cdb byte 'num' comes from */
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INVALID_FIELD_IN_CDB, 0);
		return check_condition_result;
	}
3160 3161 3162 3163 3164 3165 3166
	dnum = 2 * num;
	arr = kzalloc(dnum * lb_size, GFP_ATOMIC);
	if (NULL == arr) {
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INSUFF_RES_ASC,
				INSUFF_RES_ASCQ);
		return check_condition_result;
	}
3167 3168 3169 3170 3171 3172 3173 3174 3175 3176

	write_lock_irqsave(&atomic_rw, iflags);

	/* trick do_device_access() to fetch both compare and write buffers
	 * from data-in into arr. Safe (atomic) since write_lock held. */
	fake_storep_hold = fake_storep;
	fake_storep = arr;
	ret = do_device_access(scp, 0, dnum, true);
	fake_storep = fake_storep_hold;
	if (ret == -1) {
3177 3178
		retval = DID_ERROR << 16;
		goto cleanup;
3179
	} else if (sdebug_verbose && (ret < (dnum * lb_size)))
3180 3181 3182 3183 3184
		sdev_printk(KERN_INFO, scp->device, "%s: compare_write: cdb "
			    "indicated=%u, IO sent=%d bytes\n", my_name,
			    dnum * lb_size, ret);
	if (!comp_write_worker(lba, num, arr)) {
		mk_sense_buffer(scp, MISCOMPARE, MISCOMPARE_VERIFY_ASC, 0);
3185 3186
		retval = check_condition_result;
		goto cleanup;
3187 3188 3189
	}
	if (scsi_debug_lbp())
		map_region(lba, num);
3190
cleanup:
3191
	write_unlock_irqrestore(&atomic_rw, iflags);
3192 3193
	kfree(arr);
	return retval;
3194 3195
}

3196 3197 3198 3199 3200 3201
struct unmap_block_desc {
	__be64	lba;
	__be32	blocks;
	__be32	__reserved;
};

3202
static int resp_unmap(struct scsi_cmnd *scp, struct sdebug_dev_info *devip)
3203 3204 3205 3206 3207
{
	unsigned char *buf;
	struct unmap_block_desc *desc;
	unsigned int i, payload_len, descriptors;
	int ret;
3208
	unsigned long iflags;
3209 3210


3211 3212 3213 3214
	if (!scsi_debug_lbp())
		return 0;	/* fib and say its done */
	payload_len = get_unaligned_be16(scp->cmnd + 7);
	BUG_ON(scsi_bufflen(scp) != payload_len);
3215 3216

	descriptors = (payload_len - 8) / 16;
3217
	if (descriptors > sdebug_unmap_max_desc) {
3218 3219 3220
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 7, -1);
		return check_condition_result;
	}
3221

3222
	buf = kzalloc(scsi_bufflen(scp), GFP_ATOMIC);
3223 3224 3225
	if (!buf) {
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INSUFF_RES_ASC,
				INSUFF_RES_ASCQ);
3226
		return check_condition_result;
3227
	}
3228

3229
	scsi_sg_copy_to_buffer(scp, buf, scsi_bufflen(scp));
3230 3231 3232 3233 3234 3235

	BUG_ON(get_unaligned_be16(&buf[0]) != payload_len - 2);
	BUG_ON(get_unaligned_be16(&buf[2]) != descriptors * 16);

	desc = (void *)&buf[8];

3236 3237
	write_lock_irqsave(&atomic_rw, iflags);

3238 3239 3240 3241
	for (i = 0 ; i < descriptors ; i++) {
		unsigned long long lba = get_unaligned_be64(&desc[i].lba);
		unsigned int num = get_unaligned_be32(&desc[i].blocks);

3242
		ret = check_device_access_params(scp, lba, num);
3243 3244 3245 3246 3247 3248 3249 3250 3251
		if (ret)
			goto out;

		unmap_region(lba, num);
	}

	ret = 0;

out:
3252
	write_unlock_irqrestore(&atomic_rw, iflags);
3253 3254 3255 3256 3257 3258 3259
	kfree(buf);

	return ret;
}

#define SDEBUG_GET_LBA_STATUS_LEN 32

3260 3261
static int resp_get_lba_status(struct scsi_cmnd *scp,
			       struct sdebug_dev_info *devip)
3262
{
3263 3264 3265 3266
	u8 *cmd = scp->cmnd;
	u64 lba;
	u32 alloc_len, mapped, num;
	u8 arr[SDEBUG_GET_LBA_STATUS_LEN];
3267 3268
	int ret;

3269 3270
	lba = get_unaligned_be64(cmd + 2);
	alloc_len = get_unaligned_be32(cmd + 10);
3271 3272 3273 3274

	if (alloc_len < 24)
		return 0;

3275
	ret = check_device_access_params(scp, lba, 1);
3276 3277 3278
	if (ret)
		return ret;

3279 3280 3281 3282 3283 3284 3285 3286 3287 3288 3289
	if (scsi_debug_lbp())
		mapped = map_state(lba, &num);
	else {
		mapped = 1;
		/* following just in case virtual_gb changed */
		sdebug_capacity = get_sdebug_capacity();
		if (sdebug_capacity - lba <= 0xffffffff)
			num = sdebug_capacity - lba;
		else
			num = 0xffffffff;
	}
3290 3291

	memset(arr, 0, SDEBUG_GET_LBA_STATUS_LEN);
3292 3293 3294 3295
	put_unaligned_be32(20, arr);		/* Parameter Data Length */
	put_unaligned_be64(lba, arr + 8);	/* LBA */
	put_unaligned_be32(num, arr + 16);	/* Number of blocks */
	arr[20] = !mapped;		/* prov_stat=0: mapped; 1: dealloc */
3296

3297
	return fill_from_dev_buffer(scp, arr, SDEBUG_GET_LBA_STATUS_LEN);
3298 3299
}

3300 3301
#define RL_BUCKET_ELEMS 8

3302 3303 3304 3305 3306 3307 3308 3309 3310 3311
/* Even though each pseudo target has a REPORT LUNS "well known logical unit"
 * (W-LUN), the normal Linux scanning logic does not associate it with a
 * device (e.g. /dev/sg7). The following magic will make that association:
 *   "cd /sys/class/scsi_host/host<n> ; echo '- - 49409' > scan"
 * where <n> is a host number. If there are multiple targets in a host then
 * the above will associate a W-LUN to each target. To only get a W-LUN
 * for target 2, then use "echo '- 2 49409' > scan" .
 */
static int resp_report_luns(struct scsi_cmnd *scp,
			    struct sdebug_dev_info *devip)
L
Linus Torvalds 已提交
3312
{
3313
	unsigned char *cmd = scp->cmnd;
L
Linus Torvalds 已提交
3314
	unsigned int alloc_len;
3315
	unsigned char select_report;
3316
	u64 lun;
3317
	struct scsi_lun *lun_p;
3318
	u8 arr[RL_BUCKET_ELEMS * sizeof(struct scsi_lun)];
3319 3320 3321 3322
	unsigned int lun_cnt;	/* normal LUN count (max: 256) */
	unsigned int wlun_cnt;	/* report luns W-LUN count */
	unsigned int tlun_cnt;	/* total LUN count */
	unsigned int rlen;	/* response length (in bytes) */
3323 3324 3325
	int k, j, n, res;
	unsigned int off_rsp = 0;
	const int sz_lun = sizeof(struct scsi_lun);
L
Linus Torvalds 已提交
3326

3327
	clear_luns_changed_on_target(devip);
3328 3329 3330 3331 3332 3333 3334

	select_report = cmd[2];
	alloc_len = get_unaligned_be32(cmd + 6);

	if (alloc_len < 4) {
		pr_err("alloc len too small %d\n", alloc_len);
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 6, -1);
L
Linus Torvalds 已提交
3335 3336
		return check_condition_result;
	}
3337 3338 3339 3340 3341 3342 3343

	switch (select_report) {
	case 0:		/* all LUNs apart from W-LUNs */
		lun_cnt = sdebug_max_luns;
		wlun_cnt = 0;
		break;
	case 1:		/* only W-LUNs */
D
Douglas Gilbert 已提交
3344
		lun_cnt = 0;
3345 3346 3347 3348 3349 3350 3351 3352 3353 3354 3355 3356 3357 3358 3359 3360
		wlun_cnt = 1;
		break;
	case 2:		/* all LUNs */
		lun_cnt = sdebug_max_luns;
		wlun_cnt = 1;
		break;
	case 0x10:	/* only administrative LUs */
	case 0x11:	/* see SPC-5 */
	case 0x12:	/* only subsiduary LUs owned by referenced LU */
	default:
		pr_debug("select report invalid %d\n", select_report);
		mk_sense_invalid_fld(scp, SDEB_IN_CDB, 2, -1);
		return check_condition_result;
	}

	if (sdebug_no_lun_0 && (lun_cnt > 0))
D
Douglas Gilbert 已提交
3361
		--lun_cnt;
3362 3363

	tlun_cnt = lun_cnt + wlun_cnt;
3364 3365
	rlen = tlun_cnt * sz_lun;	/* excluding 8 byte header */
	scsi_set_resid(scp, scsi_bufflen(scp));
3366 3367 3368
	pr_debug("select_report %d luns = %d wluns = %d no_lun0 %d\n",
		 select_report, lun_cnt, wlun_cnt, sdebug_no_lun_0);

3369
	/* loops rely on sizeof response header same as sizeof lun (both 8) */
3370
	lun = sdebug_no_lun_0 ? 1 : 0;
3371 3372 3373 3374 3375 3376 3377 3378 3379 3380 3381 3382 3383 3384 3385 3386 3387 3388 3389 3390 3391 3392 3393 3394 3395 3396 3397
	for (k = 0, j = 0, res = 0; true; ++k, j = 0) {
		memset(arr, 0, sizeof(arr));
		lun_p = (struct scsi_lun *)&arr[0];
		if (k == 0) {
			put_unaligned_be32(rlen, &arr[0]);
			++lun_p;
			j = 1;
		}
		for ( ; j < RL_BUCKET_ELEMS; ++j, ++lun_p) {
			if ((k * RL_BUCKET_ELEMS) + j > lun_cnt)
				break;
			int_to_scsilun(lun++, lun_p);
		}
		if (j < RL_BUCKET_ELEMS)
			break;
		n = j * sz_lun;
		res = p_fill_from_dev_buffer(scp, arr, n, off_rsp);
		if (res)
			return res;
		off_rsp += n;
	}
	if (wlun_cnt) {
		int_to_scsilun(SCSI_W_LUN_REPORT_LUNS, lun_p);
		++j;
	}
	if (j > 0)
		res = p_fill_from_dev_buffer(scp, arr, j * sz_lun, off_rsp);
3398
	return res;
L
Linus Torvalds 已提交
3399 3400
}

3401 3402 3403
static int resp_xdwriteread(struct scsi_cmnd *scp, unsigned long long lba,
			    unsigned int num, struct sdebug_dev_info *devip)
{
3404
	int j;
3405 3406 3407
	unsigned char *kaddr, *buf;
	unsigned int offset;
	struct scsi_data_buffer *sdb = scsi_in(scp);
3408
	struct sg_mapping_iter miter;
3409 3410

	/* better not to use temporary buffer. */
3411
	buf = kzalloc(scsi_bufflen(scp), GFP_ATOMIC);
3412
	if (!buf) {
3413 3414
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INSUFF_RES_ASC,
				INSUFF_RES_ASCQ);
3415 3416
		return check_condition_result;
	}
3417

3418
	scsi_sg_copy_to_buffer(scp, buf, scsi_bufflen(scp));
3419 3420

	offset = 0;
3421 3422
	sg_miter_start(&miter, sdb->table.sgl, sdb->table.nents,
			SG_MITER_ATOMIC | SG_MITER_TO_SG);
3423

3424 3425 3426 3427
	while (sg_miter_next(&miter)) {
		kaddr = miter.addr;
		for (j = 0; j < miter.length; j++)
			*(kaddr + j) ^= *(buf + offset + j);
3428

3429
		offset += miter.length;
3430
	}
3431
	sg_miter_stop(&miter);
3432 3433
	kfree(buf);

3434
	return 0;
3435 3436
}

3437 3438
static int resp_xdwriteread_10(struct scsi_cmnd *scp,
			       struct sdebug_dev_info *devip)
3439 3440 3441 3442 3443 3444 3445 3446 3447 3448 3449 3450 3451 3452 3453 3454 3455 3456 3457 3458 3459 3460 3461 3462
{
	u8 *cmd = scp->cmnd;
	u64 lba;
	u32 num;
	int errsts;

	if (!scsi_bidi_cmnd(scp)) {
		mk_sense_buffer(scp, ILLEGAL_REQUEST, INSUFF_RES_ASC,
				INSUFF_RES_ASCQ);
		return check_condition_result;
	}
	errsts = resp_read_dt0(scp, devip);
	if (errsts)
		return errsts;
	if (!(cmd[1] & 0x4)) {		/* DISABLE_WRITE is not set */
		errsts = resp_write_dt0(scp, devip);
		if (errsts)
			return errsts;
	}
	lba = get_unaligned_be32(cmd + 2);
	num = get_unaligned_be16(cmd + 7);
	return resp_xdwriteread(scp, lba, num, devip);
}

3463 3464 3465 3466 3467 3468 3469 3470 3471 3472 3473 3474 3475 3476 3477 3478 3479 3480 3481
static struct sdebug_queue *get_queue(struct scsi_cmnd *cmnd)
{
	struct sdebug_queue *sqp = sdebug_q_arr;

	if (sdebug_mq_active) {
		u32 tag = blk_mq_unique_tag(cmnd->request);
		u16 hwq = blk_mq_unique_tag_to_hwq(tag);

		if (unlikely(hwq >= submit_queues)) {
			pr_warn("Unexpected hwq=%d, apply modulo\n", hwq);
			hwq %= submit_queues;
		}
		pr_debug("tag=%u, hwq=%d\n", tag, hwq);
		return sqp + hwq;
	} else
		return sqp;
}

/* Queued (deferred) command completions converge here. */
3482
static void sdebug_q_cmd_complete(struct sdebug_defer *sd_dp)
L
Linus Torvalds 已提交
3483
{
3484
	int qc_idx;
3485
	int retiring = 0;
L
Linus Torvalds 已提交
3486
	unsigned long iflags;
3487
	struct sdebug_queue *sqp;
3488 3489 3490
	struct sdebug_queued_cmd *sqcp;
	struct scsi_cmnd *scp;
	struct sdebug_dev_info *devip;
L
Linus Torvalds 已提交
3491

3492 3493 3494 3495 3496 3497 3498 3499 3500
	qc_idx = sd_dp->qc_idx;
	sqp = sdebug_q_arr + sd_dp->sqa_idx;
	if (sdebug_statistics) {
		atomic_inc(&sdebug_completions);
		if (raw_smp_processor_id() != sd_dp->issuing_cpu)
			atomic_inc(&sdebug_miss_cpus);
	}
	if (unlikely((qc_idx < 0) || (qc_idx >= SDEBUG_CANQUEUE))) {
		pr_err("wild qc_idx=%d\n", qc_idx);
L
Linus Torvalds 已提交
3501 3502
		return;
	}
3503 3504
	spin_lock_irqsave(&sqp->qc_lock, iflags);
	sqcp = &sqp->qc_arr[qc_idx];
3505
	scp = sqcp->a_cmnd;
D
Douglas Gilbert 已提交
3506
	if (unlikely(scp == NULL)) {
3507 3508 3509
		spin_unlock_irqrestore(&sqp->qc_lock, iflags);
		pr_err("scp is NULL, sqa_idx=%d, qc_idx=%d\n",
		       sd_dp->sqa_idx, qc_idx);
3510 3511 3512
		return;
	}
	devip = (struct sdebug_dev_info *)scp->device->hostdata;
3513
	if (likely(devip))
3514 3515
		atomic_dec(&devip->num_in_q);
	else
3516
		pr_err("devip=NULL\n");
3517
	if (unlikely(atomic_read(&retired_max_queue) > 0))
3518 3519 3520
		retiring = 1;

	sqcp->a_cmnd = NULL;
3521 3522
	if (unlikely(!test_and_clear_bit(qc_idx, sqp->in_use_bm))) {
		spin_unlock_irqrestore(&sqp->qc_lock, iflags);
3523
		pr_err("Unexpected completion\n");
L
Linus Torvalds 已提交
3524 3525
		return;
	}
3526 3527 3528 3529 3530

	if (unlikely(retiring)) {	/* user has reduced max_queue */
		int k, retval;

		retval = atomic_read(&retired_max_queue);
3531 3532
		if (qc_idx >= retval) {
			spin_unlock_irqrestore(&sqp->qc_lock, iflags);
3533
			pr_err("index %d too large\n", retval);
3534 3535
			return;
		}
3536
		k = find_last_bit(sqp->in_use_bm, retval);
3537
		if ((k < sdebug_max_queue) || (k == retval))
3538 3539 3540
			atomic_set(&retired_max_queue, 0);
		else
			atomic_set(&retired_max_queue, k + 1);
L
Linus Torvalds 已提交
3541
	}
3542
	spin_unlock_irqrestore(&sqp->qc_lock, iflags);
3543
	scp->scsi_done(scp); /* callback to mid level */
L
Linus Torvalds 已提交
3544 3545
}

3546
/* When high resolution timer goes off this function is called. */
3547
static enum hrtimer_restart sdebug_q_cmd_hrt_complete(struct hrtimer *timer)
3548
{
3549 3550 3551
	struct sdebug_defer *sd_dp = container_of(timer, struct sdebug_defer,
						  hrt);
	sdebug_q_cmd_complete(sd_dp);
3552 3553
	return HRTIMER_NORESTART;
}
L
Linus Torvalds 已提交
3554

3555
/* When work queue schedules work, it calls this function. */
3556
static void sdebug_q_cmd_wq_complete(struct work_struct *work)
3557 3558 3559 3560 3561 3562
{
	struct sdebug_defer *sd_dp = container_of(work, struct sdebug_defer,
						  ew.work);
	sdebug_q_cmd_complete(sd_dp);
}

D
Douglas Gilbert 已提交
3563 3564 3565
static bool got_shared_uuid;
static uuid_be shared_uuid;

3566 3567
static struct sdebug_dev_info *sdebug_device_create(
			struct sdebug_host_info *sdbg_host, gfp_t flags)
3568 3569 3570 3571 3572
{
	struct sdebug_dev_info *devip;

	devip = kzalloc(sizeof(*devip), flags);
	if (devip) {
D
Douglas Gilbert 已提交
3573 3574 3575 3576 3577 3578 3579 3580 3581 3582 3583
		if (sdebug_uuid_ctl == 1)
			uuid_be_gen(&devip->lu_name);
		else if (sdebug_uuid_ctl == 2) {
			if (got_shared_uuid)
				devip->lu_name = shared_uuid;
			else {
				uuid_be_gen(&shared_uuid);
				got_shared_uuid = true;
				devip->lu_name = shared_uuid;
			}
		}
3584 3585 3586 3587 3588 3589
		devip->sdbg_host = sdbg_host;
		list_add_tail(&devip->dev_list, &sdbg_host->dev_info_list);
	}
	return devip;
}

3590
static struct sdebug_dev_info *find_build_dev_info(struct scsi_device *sdev)
L
Linus Torvalds 已提交
3591
{
3592 3593 3594
	struct sdebug_host_info *sdbg_host;
	struct sdebug_dev_info *open_devip = NULL;
	struct sdebug_dev_info *devip;
L
Linus Torvalds 已提交
3595

3596 3597
	sdbg_host = *(struct sdebug_host_info **)shost_priv(sdev->host);
	if (!sdbg_host) {
3598
		pr_err("Host info NULL\n");
L
Linus Torvalds 已提交
3599 3600 3601 3602 3603 3604 3605 3606 3607 3608 3609 3610
		return NULL;
        }
	list_for_each_entry(devip, &sdbg_host->dev_info_list, dev_list) {
		if ((devip->used) && (devip->channel == sdev->channel) &&
                    (devip->target == sdev->id) &&
                    (devip->lun == sdev->lun))
                        return devip;
		else {
			if ((!devip->used) && (!open_devip))
				open_devip = devip;
		}
	}
3611 3612 3613
	if (!open_devip) { /* try and make a new one */
		open_devip = sdebug_device_create(sdbg_host, GFP_ATOMIC);
		if (!open_devip) {
3614
			pr_err("out of memory at line %d\n", __LINE__);
L
Linus Torvalds 已提交
3615 3616 3617
			return NULL;
		}
	}
3618 3619 3620 3621 3622

	open_devip->channel = sdev->channel;
	open_devip->target = sdev->id;
	open_devip->lun = sdev->lun;
	open_devip->sdbg_host = sdbg_host;
3623 3624
	atomic_set(&open_devip->num_in_q, 0);
	set_bit(SDEBUG_UA_POR, open_devip->uas_bm);
3625
	open_devip->used = true;
3626
	return open_devip;
L
Linus Torvalds 已提交
3627 3628
}

3629
static int scsi_debug_slave_alloc(struct scsi_device *sdp)
L
Linus Torvalds 已提交
3630
{
3631
	if (sdebug_verbose)
3632
		pr_info("slave_alloc <%u %u %u %llu>\n",
3633
		       sdp->host->host_no, sdp->channel, sdp->id, sdp->lun);
N
Nick Piggin 已提交
3634
	queue_flag_set_unlocked(QUEUE_FLAG_BIDI, sdp->request_queue);
3635 3636
	return 0;
}
L
Linus Torvalds 已提交
3637

3638 3639
static int scsi_debug_slave_configure(struct scsi_device *sdp)
{
3640 3641
	struct sdebug_dev_info *devip =
			(struct sdebug_dev_info *)sdp->hostdata;
3642

3643
	if (sdebug_verbose)
3644
		pr_info("slave_configure <%u %u %u %llu>\n",
3645
		       sdp->host->host_no, sdp->channel, sdp->id, sdp->lun);
D
Douglas Gilbert 已提交
3646 3647 3648
	if (sdp->host->max_cmd_len != SDEBUG_MAX_CMD_LEN)
		sdp->host->max_cmd_len = SDEBUG_MAX_CMD_LEN;
	if (devip == NULL) {
3649
		devip = find_build_dev_info(sdp);
D
Douglas Gilbert 已提交
3650
		if (devip == NULL)
3651 3652
			return 1;  /* no resources, will be marked offline */
	}
3653
	sdp->hostdata = devip;
3654
	blk_queue_max_segment_size(sdp->request_queue, -1U);
3655
	if (sdebug_no_uld)
3656
		sdp->no_uld_attach = 1;
3657 3658 3659 3660 3661 3662 3663
	return 0;
}

static void scsi_debug_slave_destroy(struct scsi_device *sdp)
{
	struct sdebug_dev_info *devip =
		(struct sdebug_dev_info *)sdp->hostdata;
3664

3665
	if (sdebug_verbose)
3666
		pr_info("slave_destroy <%u %u %u %llu>\n",
3667 3668
		       sdp->host->host_no, sdp->channel, sdp->id, sdp->lun);
	if (devip) {
L
Lucas De Marchi 已提交
3669
		/* make this slot available for re-use */
3670
		devip->used = false;
3671 3672 3673 3674
		sdp->hostdata = NULL;
	}
}

3675 3676 3677 3678 3679 3680 3681 3682 3683 3684
static void stop_qc_helper(struct sdebug_defer *sd_dp)
{
	if (!sd_dp)
		return;
	if ((sdebug_jdelay > 0) || (sdebug_ndelay > 0))
		hrtimer_cancel(&sd_dp->hrt);
	else if (sdebug_jdelay < 0)
		cancel_work_sync(&sd_dp->ew.work);
}

3685 3686 3687
/* If @cmnd found deletes its timer or work queue and returns true; else
   returns false */
static bool stop_queued_cmnd(struct scsi_cmnd *cmnd)
3688 3689
{
	unsigned long iflags;
3690 3691
	int j, k, qmax, r_qmax;
	struct sdebug_queue *sqp;
3692
	struct sdebug_queued_cmd *sqcp;
3693
	struct sdebug_dev_info *devip;
3694
	struct sdebug_defer *sd_dp;
3695

3696 3697 3698 3699 3700 3701 3702 3703 3704 3705 3706 3707 3708 3709 3710 3711 3712 3713 3714 3715 3716 3717
	for (j = 0, sqp = sdebug_q_arr; j < submit_queues; ++j, ++sqp) {
		spin_lock_irqsave(&sqp->qc_lock, iflags);
		qmax = sdebug_max_queue;
		r_qmax = atomic_read(&retired_max_queue);
		if (r_qmax > qmax)
			qmax = r_qmax;
		for (k = 0; k < qmax; ++k) {
			if (test_bit(k, sqp->in_use_bm)) {
				sqcp = &sqp->qc_arr[k];
				if (cmnd != sqcp->a_cmnd)
					continue;
				/* found */
				devip = (struct sdebug_dev_info *)
						cmnd->device->hostdata;
				if (devip)
					atomic_dec(&devip->num_in_q);
				sqcp->a_cmnd = NULL;
				sd_dp = sqcp->sd_dp;
				spin_unlock_irqrestore(&sqp->qc_lock, iflags);
				stop_qc_helper(sd_dp);
				clear_bit(k, sqp->in_use_bm);
				return true;
3718
			}
3719
		}
3720
		spin_unlock_irqrestore(&sqp->qc_lock, iflags);
3721
	}
3722
	return false;
3723 3724
}

3725
/* Deletes (stops) timers or work queues of all queued commands */
3726 3727 3728
static void stop_all_queued(void)
{
	unsigned long iflags;
3729 3730
	int j, k;
	struct sdebug_queue *sqp;
3731
	struct sdebug_queued_cmd *sqcp;
3732
	struct sdebug_dev_info *devip;
3733
	struct sdebug_defer *sd_dp;
3734

3735 3736 3737 3738 3739 3740 3741 3742 3743 3744 3745 3746 3747 3748 3749 3750 3751
	for (j = 0, sqp = sdebug_q_arr; j < submit_queues; ++j, ++sqp) {
		spin_lock_irqsave(&sqp->qc_lock, iflags);
		for (k = 0; k < SDEBUG_CANQUEUE; ++k) {
			if (test_bit(k, sqp->in_use_bm)) {
				sqcp = &sqp->qc_arr[k];
				if (sqcp->a_cmnd == NULL)
					continue;
				devip = (struct sdebug_dev_info *)
					sqcp->a_cmnd->device->hostdata;
				if (devip)
					atomic_dec(&devip->num_in_q);
				sqcp->a_cmnd = NULL;
				sd_dp = sqcp->sd_dp;
				spin_unlock_irqrestore(&sqp->qc_lock, iflags);
				stop_qc_helper(sd_dp);
				clear_bit(k, sqp->in_use_bm);
				spin_lock_irqsave(&sqp->qc_lock, iflags);
3752
			}
3753
		}
3754
		spin_unlock_irqrestore(&sqp->qc_lock, iflags);
3755
	}
L
Linus Torvalds 已提交
3756 3757
}

3758 3759
/* Free queued command memory on heap */
static void free_all_queued(void)
L
Linus Torvalds 已提交
3760
{
3761 3762
	int j, k;
	struct sdebug_queue *sqp;
3763 3764
	struct sdebug_queued_cmd *sqcp;

3765 3766 3767 3768 3769 3770
	for (j = 0, sqp = sdebug_q_arr; j < submit_queues; ++j, ++sqp) {
		for (k = 0; k < SDEBUG_CANQUEUE; ++k) {
			sqcp = &sqp->qc_arr[k];
			kfree(sqcp->sd_dp);
			sqcp->sd_dp = NULL;
		}
3771
	}
L
Linus Torvalds 已提交
3772 3773
}

3774
static int scsi_debug_abort(struct scsi_cmnd *SCpnt)
L
Linus Torvalds 已提交
3775
{
3776 3777
	bool ok;

3778 3779
	++num_aborts;
	if (SCpnt) {
3780 3781 3782 3783 3784
		ok = stop_queued_cmnd(SCpnt);
		if (SCpnt->device && (SDEBUG_OPT_ALL_NOISE & sdebug_opts))
			sdev_printk(KERN_INFO, SCpnt->device,
				    "%s: command%s found\n", __func__,
				    ok ? "" : " not");
3785 3786
	}
	return SUCCESS;
L
Linus Torvalds 已提交
3787 3788 3789 3790 3791
}

static int scsi_debug_device_reset(struct scsi_cmnd * SCpnt)
{
	++num_dev_resets;
3792 3793
	if (SCpnt && SCpnt->device) {
		struct scsi_device *sdp = SCpnt->device;
3794 3795
		struct sdebug_dev_info *devip =
				(struct sdebug_dev_info *)sdp->hostdata;
3796

3797
		if (SDEBUG_OPT_ALL_NOISE & sdebug_opts)
3798
			sdev_printk(KERN_INFO, sdp, "%s\n", __func__);
L
Linus Torvalds 已提交
3799
		if (devip)
3800 3801 3802 3803 3804 3805 3806 3807 3808 3809 3810 3811 3812 3813 3814 3815 3816 3817 3818
			set_bit(SDEBUG_UA_POR, devip->uas_bm);
	}
	return SUCCESS;
}

static int scsi_debug_target_reset(struct scsi_cmnd *SCpnt)
{
	struct sdebug_host_info *sdbg_host;
	struct sdebug_dev_info *devip;
	struct scsi_device *sdp;
	struct Scsi_Host *hp;
	int k = 0;

	++num_target_resets;
	if (!SCpnt)
		goto lie;
	sdp = SCpnt->device;
	if (!sdp)
		goto lie;
3819
	if (SDEBUG_OPT_ALL_NOISE & sdebug_opts)
3820 3821 3822 3823 3824 3825 3826 3827 3828 3829 3830 3831 3832
		sdev_printk(KERN_INFO, sdp, "%s\n", __func__);
	hp = sdp->host;
	if (!hp)
		goto lie;
	sdbg_host = *(struct sdebug_host_info **)shost_priv(hp);
	if (sdbg_host) {
		list_for_each_entry(devip,
				    &sdbg_host->dev_info_list,
				    dev_list)
			if (devip->target == sdp->id) {
				set_bit(SDEBUG_UA_BUS_RESET, devip->uas_bm);
				++k;
			}
L
Linus Torvalds 已提交
3833
	}
3834
	if (SDEBUG_OPT_RESET_NOISE & sdebug_opts)
3835 3836 3837
		sdev_printk(KERN_INFO, sdp,
			    "%s: %d device(s) found in target\n", __func__, k);
lie:
L
Linus Torvalds 已提交
3838 3839 3840 3841 3842 3843
	return SUCCESS;
}

static int scsi_debug_bus_reset(struct scsi_cmnd * SCpnt)
{
	struct sdebug_host_info *sdbg_host;
3844
	struct sdebug_dev_info *devip;
L
Linus Torvalds 已提交
3845 3846
        struct scsi_device * sdp;
        struct Scsi_Host * hp;
3847
	int k = 0;
L
Linus Torvalds 已提交
3848 3849

	++num_bus_resets;
3850 3851 3852
	if (!(SCpnt && SCpnt->device))
		goto lie;
	sdp = SCpnt->device;
3853
	if (SDEBUG_OPT_ALL_NOISE & sdebug_opts)
3854 3855 3856
		sdev_printk(KERN_INFO, sdp, "%s\n", __func__);
	hp = sdp->host;
	if (hp) {
3857
		sdbg_host = *(struct sdebug_host_info **)shost_priv(hp);
L
Linus Torvalds 已提交
3858
		if (sdbg_host) {
3859
			list_for_each_entry(devip,
L
Linus Torvalds 已提交
3860
                                            &sdbg_host->dev_info_list,
3861 3862 3863 3864
					    dev_list) {
				set_bit(SDEBUG_UA_BUS_RESET, devip->uas_bm);
				++k;
			}
L
Linus Torvalds 已提交
3865 3866
		}
	}
3867
	if (SDEBUG_OPT_RESET_NOISE & sdebug_opts)
3868 3869 3870
		sdev_printk(KERN_INFO, sdp,
			    "%s: %d device(s) found in host\n", __func__, k);
lie:
L
Linus Torvalds 已提交
3871 3872 3873 3874 3875 3876
	return SUCCESS;
}

static int scsi_debug_host_reset(struct scsi_cmnd * SCpnt)
{
	struct sdebug_host_info * sdbg_host;
3877 3878
	struct sdebug_dev_info *devip;
	int k = 0;
L
Linus Torvalds 已提交
3879 3880

	++num_host_resets;
3881
	if ((SCpnt->device) && (SDEBUG_OPT_ALL_NOISE & sdebug_opts))
3882
		sdev_printk(KERN_INFO, SCpnt->device, "%s\n", __func__);
L
Linus Torvalds 已提交
3883 3884
        spin_lock(&sdebug_host_list_lock);
        list_for_each_entry(sdbg_host, &sdebug_host_list, host_list) {
3885 3886 3887 3888 3889
		list_for_each_entry(devip, &sdbg_host->dev_info_list,
				    dev_list) {
			set_bit(SDEBUG_UA_BUS_RESET, devip->uas_bm);
			++k;
		}
L
Linus Torvalds 已提交
3890 3891 3892
        }
        spin_unlock(&sdebug_host_list_lock);
	stop_all_queued();
3893
	if (SDEBUG_OPT_RESET_NOISE & sdebug_opts)
3894 3895
		sdev_printk(KERN_INFO, SCpnt->device,
			    "%s: %d device(s) found\n", __func__, k);
L
Linus Torvalds 已提交
3896 3897 3898
	return SUCCESS;
}

3899
static void __init sdebug_build_parts(unsigned char *ramp,
3900
				      unsigned long store_size)
L
Linus Torvalds 已提交
3901 3902 3903 3904 3905 3906 3907
{
	struct partition * pp;
	int starts[SDEBUG_MAX_PARTS + 2];
	int sectors_per_part, num_sectors, k;
	int heads_by_sects, start_sec, end_sec;

	/* assume partition table already zeroed */
3908
	if ((sdebug_num_parts < 1) || (store_size < 1048576))
L
Linus Torvalds 已提交
3909
		return;
3910 3911
	if (sdebug_num_parts > SDEBUG_MAX_PARTS) {
		sdebug_num_parts = SDEBUG_MAX_PARTS;
3912
		pr_warn("reducing partitions to %d\n", SDEBUG_MAX_PARTS);
L
Linus Torvalds 已提交
3913
	}
D
Douglas Gilbert 已提交
3914
	num_sectors = (int)sdebug_store_sectors;
L
Linus Torvalds 已提交
3915
	sectors_per_part = (num_sectors - sdebug_sectors_per)
3916
			   / sdebug_num_parts;
L
Linus Torvalds 已提交
3917 3918
	heads_by_sects = sdebug_heads * sdebug_sectors_per;
        starts[0] = sdebug_sectors_per;
3919
	for (k = 1; k < sdebug_num_parts; ++k)
L
Linus Torvalds 已提交
3920 3921
		starts[k] = ((k * sectors_per_part) / heads_by_sects)
			    * heads_by_sects;
3922 3923
	starts[sdebug_num_parts] = num_sectors;
	starts[sdebug_num_parts + 1] = 0;
L
Linus Torvalds 已提交
3924 3925 3926 3927 3928 3929 3930 3931 3932 3933 3934 3935 3936 3937 3938 3939 3940 3941 3942

	ramp[510] = 0x55;	/* magic partition markings */
	ramp[511] = 0xAA;
	pp = (struct partition *)(ramp + 0x1be);
	for (k = 0; starts[k + 1]; ++k, ++pp) {
		start_sec = starts[k];
		end_sec = starts[k + 1] - 1;
		pp->boot_ind = 0;

		pp->cyl = start_sec / heads_by_sects;
		pp->head = (start_sec - (pp->cyl * heads_by_sects))
			   / sdebug_sectors_per;
		pp->sector = (start_sec % sdebug_sectors_per) + 1;

		pp->end_cyl = end_sec / heads_by_sects;
		pp->end_head = (end_sec - (pp->end_cyl * heads_by_sects))
			       / sdebug_sectors_per;
		pp->end_sector = (end_sec % sdebug_sectors_per) + 1;

3943 3944
		pp->start_sect = cpu_to_le32(start_sec);
		pp->nr_sects = cpu_to_le32(end_sec - start_sec + 1);
L
Linus Torvalds 已提交
3945 3946 3947 3948
		pp->sys_ind = 0x83;	/* plain Linux partition */
	}
}

3949 3950 3951 3952 3953 3954 3955 3956 3957 3958 3959 3960 3961 3962 3963 3964 3965 3966 3967 3968 3969 3970 3971 3972 3973 3974 3975 3976 3977 3978 3979 3980 3981 3982 3983 3984 3985 3986 3987 3988 3989 3990 3991 3992 3993 3994 3995 3996 3997 3998
static void block_unblock_all_queues(bool block)
{
	int j;
	struct sdebug_queue *sqp;

	for (j = 0, sqp = sdebug_q_arr; j < submit_queues; ++j, ++sqp)
		atomic_set(&sqp->blocked, (int)block);
}

/* Adjust (by rounding down) the sdebug_cmnd_count so abs(every_nth)-1
 * commands will be processed normally before triggers occur.
 */
static void tweak_cmnd_count(void)
{
	int count, modulo;

	modulo = abs(sdebug_every_nth);
	if (modulo < 2)
		return;
	block_unblock_all_queues(true);
	count = atomic_read(&sdebug_cmnd_count);
	atomic_set(&sdebug_cmnd_count, (count / modulo) * modulo);
	block_unblock_all_queues(false);
}

static void clear_queue_stats(void)
{
	atomic_set(&sdebug_cmnd_count, 0);
	atomic_set(&sdebug_completions, 0);
	atomic_set(&sdebug_miss_cpus, 0);
	atomic_set(&sdebug_a_tsf, 0);
}

static void setup_inject(struct sdebug_queue *sqp,
			 struct sdebug_queued_cmd *sqcp)
{
	if ((atomic_read(&sdebug_cmnd_count) % abs(sdebug_every_nth)) > 0)
		return;
	sqcp->inj_recovered = !!(SDEBUG_OPT_RECOVERED_ERR & sdebug_opts);
	sqcp->inj_transport = !!(SDEBUG_OPT_TRANSPORT_ERR & sdebug_opts);
	sqcp->inj_dif = !!(SDEBUG_OPT_DIF_ERR & sdebug_opts);
	sqcp->inj_dix = !!(SDEBUG_OPT_DIX_ERR & sdebug_opts);
	sqcp->inj_short = !!(SDEBUG_OPT_SHORT_TRANSFER & sdebug_opts);
}

/* Complete the processing of the thread that queued a SCSI command to this
 * driver. It either completes the command by calling cmnd_done() or
 * schedules a hr timer or work queue then returns 0. Returns
 * SCSI_MLQUEUE_HOST_BUSY if temporarily out of resources.
 */
3999 4000
static int schedule_resp(struct scsi_cmnd *cmnd, struct sdebug_dev_info *devip,
			 int scsi_result, int delta_jiff)
L
Linus Torvalds 已提交
4001
{
4002
	unsigned long iflags;
4003
	int k, num_in_q, qdepth, inject;
4004 4005
	struct sdebug_queue *sqp;
	struct sdebug_queued_cmd *sqcp;
4006
	struct scsi_device *sdp;
4007
	struct sdebug_defer *sd_dp;
4008

D
Douglas Gilbert 已提交
4009 4010
	if (unlikely(devip == NULL)) {
		if (scsi_result == 0)
4011 4012
			scsi_result = DID_NO_CONNECT << 16;
		goto respond_in_thread;
4013
	}
4014 4015
	sdp = cmnd->device;

4016
	if (unlikely(sdebug_verbose && scsi_result))
4017 4018
		sdev_printk(KERN_INFO, sdp, "%s: non-zero result=0x%x\n",
			    __func__, scsi_result);
4019 4020
	if (delta_jiff == 0)
		goto respond_in_thread;
L
Linus Torvalds 已提交
4021

4022
	/* schedule the response at a later time if resources permit */
4023 4024 4025 4026 4027 4028
	sqp = get_queue(cmnd);
	spin_lock_irqsave(&sqp->qc_lock, iflags);
	if (unlikely(atomic_read(&sqp->blocked))) {
		spin_unlock_irqrestore(&sqp->qc_lock, iflags);
		return SCSI_MLQUEUE_HOST_BUSY;
	}
4029 4030 4031
	num_in_q = atomic_read(&devip->num_in_q);
	qdepth = cmnd->device->queue_depth;
	inject = 0;
4032
	if (unlikely((qdepth > 0) && (num_in_q >= qdepth))) {
4033
		if (scsi_result) {
4034
			spin_unlock_irqrestore(&sqp->qc_lock, iflags);
4035 4036 4037
			goto respond_in_thread;
		} else
			scsi_result = device_qfull_result;
4038
	} else if (unlikely(sdebug_every_nth &&
4039 4040
			    (SDEBUG_OPT_RARE_TSF & sdebug_opts) &&
			    (scsi_result == 0))) {
4041 4042
		if ((num_in_q == (qdepth - 1)) &&
		    (atomic_inc_return(&sdebug_a_tsf) >=
4043
		     abs(sdebug_every_nth))) {
4044 4045
			atomic_set(&sdebug_a_tsf, 0);
			inject = 1;
4046
			scsi_result = device_qfull_result;
L
Linus Torvalds 已提交
4047 4048 4049
		}
	}

4050
	k = find_first_zero_bit(sqp->in_use_bm, sdebug_max_queue);
4051
	if (unlikely(k >= sdebug_max_queue)) {
4052
		spin_unlock_irqrestore(&sqp->qc_lock, iflags);
4053 4054
		if (scsi_result)
			goto respond_in_thread;
4055
		else if (SDEBUG_OPT_ALL_TSF & sdebug_opts)
4056
			scsi_result = device_qfull_result;
4057
		if (SDEBUG_OPT_Q_NOISE & sdebug_opts)
4058
			sdev_printk(KERN_INFO, sdp,
4059
				    "%s: max_queue=%d exceeded, %s\n",
4060
				    __func__, sdebug_max_queue,
4061 4062 4063 4064 4065
				    (scsi_result ?  "status: TASK SET FULL" :
						    "report: host busy"));
		if (scsi_result)
			goto respond_in_thread;
		else
4066 4067
			return SCSI_MLQUEUE_HOST_BUSY;
	}
4068
	__set_bit(k, sqp->in_use_bm);
4069
	atomic_inc(&devip->num_in_q);
4070
	sqcp = &sqp->qc_arr[k];
4071
	sqcp->a_cmnd = cmnd;
4072
	cmnd->host_scribble = (unsigned char *)sqcp;
4073
	cmnd->result = scsi_result;
4074
	sd_dp = sqcp->sd_dp;
4075 4076 4077
	spin_unlock_irqrestore(&sqp->qc_lock, iflags);
	if (unlikely(sdebug_every_nth && sdebug_any_injecting_opt))
		setup_inject(sqp, sqcp);
D
Douglas Gilbert 已提交
4078
	if (delta_jiff > 0 || sdebug_ndelay > 0) {
4079
		ktime_t kt;
4080

4081 4082 4083 4084 4085 4086 4087
		if (delta_jiff > 0) {
			struct timespec ts;

			jiffies_to_timespec(delta_jiff, &ts);
			kt = ktime_set(ts.tv_sec, ts.tv_nsec);
		} else
			kt = ktime_set(0, sdebug_ndelay);
4088 4089 4090
		if (NULL == sd_dp) {
			sd_dp = kzalloc(sizeof(*sd_dp), GFP_ATOMIC);
			if (NULL == sd_dp)
4091
				return SCSI_MLQUEUE_HOST_BUSY;
4092 4093
			sqcp->sd_dp = sd_dp;
			hrtimer_init(&sd_dp->hrt, CLOCK_MONOTONIC,
4094
				     HRTIMER_MODE_REL_PINNED);
4095
			sd_dp->hrt.function = sdebug_q_cmd_hrt_complete;
4096 4097
			sd_dp->sqa_idx = sqp - sdebug_q_arr;
			sd_dp->qc_idx = k;
L
Linus Torvalds 已提交
4098
		}
4099 4100 4101 4102
		if (sdebug_statistics)
			sd_dp->issuing_cpu = raw_smp_processor_id();
		hrtimer_start(&sd_dp->hrt, kt, HRTIMER_MODE_REL_PINNED);
	} else {	/* jdelay < 0, use work queue */
4103 4104 4105
		if (NULL == sd_dp) {
			sd_dp = kzalloc(sizeof(*sqcp->sd_dp), GFP_ATOMIC);
			if (NULL == sd_dp)
4106
				return SCSI_MLQUEUE_HOST_BUSY;
4107
			sqcp->sd_dp = sd_dp;
4108 4109
			sd_dp->sqa_idx = sqp - sdebug_q_arr;
			sd_dp->qc_idx = k;
4110
			INIT_WORK(&sd_dp->ew.work, sdebug_q_cmd_wq_complete);
4111
		}
4112 4113
		if (sdebug_statistics)
			sd_dp->issuing_cpu = raw_smp_processor_id();
4114
		schedule_work(&sd_dp->ew.work);
L
Linus Torvalds 已提交
4115
	}
4116 4117
	if (unlikely((SDEBUG_OPT_Q_NOISE & sdebug_opts) &&
		     (scsi_result == device_qfull_result)))
4118 4119 4120 4121 4122
		sdev_printk(KERN_INFO, sdp,
			    "%s: num_in_q=%d +1, %s%s\n", __func__,
			    num_in_q, (inject ? "<inject> " : ""),
			    "status: TASK SET FULL");
	return 0;
4123 4124 4125 4126 4127

respond_in_thread:	/* call back to mid-layer using invocation thread */
	cmnd->result = scsi_result;
	cmnd->scsi_done(cmnd);
	return 0;
L
Linus Torvalds 已提交
4128
}
4129

D
Douglas Gilbert 已提交
4130 4131 4132 4133 4134 4135
/* Note: The following macros create attribute files in the
   /sys/module/scsi_debug/parameters directory. Unfortunately this
   driver is unaware of a change and cannot trigger auxiliary actions
   as it can when the corresponding attribute in the
   /sys/bus/pseudo/drivers/scsi_debug directory is changed.
 */
4136 4137 4138
module_param_named(add_host, sdebug_add_host, int, S_IRUGO | S_IWUSR);
module_param_named(ato, sdebug_ato, int, S_IRUGO);
module_param_named(clustering, sdebug_clustering, bool, S_IRUGO | S_IWUSR);
4139
module_param_named(delay, sdebug_jdelay, int, S_IRUGO | S_IWUSR);
4140 4141 4142 4143 4144 4145 4146 4147 4148 4149 4150 4151 4152 4153 4154 4155 4156 4157 4158 4159 4160 4161 4162 4163 4164 4165 4166
module_param_named(dev_size_mb, sdebug_dev_size_mb, int, S_IRUGO);
module_param_named(dif, sdebug_dif, int, S_IRUGO);
module_param_named(dix, sdebug_dix, int, S_IRUGO);
module_param_named(dsense, sdebug_dsense, int, S_IRUGO | S_IWUSR);
module_param_named(every_nth, sdebug_every_nth, int, S_IRUGO | S_IWUSR);
module_param_named(fake_rw, sdebug_fake_rw, int, S_IRUGO | S_IWUSR);
module_param_named(guard, sdebug_guard, uint, S_IRUGO);
module_param_named(host_lock, sdebug_host_lock, bool, S_IRUGO | S_IWUSR);
module_param_named(lbpu, sdebug_lbpu, int, S_IRUGO);
module_param_named(lbpws, sdebug_lbpws, int, S_IRUGO);
module_param_named(lbpws10, sdebug_lbpws10, int, S_IRUGO);
module_param_named(lbprz, sdebug_lbprz, int, S_IRUGO);
module_param_named(lowest_aligned, sdebug_lowest_aligned, int, S_IRUGO);
module_param_named(max_luns, sdebug_max_luns, int, S_IRUGO | S_IWUSR);
module_param_named(max_queue, sdebug_max_queue, int, S_IRUGO | S_IWUSR);
module_param_named(ndelay, sdebug_ndelay, int, S_IRUGO | S_IWUSR);
module_param_named(no_lun_0, sdebug_no_lun_0, int, S_IRUGO | S_IWUSR);
module_param_named(no_uld, sdebug_no_uld, int, S_IRUGO);
module_param_named(num_parts, sdebug_num_parts, int, S_IRUGO);
module_param_named(num_tgts, sdebug_num_tgts, int, S_IRUGO | S_IWUSR);
module_param_named(opt_blks, sdebug_opt_blks, int, S_IRUGO);
module_param_named(opts, sdebug_opts, int, S_IRUGO | S_IWUSR);
module_param_named(physblk_exp, sdebug_physblk_exp, int, S_IRUGO);
module_param_named(ptype, sdebug_ptype, int, S_IRUGO | S_IWUSR);
module_param_named(removable, sdebug_removable, bool, S_IRUGO | S_IWUSR);
module_param_named(scsi_level, sdebug_scsi_level, int, S_IRUGO);
module_param_named(sector_size, sdebug_sector_size, int, S_IRUGO);
4167
module_param_named(statistics, sdebug_statistics, bool, S_IRUGO | S_IWUSR);
4168
module_param_named(strict, sdebug_strict, bool, S_IRUGO | S_IWUSR);
4169
module_param_named(submit_queues, submit_queues, int, S_IRUGO);
4170 4171 4172 4173 4174
module_param_named(unmap_alignment, sdebug_unmap_alignment, int, S_IRUGO);
module_param_named(unmap_granularity, sdebug_unmap_granularity, int, S_IRUGO);
module_param_named(unmap_max_blocks, sdebug_unmap_max_blocks, int, S_IRUGO);
module_param_named(unmap_max_desc, sdebug_unmap_max_desc, int, S_IRUGO);
module_param_named(virtual_gb, sdebug_virtual_gb, int, S_IRUGO | S_IWUSR);
D
Douglas Gilbert 已提交
4175
module_param_named(uuid_ctl, sdebug_uuid_ctl, int, S_IRUGO);
4176
module_param_named(vpd_use_hostno, sdebug_vpd_use_hostno, int,
4177
		   S_IRUGO | S_IWUSR);
4178
module_param_named(write_same_length, sdebug_write_same_length, int,
4179
		   S_IRUGO | S_IWUSR);
L
Linus Torvalds 已提交
4180 4181 4182 4183

MODULE_AUTHOR("Eric Youngdale + Douglas Gilbert");
MODULE_DESCRIPTION("SCSI debug adapter driver");
MODULE_LICENSE("GPL");
D
Douglas Gilbert 已提交
4184
MODULE_VERSION(SDEBUG_VERSION);
L
Linus Torvalds 已提交
4185 4186

MODULE_PARM_DESC(add_host, "0..127 hosts allowed(def=1)");
4187
MODULE_PARM_DESC(ato, "application tag ownership: 0=disk 1=host (def=1)");
4188
MODULE_PARM_DESC(clustering, "when set enables larger transfers (def=0)");
4189
MODULE_PARM_DESC(delay, "response delay (def=1 jiffy); 0:imm, -1,-2:tiny");
4190
MODULE_PARM_DESC(dev_size_mb, "size in MiB of ram shared by devs(def=8)");
4191 4192
MODULE_PARM_DESC(dif, "data integrity field type: 0-3 (def=0)");
MODULE_PARM_DESC(dix, "data integrity extensions mask (def=0)");
D
Douglas Gilbert 已提交
4193
MODULE_PARM_DESC(dsense, "use descriptor sense format(def=0 -> fixed)");
4194
MODULE_PARM_DESC(every_nth, "timeout every nth command(def=0)");
D
Douglas Gilbert 已提交
4195
MODULE_PARM_DESC(fake_rw, "fake reads/writes instead of copying (def=0)");
4196
MODULE_PARM_DESC(guard, "protection checksum: 0=crc, 1=ip (def=0)");
4197
MODULE_PARM_DESC(host_lock, "host_lock is ignored (def=0)");
4198 4199 4200
MODULE_PARM_DESC(lbpu, "enable LBP, support UNMAP command (def=0)");
MODULE_PARM_DESC(lbpws, "enable LBP, support WRITE SAME(16) with UNMAP bit (def=0)");
MODULE_PARM_DESC(lbpws10, "enable LBP, support WRITE SAME(10) with UNMAP bit (def=0)");
4201 4202
MODULE_PARM_DESC(lbprz,
	"on read unmapped LBs return 0 when 1 (def), return 0xff when 2");
4203
MODULE_PARM_DESC(lowest_aligned, "lowest aligned lba (def=0)");
D
Douglas Gilbert 已提交
4204
MODULE_PARM_DESC(max_luns, "number of LUNs per target to simulate(def=1)");
4205 4206
MODULE_PARM_DESC(max_queue, "max number of queued commands (1 to max(def))");
MODULE_PARM_DESC(ndelay, "response delay in nanoseconds (def=0 -> ignore)");
D
Douglas Gilbert 已提交
4207
MODULE_PARM_DESC(no_lun_0, "no LU number 0 (def=0 -> have lun 0)");
4208
MODULE_PARM_DESC(no_uld, "stop ULD (e.g. sd driver) attaching (def=0))");
L
Linus Torvalds 已提交
4209
MODULE_PARM_DESC(num_parts, "number of partitions(def=0)");
D
Douglas Gilbert 已提交
4210
MODULE_PARM_DESC(num_tgts, "number of targets per host to simulate(def=1)");
4211
MODULE_PARM_DESC(opt_blks, "optimal transfer length in blocks (def=1024)");
4212
MODULE_PARM_DESC(opts, "1->noise, 2->medium_err, 4->timeout, 8->recovered_err... (def=0)");
4213
MODULE_PARM_DESC(physblk_exp, "physical block exponent (def=0)");
L
Linus Torvalds 已提交
4214
MODULE_PARM_DESC(ptype, "SCSI peripheral type(def=0[disk])");
4215
MODULE_PARM_DESC(removable, "claim to have removable media (def=0)");
4216
MODULE_PARM_DESC(scsi_level, "SCSI level to simulate(def=7[SPC-5])");
4217
MODULE_PARM_DESC(sector_size, "logical block size in bytes (def=512)");
4218
MODULE_PARM_DESC(statistics, "collect statistics on commands, queues (def=0)");
4219
MODULE_PARM_DESC(strict, "stricter checks: reserved field in cdb (def=0)");
4220
MODULE_PARM_DESC(submit_queues, "support for block multi-queue (def=1)");
4221 4222
MODULE_PARM_DESC(unmap_alignment, "lowest aligned thin provisioning lba (def=0)");
MODULE_PARM_DESC(unmap_granularity, "thin provisioning granularity in blocks (def=1)");
4223 4224
MODULE_PARM_DESC(unmap_max_blocks, "max # of blocks can be unmapped in one cmd (def=0xffffffff)");
MODULE_PARM_DESC(unmap_max_desc, "max # of ranges that can be unmapped in one cmd (def=256)");
D
Douglas Gilbert 已提交
4225 4226
MODULE_PARM_DESC(uuid_ctl,
		 "1->use uuid for lu name, 0->don't, 2->all use same (def=0)");
4227
MODULE_PARM_DESC(virtual_gb, "virtual gigabyte (GiB) size (def=0 -> use dev_size_mb)");
4228 4229
MODULE_PARM_DESC(vpd_use_hostno, "0 -> dev ids ignore hostno (def=1 -> unique dev ids)");
MODULE_PARM_DESC(write_same_length, "Maximum blocks per WRITE SAME cmd (def=0xffff)");
L
Linus Torvalds 已提交
4230

4231 4232
#define SDEBUG_INFO_LEN 256
static char sdebug_info[SDEBUG_INFO_LEN];
L
Linus Torvalds 已提交
4233 4234 4235

static const char * scsi_debug_info(struct Scsi_Host * shp)
{
4236 4237
	int k;

4238 4239 4240
	k = scnprintf(sdebug_info, SDEBUG_INFO_LEN, "%s: version %s [%s]\n",
		      my_name, SDEBUG_VERSION, sdebug_version_date);
	if (k >= (SDEBUG_INFO_LEN - 1))
4241
		return sdebug_info;
4242 4243 4244 4245
	scnprintf(sdebug_info + k, SDEBUG_INFO_LEN - k,
		  "  dev_size_mb=%d, opts=0x%x, submit_queues=%d, %s=%d",
		  sdebug_dev_size_mb, sdebug_opts, submit_queues,
		  "statistics", (int)sdebug_statistics);
L
Linus Torvalds 已提交
4246 4247 4248
	return sdebug_info;
}

4249
/* 'echo <val> > /proc/scsi/scsi_debug/<host_id>' writes to opts */
4250 4251
static int scsi_debug_write_info(struct Scsi_Host *host, char *buffer,
				 int length)
L
Linus Torvalds 已提交
4252
{
A
Al Viro 已提交
4253 4254 4255
	char arr[16];
	int opts;
	int minLen = length > 15 ? 15 : length;
L
Linus Torvalds 已提交
4256

A
Al Viro 已提交
4257 4258 4259 4260 4261 4262
	if (!capable(CAP_SYS_ADMIN) || !capable(CAP_SYS_RAWIO))
		return -EACCES;
	memcpy(arr, buffer, minLen);
	arr[minLen] = '\0';
	if (1 != sscanf(arr, "%d", &opts))
		return -EINVAL;
4263 4264 4265 4266
	sdebug_opts = opts;
	sdebug_verbose = !!(SDEBUG_OPT_NOISE & opts);
	sdebug_any_injecting_opt = !!(SDEBUG_OPT_ALL_INJECTING & opts);
	if (sdebug_every_nth != 0)
4267
		tweak_cmnd_count();
A
Al Viro 已提交
4268 4269
	return length;
}
L
Linus Torvalds 已提交
4270

4271 4272 4273
/* Output seen with 'cat /proc/scsi/scsi_debug/<host_id>'. It will be the
 * same for each scsi_debug host (if more than one). Some of the counters
 * output are not atomics so might be inaccurate in a busy system. */
A
Al Viro 已提交
4274 4275
static int scsi_debug_show_info(struct seq_file *m, struct Scsi_Host *host)
{
4276 4277 4278 4279 4280 4281 4282 4283 4284 4285 4286 4287 4288 4289 4290 4291 4292 4293 4294 4295 4296 4297 4298 4299 4300 4301 4302 4303 4304 4305 4306 4307 4308 4309 4310 4311 4312
	int f, j, l;
	struct sdebug_queue *sqp;

	seq_printf(m, "scsi_debug adapter driver, version %s [%s]\n",
		   SDEBUG_VERSION, sdebug_version_date);
	seq_printf(m, "num_tgts=%d, %ssize=%d MB, opts=0x%x, every_nth=%d\n",
		   sdebug_num_tgts, "shared (ram) ", sdebug_dev_size_mb,
		   sdebug_opts, sdebug_every_nth);
	seq_printf(m, "delay=%d, ndelay=%d, max_luns=%d, sector_size=%d %s\n",
		   sdebug_jdelay, sdebug_ndelay, sdebug_max_luns,
		   sdebug_sector_size, "bytes");
	seq_printf(m, "cylinders=%d, heads=%d, sectors=%d, command aborts=%d\n",
		   sdebug_cylinders_per, sdebug_heads, sdebug_sectors_per,
		   num_aborts);
	seq_printf(m, "RESETs: device=%d, target=%d, bus=%d, host=%d\n",
		   num_dev_resets, num_target_resets, num_bus_resets,
		   num_host_resets);
	seq_printf(m, "dix_reads=%d, dix_writes=%d, dif_errors=%d\n",
		   dix_reads, dix_writes, dif_errors);
	seq_printf(m, "usec_in_jiffy=%lu, %s=%d, mq_active=%d\n",
		   TICK_NSEC / 1000, "statistics", sdebug_statistics,
		   sdebug_mq_active);
	seq_printf(m, "cmnd_count=%d, completions=%d, %s=%d, a_tsf=%d\n",
		   atomic_read(&sdebug_cmnd_count),
		   atomic_read(&sdebug_completions),
		   "miss_cpus", atomic_read(&sdebug_miss_cpus),
		   atomic_read(&sdebug_a_tsf));

	seq_printf(m, "submit_queues=%d\n", submit_queues);
	for (j = 0, sqp = sdebug_q_arr; j < submit_queues; ++j, ++sqp) {
		seq_printf(m, "  queue %d:\n", j);
		f = find_first_bit(sqp->in_use_bm, sdebug_max_queue);
		if (f != sdebug_max_queue) {
			l = find_last_bit(sqp->in_use_bm, sdebug_max_queue);
			seq_printf(m, "    in_use_bm BUSY: %s: %d,%d\n",
				   "first,last bits", f, l);
		}
4313
	}
A
Al Viro 已提交
4314
	return 0;
L
Linus Torvalds 已提交
4315 4316
}

4317
static ssize_t delay_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4318
{
4319
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_jdelay);
L
Linus Torvalds 已提交
4320
}
4321 4322 4323
/* Returns -EBUSY if jdelay is being changed and commands are queued. The unit
 * of delay is jiffies.
 */
4324 4325
static ssize_t delay_store(struct device_driver *ddp, const char *buf,
			   size_t count)
L
Linus Torvalds 已提交
4326
{
4327
	int jdelay, res;
4328

D
Douglas Gilbert 已提交
4329
	if (count > 0 && sscanf(buf, "%d", &jdelay) == 1) {
4330
		res = count;
4331
		if (sdebug_jdelay != jdelay) {
4332 4333 4334 4335 4336 4337 4338 4339 4340 4341 4342 4343 4344 4345
			int j, k;
			struct sdebug_queue *sqp;

			block_unblock_all_queues(true);
			for (j = 0, sqp = sdebug_q_arr; j < submit_queues;
			     ++j, ++sqp) {
				k = find_first_bit(sqp->in_use_bm,
						   sdebug_max_queue);
				if (k != sdebug_max_queue) {
					res = -EBUSY;   /* queued commands */
					break;
				}
			}
			if (res > 0) {
4346 4347 4348
				/* make sure sdebug_defer instances get
				 * re-allocated for new delay variant */
				free_all_queued();
4349
				sdebug_jdelay = jdelay;
4350
				sdebug_ndelay = 0;
4351
			}
4352
			block_unblock_all_queues(false);
L
Linus Torvalds 已提交
4353
		}
4354
		return res;
L
Linus Torvalds 已提交
4355 4356 4357
	}
	return -EINVAL;
}
4358
static DRIVER_ATTR_RW(delay);
L
Linus Torvalds 已提交
4359

4360 4361
static ssize_t ndelay_show(struct device_driver *ddp, char *buf)
{
4362
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_ndelay);
4363 4364
}
/* Returns -EBUSY if ndelay is being changed and commands are queued */
4365
/* If > 0 and accepted then sdebug_jdelay is set to JDELAY_OVERRIDDEN */
4366
static ssize_t ndelay_store(struct device_driver *ddp, const char *buf,
4367
			    size_t count)
4368
{
4369
	int ndelay, res;
4370 4371

	if ((count > 0) && (1 == sscanf(buf, "%d", &ndelay)) &&
4372
	    (ndelay >= 0) && (ndelay < (1000 * 1000 * 1000))) {
4373
		res = count;
4374
		if (sdebug_ndelay != ndelay) {
4375 4376 4377 4378 4379 4380 4381 4382 4383 4384 4385 4386 4387 4388
			int j, k;
			struct sdebug_queue *sqp;

			block_unblock_all_queues(true);
			for (j = 0, sqp = sdebug_q_arr; j < submit_queues;
			     ++j, ++sqp) {
				k = find_first_bit(sqp->in_use_bm,
						   sdebug_max_queue);
				if (k != sdebug_max_queue) {
					res = -EBUSY;   /* queued commands */
					break;
				}
			}
			if (res > 0) {
4389 4390 4391
				/* make sure sdebug_defer instances get
				 * re-allocated for new delay variant */
				free_all_queued();
4392
				sdebug_ndelay = ndelay;
4393 4394
				sdebug_jdelay = ndelay  ? JDELAY_OVERRIDDEN
							: DEF_JDELAY;
4395
			}
4396
			block_unblock_all_queues(false);
4397 4398 4399 4400 4401 4402 4403
		}
		return res;
	}
	return -EINVAL;
}
static DRIVER_ATTR_RW(ndelay);

4404
static ssize_t opts_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4405
{
4406
	return scnprintf(buf, PAGE_SIZE, "0x%x\n", sdebug_opts);
L
Linus Torvalds 已提交
4407 4408
}

4409 4410
static ssize_t opts_store(struct device_driver *ddp, const char *buf,
			  size_t count)
L
Linus Torvalds 已提交
4411 4412 4413 4414 4415
{
        int opts;
	char work[20];

        if (1 == sscanf(buf, "%10s", work)) {
4416
		if (0 == strncasecmp(work,"0x", 2)) {
L
Linus Torvalds 已提交
4417 4418 4419 4420 4421 4422 4423 4424 4425
			if (1 == sscanf(&work[2], "%x", &opts))
				goto opts_done;
		} else {
			if (1 == sscanf(work, "%d", &opts))
				goto opts_done;
		}
	}
	return -EINVAL;
opts_done:
4426 4427 4428
	sdebug_opts = opts;
	sdebug_verbose = !!(SDEBUG_OPT_NOISE & opts);
	sdebug_any_injecting_opt = !!(SDEBUG_OPT_ALL_INJECTING & opts);
4429
	tweak_cmnd_count();
L
Linus Torvalds 已提交
4430 4431
	return count;
}
4432
static DRIVER_ATTR_RW(opts);
L
Linus Torvalds 已提交
4433

4434
static ssize_t ptype_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4435
{
4436
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_ptype);
L
Linus Torvalds 已提交
4437
}
4438 4439
static ssize_t ptype_store(struct device_driver *ddp, const char *buf,
			   size_t count)
L
Linus Torvalds 已提交
4440 4441 4442 4443
{
        int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4444
		sdebug_ptype = n;
L
Linus Torvalds 已提交
4445 4446 4447 4448
		return count;
	}
	return -EINVAL;
}
4449
static DRIVER_ATTR_RW(ptype);
L
Linus Torvalds 已提交
4450

4451
static ssize_t dsense_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4452
{
4453
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_dsense);
L
Linus Torvalds 已提交
4454
}
4455 4456
static ssize_t dsense_store(struct device_driver *ddp, const char *buf,
			    size_t count)
L
Linus Torvalds 已提交
4457 4458 4459 4460
{
        int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4461
		sdebug_dsense = n;
L
Linus Torvalds 已提交
4462 4463 4464 4465
		return count;
	}
	return -EINVAL;
}
4466
static DRIVER_ATTR_RW(dsense);
L
Linus Torvalds 已提交
4467

4468
static ssize_t fake_rw_show(struct device_driver *ddp, char *buf)
D
Douglas Gilbert 已提交
4469
{
4470
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_fake_rw);
D
Douglas Gilbert 已提交
4471
}
4472 4473
static ssize_t fake_rw_store(struct device_driver *ddp, const char *buf,
			     size_t count)
D
Douglas Gilbert 已提交
4474 4475 4476 4477
{
        int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4478
		n = (n > 0);
4479 4480
		sdebug_fake_rw = (sdebug_fake_rw > 0);
		if (sdebug_fake_rw != n) {
4481 4482
			if ((0 == n) && (NULL == fake_storep)) {
				unsigned long sz =
4483
					(unsigned long)sdebug_dev_size_mb *
4484 4485 4486 4487
					1048576;

				fake_storep = vmalloc(sz);
				if (NULL == fake_storep) {
4488
					pr_err("out of memory, 9\n");
4489 4490 4491 4492
					return -ENOMEM;
				}
				memset(fake_storep, 0, sz);
			}
4493
			sdebug_fake_rw = n;
4494
		}
D
Douglas Gilbert 已提交
4495 4496 4497 4498
		return count;
	}
	return -EINVAL;
}
4499
static DRIVER_ATTR_RW(fake_rw);
D
Douglas Gilbert 已提交
4500

4501
static ssize_t no_lun_0_show(struct device_driver *ddp, char *buf)
D
Douglas Gilbert 已提交
4502
{
4503
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_no_lun_0);
D
Douglas Gilbert 已提交
4504
}
4505 4506
static ssize_t no_lun_0_store(struct device_driver *ddp, const char *buf,
			      size_t count)
D
Douglas Gilbert 已提交
4507 4508 4509 4510
{
        int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4511
		sdebug_no_lun_0 = n;
D
Douglas Gilbert 已提交
4512 4513 4514 4515
		return count;
	}
	return -EINVAL;
}
4516
static DRIVER_ATTR_RW(no_lun_0);
D
Douglas Gilbert 已提交
4517

4518
static ssize_t num_tgts_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4519
{
4520
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_num_tgts);
L
Linus Torvalds 已提交
4521
}
4522 4523
static ssize_t num_tgts_store(struct device_driver *ddp, const char *buf,
			      size_t count)
L
Linus Torvalds 已提交
4524 4525 4526 4527
{
        int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4528
		sdebug_num_tgts = n;
L
Linus Torvalds 已提交
4529 4530 4531 4532 4533
		sdebug_max_tgts_luns();
		return count;
	}
	return -EINVAL;
}
4534
static DRIVER_ATTR_RW(num_tgts);
L
Linus Torvalds 已提交
4535

4536
static ssize_t dev_size_mb_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4537
{
4538
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_dev_size_mb);
L
Linus Torvalds 已提交
4539
}
4540
static DRIVER_ATTR_RO(dev_size_mb);
L
Linus Torvalds 已提交
4541

4542
static ssize_t num_parts_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4543
{
4544
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_num_parts);
L
Linus Torvalds 已提交
4545
}
4546
static DRIVER_ATTR_RO(num_parts);
L
Linus Torvalds 已提交
4547

4548
static ssize_t every_nth_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4549
{
4550
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_every_nth);
L
Linus Torvalds 已提交
4551
}
4552 4553
static ssize_t every_nth_store(struct device_driver *ddp, const char *buf,
			       size_t count)
L
Linus Torvalds 已提交
4554 4555 4556 4557
{
        int nth;

	if ((count > 0) && (1 == sscanf(buf, "%d", &nth))) {
4558
		sdebug_every_nth = nth;
4559 4560 4561 4562 4563
		if (nth && !sdebug_statistics) {
			pr_info("every_nth needs statistics=1, set it\n");
			sdebug_statistics = true;
		}
		tweak_cmnd_count();
L
Linus Torvalds 已提交
4564 4565 4566 4567
		return count;
	}
	return -EINVAL;
}
4568
static DRIVER_ATTR_RW(every_nth);
L
Linus Torvalds 已提交
4569

4570
static ssize_t max_luns_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4571
{
4572
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_max_luns);
L
Linus Torvalds 已提交
4573
}
4574 4575
static ssize_t max_luns_store(struct device_driver *ddp, const char *buf,
			      size_t count)
L
Linus Torvalds 已提交
4576 4577
{
        int n;
4578
	bool changed;
L
Linus Torvalds 已提交
4579 4580

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4581 4582 4583 4584
		if (n > 256) {
			pr_warn("max_luns can be no more than 256\n");
			return -EINVAL;
		}
4585 4586
		changed = (sdebug_max_luns != n);
		sdebug_max_luns = n;
L
Linus Torvalds 已提交
4587
		sdebug_max_tgts_luns();
4588
		if (changed && (sdebug_scsi_level >= 5)) {	/* >= SPC-3 */
4589 4590 4591 4592 4593 4594 4595 4596 4597 4598 4599 4600 4601 4602
			struct sdebug_host_info *sdhp;
			struct sdebug_dev_info *dp;

			spin_lock(&sdebug_host_list_lock);
			list_for_each_entry(sdhp, &sdebug_host_list,
					    host_list) {
				list_for_each_entry(dp, &sdhp->dev_info_list,
						    dev_list) {
					set_bit(SDEBUG_UA_LUNS_CHANGED,
						dp->uas_bm);
				}
			}
			spin_unlock(&sdebug_host_list_lock);
		}
L
Linus Torvalds 已提交
4603 4604 4605 4606
		return count;
	}
	return -EINVAL;
}
4607
static DRIVER_ATTR_RW(max_luns);
L
Linus Torvalds 已提交
4608

4609
static ssize_t max_queue_show(struct device_driver *ddp, char *buf)
4610
{
4611
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_max_queue);
4612
}
4613 4614
/* N.B. max_queue can be changed while there are queued commands. In flight
 * commands beyond the new max_queue will be completed. */
4615 4616
static ssize_t max_queue_store(struct device_driver *ddp, const char *buf,
			       size_t count)
4617
{
4618 4619
	int j, n, k, a;
	struct sdebug_queue *sqp;
4620 4621

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n > 0) &&
4622 4623 4624 4625 4626 4627 4628 4629 4630
	    (n <= SDEBUG_CANQUEUE)) {
		block_unblock_all_queues(true);
		k = 0;
		for (j = 0, sqp = sdebug_q_arr; j < submit_queues;
		     ++j, ++sqp) {
			a = find_last_bit(sqp->in_use_bm, SDEBUG_CANQUEUE);
			if (a > k)
				k = a;
		}
4631
		sdebug_max_queue = n;
4632
		if (k == SDEBUG_CANQUEUE)
4633 4634 4635 4636 4637
			atomic_set(&retired_max_queue, 0);
		else if (k >= n)
			atomic_set(&retired_max_queue, k + 1);
		else
			atomic_set(&retired_max_queue, 0);
4638
		block_unblock_all_queues(false);
4639 4640 4641 4642
		return count;
	}
	return -EINVAL;
}
4643
static DRIVER_ATTR_RW(max_queue);
4644

4645
static ssize_t no_uld_show(struct device_driver *ddp, char *buf)
4646
{
4647
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_no_uld);
4648
}
4649
static DRIVER_ATTR_RO(no_uld);
4650

4651
static ssize_t scsi_level_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4652
{
4653
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_scsi_level);
L
Linus Torvalds 已提交
4654
}
4655
static DRIVER_ATTR_RO(scsi_level);
L
Linus Torvalds 已提交
4656

4657
static ssize_t virtual_gb_show(struct device_driver *ddp, char *buf)
D
Douglas Gilbert 已提交
4658
{
4659
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_virtual_gb);
D
Douglas Gilbert 已提交
4660
}
4661 4662
static ssize_t virtual_gb_store(struct device_driver *ddp, const char *buf,
				size_t count)
D
Douglas Gilbert 已提交
4663
{
4664
        int n;
4665
	bool changed;
D
Douglas Gilbert 已提交
4666 4667

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4668 4669
		changed = (sdebug_virtual_gb != n);
		sdebug_virtual_gb = n;
4670
		sdebug_capacity = get_sdebug_capacity();
4671 4672 4673 4674
		if (changed) {
			struct sdebug_host_info *sdhp;
			struct sdebug_dev_info *dp;

4675
			spin_lock(&sdebug_host_list_lock);
4676 4677 4678 4679 4680 4681 4682 4683
			list_for_each_entry(sdhp, &sdebug_host_list,
					    host_list) {
				list_for_each_entry(dp, &sdhp->dev_info_list,
						    dev_list) {
					set_bit(SDEBUG_UA_CAPACITY_CHANGED,
						dp->uas_bm);
				}
			}
4684
			spin_unlock(&sdebug_host_list_lock);
4685
		}
D
Douglas Gilbert 已提交
4686 4687 4688 4689
		return count;
	}
	return -EINVAL;
}
4690
static DRIVER_ATTR_RW(virtual_gb);
D
Douglas Gilbert 已提交
4691

4692
static ssize_t add_host_show(struct device_driver *ddp, char *buf)
L
Linus Torvalds 已提交
4693
{
4694
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_add_host);
L
Linus Torvalds 已提交
4695 4696
}

4697 4698 4699
static int sdebug_add_adapter(void);
static void sdebug_remove_adapter(void);

4700 4701
static ssize_t add_host_store(struct device_driver *ddp, const char *buf,
			      size_t count)
L
Linus Torvalds 已提交
4702
{
4703
	int delta_hosts;
L
Linus Torvalds 已提交
4704

4705
	if (sscanf(buf, "%d", &delta_hosts) != 1)
L
Linus Torvalds 已提交
4706 4707 4708 4709 4710 4711 4712 4713 4714 4715 4716 4717
		return -EINVAL;
	if (delta_hosts > 0) {
		do {
			sdebug_add_adapter();
		} while (--delta_hosts);
	} else if (delta_hosts < 0) {
		do {
			sdebug_remove_adapter();
		} while (++delta_hosts);
	}
	return count;
}
4718
static DRIVER_ATTR_RW(add_host);
L
Linus Torvalds 已提交
4719

4720
static ssize_t vpd_use_hostno_show(struct device_driver *ddp, char *buf)
D
Douglas Gilbert 已提交
4721
{
4722
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_vpd_use_hostno);
D
Douglas Gilbert 已提交
4723
}
4724 4725
static ssize_t vpd_use_hostno_store(struct device_driver *ddp, const char *buf,
				    size_t count)
D
Douglas Gilbert 已提交
4726 4727 4728 4729
{
	int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4730
		sdebug_vpd_use_hostno = n;
D
Douglas Gilbert 已提交
4731 4732 4733 4734
		return count;
	}
	return -EINVAL;
}
4735
static DRIVER_ATTR_RW(vpd_use_hostno);
D
Douglas Gilbert 已提交
4736

4737 4738 4739 4740 4741 4742 4743 4744 4745 4746 4747 4748 4749 4750 4751 4752 4753 4754 4755 4756 4757 4758
static ssize_t statistics_show(struct device_driver *ddp, char *buf)
{
	return scnprintf(buf, PAGE_SIZE, "%d\n", (int)sdebug_statistics);
}
static ssize_t statistics_store(struct device_driver *ddp, const char *buf,
				size_t count)
{
	int n;

	if ((count > 0) && (sscanf(buf, "%d", &n) == 1) && (n >= 0)) {
		if (n > 0)
			sdebug_statistics = true;
		else {
			clear_queue_stats();
			sdebug_statistics = false;
		}
		return count;
	}
	return -EINVAL;
}
static DRIVER_ATTR_RW(statistics);

4759
static ssize_t sector_size_show(struct device_driver *ddp, char *buf)
4760
{
4761
	return scnprintf(buf, PAGE_SIZE, "%u\n", sdebug_sector_size);
4762
}
4763
static DRIVER_ATTR_RO(sector_size);
4764

4765 4766 4767 4768 4769 4770
static ssize_t submit_queues_show(struct device_driver *ddp, char *buf)
{
	return scnprintf(buf, PAGE_SIZE, "%d\n", submit_queues);
}
static DRIVER_ATTR_RO(submit_queues);

4771
static ssize_t dix_show(struct device_driver *ddp, char *buf)
4772
{
4773
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_dix);
4774
}
4775
static DRIVER_ATTR_RO(dix);
4776

4777
static ssize_t dif_show(struct device_driver *ddp, char *buf)
4778
{
4779
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_dif);
4780
}
4781
static DRIVER_ATTR_RO(dif);
4782

4783
static ssize_t guard_show(struct device_driver *ddp, char *buf)
4784
{
4785
	return scnprintf(buf, PAGE_SIZE, "%u\n", sdebug_guard);
4786
}
4787
static DRIVER_ATTR_RO(guard);
4788

4789
static ssize_t ato_show(struct device_driver *ddp, char *buf)
4790
{
4791
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_ato);
4792
}
4793
static DRIVER_ATTR_RO(ato);
4794

4795
static ssize_t map_show(struct device_driver *ddp, char *buf)
4796 4797 4798
{
	ssize_t count;

4799
	if (!scsi_debug_lbp())
4800 4801 4802
		return scnprintf(buf, PAGE_SIZE, "0-%u\n",
				 sdebug_store_sectors);

4803 4804
	count = scnprintf(buf, PAGE_SIZE - 1, "%*pbl",
			  (int)map_size, map_storep);
4805
	buf[count++] = '\n';
4806
	buf[count] = '\0';
4807 4808 4809

	return count;
}
4810
static DRIVER_ATTR_RO(map);
4811

4812
static ssize_t removable_show(struct device_driver *ddp, char *buf)
4813
{
4814
	return scnprintf(buf, PAGE_SIZE, "%d\n", sdebug_removable ? 1 : 0);
4815
}
4816 4817
static ssize_t removable_store(struct device_driver *ddp, const char *buf,
			       size_t count)
4818 4819 4820 4821
{
	int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4822
		sdebug_removable = (n > 0);
4823 4824 4825 4826
		return count;
	}
	return -EINVAL;
}
4827
static DRIVER_ATTR_RW(removable);
4828

4829 4830
static ssize_t host_lock_show(struct device_driver *ddp, char *buf)
{
4831
	return scnprintf(buf, PAGE_SIZE, "%d\n", !!sdebug_host_lock);
4832
}
4833
/* N.B. sdebug_host_lock does nothing, kept for backward compatibility */
4834 4835 4836
static ssize_t host_lock_store(struct device_driver *ddp, const char *buf,
			       size_t count)
{
4837
	int n;
4838 4839

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4840 4841
		sdebug_host_lock = (n > 0);
		return count;
4842 4843 4844 4845 4846
	}
	return -EINVAL;
}
static DRIVER_ATTR_RW(host_lock);

4847 4848
static ssize_t strict_show(struct device_driver *ddp, char *buf)
{
4849
	return scnprintf(buf, PAGE_SIZE, "%d\n", !!sdebug_strict);
4850 4851 4852 4853 4854 4855 4856
}
static ssize_t strict_store(struct device_driver *ddp, const char *buf,
			    size_t count)
{
	int n;

	if ((count > 0) && (1 == sscanf(buf, "%d", &n)) && (n >= 0)) {
4857
		sdebug_strict = (n > 0);
4858 4859 4860 4861 4862 4863
		return count;
	}
	return -EINVAL;
}
static DRIVER_ATTR_RW(strict);

D
Douglas Gilbert 已提交
4864 4865 4866 4867 4868 4869
static ssize_t uuid_ctl_show(struct device_driver *ddp, char *buf)
{
	return scnprintf(buf, PAGE_SIZE, "%d\n", !!sdebug_uuid_ctl);
}
static DRIVER_ATTR_RO(uuid_ctl);

4870

4871
/* Note: The following array creates attribute files in the
D
Douglas Gilbert 已提交
4872 4873 4874 4875 4876
   /sys/bus/pseudo/drivers/scsi_debug directory. The advantage of these
   files (over those found in the /sys/module/scsi_debug/parameters
   directory) is that auxiliary actions can be triggered when an attribute
   is changed. For example see: sdebug_add_host_store() above.
 */
4877

4878 4879 4880 4881 4882 4883 4884 4885 4886 4887 4888 4889 4890 4891 4892 4893 4894 4895 4896
static struct attribute *sdebug_drv_attrs[] = {
	&driver_attr_delay.attr,
	&driver_attr_opts.attr,
	&driver_attr_ptype.attr,
	&driver_attr_dsense.attr,
	&driver_attr_fake_rw.attr,
	&driver_attr_no_lun_0.attr,
	&driver_attr_num_tgts.attr,
	&driver_attr_dev_size_mb.attr,
	&driver_attr_num_parts.attr,
	&driver_attr_every_nth.attr,
	&driver_attr_max_luns.attr,
	&driver_attr_max_queue.attr,
	&driver_attr_no_uld.attr,
	&driver_attr_scsi_level.attr,
	&driver_attr_virtual_gb.attr,
	&driver_attr_add_host.attr,
	&driver_attr_vpd_use_hostno.attr,
	&driver_attr_sector_size.attr,
4897 4898
	&driver_attr_statistics.attr,
	&driver_attr_submit_queues.attr,
4899 4900 4901 4902 4903 4904
	&driver_attr_dix.attr,
	&driver_attr_dif.attr,
	&driver_attr_guard.attr,
	&driver_attr_ato.attr,
	&driver_attr_map.attr,
	&driver_attr_removable.attr,
4905 4906
	&driver_attr_host_lock.attr,
	&driver_attr_ndelay.attr,
4907
	&driver_attr_strict.attr,
D
Douglas Gilbert 已提交
4908
	&driver_attr_uuid_ctl.attr,
4909 4910 4911
	NULL,
};
ATTRIBUTE_GROUPS(sdebug_drv);
L
Linus Torvalds 已提交
4912

4913
static struct device *pseudo_primary;
4914

L
Linus Torvalds 已提交
4915 4916
static int __init scsi_debug_init(void)
{
4917
	unsigned long sz;
L
Linus Torvalds 已提交
4918 4919
	int host_to_add;
	int k;
4920
	int ret;
L
Linus Torvalds 已提交
4921

4922 4923
	atomic_set(&retired_max_queue, 0);

4924
	if (sdebug_ndelay >= 1000 * 1000 * 1000) {
4925
		pr_warn("ndelay must be less than 1 second, ignored\n");
4926 4927
		sdebug_ndelay = 0;
	} else if (sdebug_ndelay > 0)
4928
		sdebug_jdelay = JDELAY_OVERRIDDEN;
4929

4930
	switch (sdebug_sector_size) {
4931 4932 4933 4934 4935 4936
	case  512:
	case 1024:
	case 2048:
	case 4096:
		break;
	default:
4937
		pr_err("invalid sector_size %d\n", sdebug_sector_size);
4938 4939 4940
		return -EINVAL;
	}

4941
	switch (sdebug_dif) {
4942 4943

	case SD_DIF_TYPE0_PROTECTION:
4944
		break;
4945
	case SD_DIF_TYPE1_PROTECTION:
4946
	case SD_DIF_TYPE2_PROTECTION:
4947
	case SD_DIF_TYPE3_PROTECTION:
4948
		have_dif_prot = true;
4949 4950 4951
		break;

	default:
4952
		pr_err("dif must be 0, 1, 2 or 3\n");
4953 4954 4955
		return -EINVAL;
	}

4956
	if (sdebug_guard > 1) {
4957
		pr_err("guard must be 0 or 1\n");
4958 4959 4960
		return -EINVAL;
	}

4961
	if (sdebug_ato > 1) {
4962
		pr_err("ato must be 0 or 1\n");
4963 4964 4965
		return -EINVAL;
	}

4966 4967
	if (sdebug_physblk_exp > 15) {
		pr_err("invalid physblk_exp %u\n", sdebug_physblk_exp);
4968 4969
		return -EINVAL;
	}
4970 4971 4972 4973
	if (sdebug_max_luns > 256) {
		pr_warn("max_luns can be no more than 256, use default\n");
		sdebug_max_luns = DEF_MAX_LUNS;
	}
4974

4975 4976
	if (sdebug_lowest_aligned > 0x3fff) {
		pr_err("lowest_aligned too big: %u\n", sdebug_lowest_aligned);
4977 4978 4979
		return -EINVAL;
	}

4980 4981 4982 4983 4984 4985 4986 4987 4988 4989 4990
	if (submit_queues < 1) {
		pr_err("submit_queues must be 1 or more\n");
		return -EINVAL;
	}
	sdebug_q_arr = kcalloc(submit_queues, sizeof(struct sdebug_queue),
			       GFP_KERNEL);
	if (sdebug_q_arr == NULL)
		return -ENOMEM;
	for (k = 0; k < submit_queues; ++k)
		spin_lock_init(&sdebug_q_arr[k].qc_lock);

4991 4992 4993 4994
	if (sdebug_dev_size_mb < 1)
		sdebug_dev_size_mb = 1;  /* force minimum 1 MB ramdisk */
	sz = (unsigned long)sdebug_dev_size_mb * 1048576;
	sdebug_store_sectors = sz / sdebug_sector_size;
4995
	sdebug_capacity = get_sdebug_capacity();
L
Linus Torvalds 已提交
4996 4997 4998 4999

	/* play around with geometry, don't waste too much on track 0 */
	sdebug_heads = 8;
	sdebug_sectors_per = 32;
5000
	if (sdebug_dev_size_mb >= 256)
L
Linus Torvalds 已提交
5001
		sdebug_heads = 64;
5002
	else if (sdebug_dev_size_mb >= 16)
5003
		sdebug_heads = 32;
L
Linus Torvalds 已提交
5004 5005 5006 5007 5008 5009 5010 5011 5012 5013
	sdebug_cylinders_per = (unsigned long)sdebug_capacity /
			       (sdebug_sectors_per * sdebug_heads);
	if (sdebug_cylinders_per >= 1024) {
		/* other LLDs do this; implies >= 1GB ram disk ... */
		sdebug_heads = 255;
		sdebug_sectors_per = 63;
		sdebug_cylinders_per = (unsigned long)sdebug_capacity /
			       (sdebug_sectors_per * sdebug_heads);
	}

D
Douglas Gilbert 已提交
5014
	if (sdebug_fake_rw == 0) {
5015 5016
		fake_storep = vmalloc(sz);
		if (NULL == fake_storep) {
5017
			pr_err("out of memory, 1\n");
5018 5019
			ret = -ENOMEM;
			goto free_q_arr;
5020 5021
		}
		memset(fake_storep, 0, sz);
5022
		if (sdebug_num_parts > 0)
5023
			sdebug_build_parts(fake_storep, sz);
L
Linus Torvalds 已提交
5024 5025
	}

5026
	if (sdebug_dix) {
5027 5028 5029 5030 5031
		int dif_size;

		dif_size = sdebug_store_sectors * sizeof(struct sd_dif_tuple);
		dif_storep = vmalloc(dif_size);

5032
		pr_err("dif_storep %u bytes @ %p\n", dif_size, dif_storep);
5033 5034

		if (dif_storep == NULL) {
5035
			pr_err("out of mem. (DIX)\n");
5036 5037 5038 5039 5040 5041 5042
			ret = -ENOMEM;
			goto free_vm;
		}

		memset(dif_storep, 0xff, dif_size);
	}

5043 5044
	/* Logical Block Provisioning */
	if (scsi_debug_lbp()) {
5045 5046
		sdebug_unmap_max_blocks =
			clamp(sdebug_unmap_max_blocks, 0U, 0xffffffffU);
5047

5048 5049
		sdebug_unmap_max_desc =
			clamp(sdebug_unmap_max_desc, 0U, 256U);
5050

5051 5052
		sdebug_unmap_granularity =
			clamp(sdebug_unmap_granularity, 1U, 0xffffffffU);
5053

5054 5055 5056
		if (sdebug_unmap_alignment &&
		    sdebug_unmap_granularity <=
		    sdebug_unmap_alignment) {
5057
			pr_err("ERR: unmap_granularity <= unmap_alignment\n");
5058 5059
			ret = -EINVAL;
			goto free_vm;
5060 5061
		}

5062 5063
		map_size = lba_to_map_index(sdebug_store_sectors - 1) + 1;
		map_storep = vmalloc(BITS_TO_LONGS(map_size) * sizeof(long));
5064

5065
		pr_info("%lu provisioning blocks\n", map_size);
5066 5067

		if (map_storep == NULL) {
5068
			pr_err("out of mem. (MAP)\n");
5069 5070 5071 5072
			ret = -ENOMEM;
			goto free_vm;
		}

5073
		bitmap_zero(map_storep, map_size);
5074 5075

		/* Map first 1KB for partition table */
5076
		if (sdebug_num_parts)
5077 5078 5079
			map_region(0, 2);
	}

5080 5081
	pseudo_primary = root_device_register("pseudo_0");
	if (IS_ERR(pseudo_primary)) {
5082
		pr_warn("root_device_register() error\n");
5083
		ret = PTR_ERR(pseudo_primary);
5084 5085 5086 5087
		goto free_vm;
	}
	ret = bus_register(&pseudo_lld_bus);
	if (ret < 0) {
5088
		pr_warn("bus_register error: %d\n", ret);
5089 5090 5091 5092
		goto dev_unreg;
	}
	ret = driver_register(&sdebug_driverfs_driver);
	if (ret < 0) {
5093
		pr_warn("driver_register error: %d\n", ret);
5094 5095
		goto bus_unreg;
	}
L
Linus Torvalds 已提交
5096

5097 5098
	host_to_add = sdebug_add_host;
	sdebug_add_host = 0;
L
Linus Torvalds 已提交
5099 5100 5101

        for (k = 0; k < host_to_add; k++) {
                if (sdebug_add_adapter()) {
5102
			pr_err("sdebug_add_adapter failed k=%d\n", k);
L
Linus Torvalds 已提交
5103 5104 5105 5106
                        break;
                }
        }

5107 5108
	if (sdebug_verbose)
		pr_info("built %d host(s)\n", sdebug_add_host);
5109

L
Linus Torvalds 已提交
5110
	return 0;
5111 5112 5113 5114

bus_unreg:
	bus_unregister(&pseudo_lld_bus);
dev_unreg:
5115
	root_device_unregister(pseudo_primary);
5116
free_vm:
5117 5118
	vfree(map_storep);
	vfree(dif_storep);
5119
	vfree(fake_storep);
5120 5121
free_q_arr:
	kfree(sdebug_q_arr);
5122
	return ret;
L
Linus Torvalds 已提交
5123 5124 5125 5126
}

static void __exit scsi_debug_exit(void)
{
5127
	int k = sdebug_add_host;
L
Linus Torvalds 已提交
5128 5129

	stop_all_queued();
5130
	free_all_queued();
L
Linus Torvalds 已提交
5131 5132 5133 5134
	for (; k; k--)
		sdebug_remove_adapter();
	driver_unregister(&sdebug_driverfs_driver);
	bus_unregister(&pseudo_lld_bus);
5135
	root_device_unregister(pseudo_primary);
L
Linus Torvalds 已提交
5136

5137
	vfree(dif_storep);
L
Linus Torvalds 已提交
5138
	vfree(fake_storep);
5139
	kfree(sdebug_q_arr);
L
Linus Torvalds 已提交
5140 5141 5142 5143 5144 5145 5146 5147 5148 5149 5150 5151 5152 5153 5154 5155 5156 5157
}

device_initcall(scsi_debug_init);
module_exit(scsi_debug_exit);

static void sdebug_release_adapter(struct device * dev)
{
        struct sdebug_host_info *sdbg_host;

	sdbg_host = to_sdebug_host(dev);
        kfree(sdbg_host);
}

static int sdebug_add_adapter(void)
{
	int k, devs_per_host;
        int error = 0;
        struct sdebug_host_info *sdbg_host;
5158
	struct sdebug_dev_info *sdbg_devinfo, *tmp;
L
Linus Torvalds 已提交
5159

D
Douglas Gilbert 已提交
5160
        sdbg_host = kzalloc(sizeof(*sdbg_host),GFP_KERNEL);
L
Linus Torvalds 已提交
5161
        if (NULL == sdbg_host) {
5162
		pr_err("out of memory at line %d\n", __LINE__);
L
Linus Torvalds 已提交
5163 5164 5165 5166 5167
                return -ENOMEM;
        }

        INIT_LIST_HEAD(&sdbg_host->dev_info_list);

5168
	devs_per_host = sdebug_num_tgts * sdebug_max_luns;
L
Linus Torvalds 已提交
5169
        for (k = 0; k < devs_per_host; k++) {
5170 5171
		sdbg_devinfo = sdebug_device_create(sdbg_host, GFP_KERNEL);
		if (!sdbg_devinfo) {
5172
			pr_err("out of memory at line %d\n", __LINE__);
L
Linus Torvalds 已提交
5173 5174 5175 5176 5177 5178 5179 5180 5181 5182
                        error = -ENOMEM;
			goto clean;
                }
        }

        spin_lock(&sdebug_host_list_lock);
        list_add_tail(&sdbg_host->host_list, &sdebug_host_list);
        spin_unlock(&sdebug_host_list_lock);

        sdbg_host->dev.bus = &pseudo_lld_bus;
5183
        sdbg_host->dev.parent = pseudo_primary;
L
Linus Torvalds 已提交
5184
        sdbg_host->dev.release = &sdebug_release_adapter;
5185
	dev_set_name(&sdbg_host->dev, "adapter%d", sdebug_add_host);
L
Linus Torvalds 已提交
5186 5187 5188 5189 5190 5191

        error = device_register(&sdbg_host->dev);

        if (error)
		goto clean;

5192
	++sdebug_add_host;
L
Linus Torvalds 已提交
5193 5194 5195
        return error;

clean:
5196 5197
	list_for_each_entry_safe(sdbg_devinfo, tmp, &sdbg_host->dev_info_list,
				 dev_list) {
L
Linus Torvalds 已提交
5198 5199 5200 5201 5202 5203 5204 5205 5206 5207 5208 5209 5210 5211 5212 5213 5214 5215 5216 5217 5218 5219 5220
		list_del(&sdbg_devinfo->dev_list);
		kfree(sdbg_devinfo);
	}

	kfree(sdbg_host);
        return error;
}

static void sdebug_remove_adapter(void)
{
        struct sdebug_host_info * sdbg_host = NULL;

        spin_lock(&sdebug_host_list_lock);
        if (!list_empty(&sdebug_host_list)) {
                sdbg_host = list_entry(sdebug_host_list.prev,
                                       struct sdebug_host_info, host_list);
		list_del(&sdbg_host->host_list);
	}
        spin_unlock(&sdebug_host_list_lock);

	if (!sdbg_host)
		return;

5221 5222
	device_unregister(&sdbg_host->dev);
	--sdebug_add_host;
L
Linus Torvalds 已提交
5223 5224
}

5225
static int sdebug_change_qdepth(struct scsi_device *sdev, int qdepth)
5226 5227 5228 5229
{
	int num_in_q = 0;
	struct sdebug_dev_info *devip;

5230
	block_unblock_all_queues(true);
5231 5232
	devip = (struct sdebug_dev_info *)sdev->hostdata;
	if (NULL == devip) {
5233
		block_unblock_all_queues(false);
5234 5235 5236
		return	-ENODEV;
	}
	num_in_q = atomic_read(&devip->num_in_q);
5237 5238 5239

	if (qdepth < 1)
		qdepth = 1;
5240 5241 5242
	/* allow to exceed max host qc_arr elements for testing */
	if (qdepth > SDEBUG_CANQUEUE + 10)
		qdepth = SDEBUG_CANQUEUE + 10;
5243
	scsi_change_queue_depth(sdev, qdepth);
5244

5245
	if (SDEBUG_OPT_Q_NOISE & sdebug_opts) {
5246
		sdev_printk(KERN_INFO, sdev, "%s: qdepth=%d, num_in_q=%d\n",
5247
			    __func__, qdepth, num_in_q);
5248
	}
5249
	block_unblock_all_queues(false);
5250 5251 5252
	return sdev->queue_depth;
}

5253
static bool fake_timeout(struct scsi_cmnd *scp)
5254
{
5255
	if (0 == (atomic_read(&sdebug_cmnd_count) % abs(sdebug_every_nth))) {
5256 5257 5258
		if (sdebug_every_nth < -1)
			sdebug_every_nth = -1;
		if (SDEBUG_OPT_TIMEOUT & sdebug_opts)
5259
			return true; /* ignore command causing timeout */
5260
		else if (SDEBUG_OPT_MAC_TIMEOUT & sdebug_opts &&
5261
			 scsi_medium_access_command(scp))
5262
			return true; /* time out reads and writes */
5263
	}
5264
	return false;
5265 5266
}

5267 5268
static int scsi_debug_queuecommand(struct Scsi_Host *shost,
				   struct scsi_cmnd *scp)
5269 5270 5271 5272 5273 5274 5275 5276 5277 5278 5279 5280 5281 5282 5283 5284
{
	u8 sdeb_i;
	struct scsi_device *sdp = scp->device;
	const struct opcode_info_t *oip;
	const struct opcode_info_t *r_oip;
	struct sdebug_dev_info *devip;
	u8 *cmd = scp->cmnd;
	int (*r_pfp)(struct scsi_cmnd *, struct sdebug_dev_info *);
	int k, na;
	int errsts = 0;
	u32 flags;
	u16 sa;
	u8 opcode = cmd[0];
	bool has_wlun_rl;

	scsi_set_resid(scp, 0);
5285 5286
	if (sdebug_statistics)
		atomic_inc(&sdebug_cmnd_count);
5287 5288
	if (unlikely(sdebug_verbose &&
		     !(SDEBUG_OPT_NO_CDB_NOISE & sdebug_opts))) {
5289 5290 5291 5292 5293 5294 5295 5296 5297 5298 5299 5300
		char b[120];
		int n, len, sb;

		len = scp->cmd_len;
		sb = (int)sizeof(b);
		if (len > 32)
			strcpy(b, "too long, over 32 bytes");
		else {
			for (k = 0, n = 0; k < len && n < sb; ++k)
				n += scnprintf(b + n, sb - n, "%02x ",
					       (u32)cmd[k]);
		}
5301 5302 5303 5304 5305 5306 5307
		if (sdebug_mq_active)
			sdev_printk(KERN_INFO, sdp, "%s: tag=%u, cmd %s\n",
				    my_name, blk_mq_unique_tag(scp->request),
				    b);
		else
			sdev_printk(KERN_INFO, sdp, "%s: cmd %s\n", my_name,
				    b);
5308
	}
5309
	has_wlun_rl = (sdp->lun == SCSI_W_LUN_REPORT_LUNS);
5310 5311
	if (unlikely((sdp->lun >= sdebug_max_luns) && !has_wlun_rl))
		goto err_out;
5312 5313 5314 5315

	sdeb_i = opcode_ind_arr[opcode];	/* fully mapped */
	oip = &opcode_info_arr[sdeb_i];		/* safe if table consistent */
	devip = (struct sdebug_dev_info *)sdp->hostdata;
5316 5317
	if (unlikely(!devip)) {
		devip = find_build_dev_info(sdp);
5318
		if (NULL == devip)
5319
			goto err_out;
5320 5321 5322 5323 5324 5325 5326 5327 5328 5329 5330 5331 5332 5333 5334 5335 5336 5337 5338 5339 5340 5341 5342 5343 5344 5345 5346 5347 5348 5349 5350
	}
	na = oip->num_attached;
	r_pfp = oip->pfp;
	if (na) {	/* multiple commands with this opcode */
		r_oip = oip;
		if (FF_SA & r_oip->flags) {
			if (F_SA_LOW & oip->flags)
				sa = 0x1f & cmd[1];
			else
				sa = get_unaligned_be16(cmd + 8);
			for (k = 0; k <= na; oip = r_oip->arrp + k++) {
				if (opcode == oip->opcode && sa == oip->sa)
					break;
			}
		} else {   /* since no service action only check opcode */
			for (k = 0; k <= na; oip = r_oip->arrp + k++) {
				if (opcode == oip->opcode)
					break;
			}
		}
		if (k > na) {
			if (F_SA_LOW & r_oip->flags)
				mk_sense_invalid_fld(scp, SDEB_IN_CDB, 1, 4);
			else if (F_SA_HIGH & r_oip->flags)
				mk_sense_invalid_fld(scp, SDEB_IN_CDB, 8, 7);
			else
				mk_sense_invalid_opcode(scp);
			goto check_cond;
		}
	}	/* else (when na==0) we assume the oip is a match */
	flags = oip->flags;
5351
	if (unlikely(F_INV_OP & flags)) {
5352 5353 5354
		mk_sense_invalid_opcode(scp);
		goto check_cond;
	}
5355
	if (unlikely(has_wlun_rl && !(F_RL_WLUN_OK & flags))) {
5356 5357 5358
		if (sdebug_verbose)
			sdev_printk(KERN_INFO, sdp, "%s: Opcode 0x%x not%s\n",
				    my_name, opcode, " supported for wlun");
5359 5360 5361
		mk_sense_invalid_opcode(scp);
		goto check_cond;
	}
5362
	if (unlikely(sdebug_strict)) {	/* check cdb against mask */
5363 5364 5365 5366 5367 5368 5369 5370 5371 5372 5373 5374 5375 5376 5377
		u8 rem;
		int j;

		for (k = 1; k < oip->len_mask[0] && k < 16; ++k) {
			rem = ~oip->len_mask[k] & cmd[k];
			if (rem) {
				for (j = 7; j >= 0; --j, rem <<= 1) {
					if (0x80 & rem)
						break;
				}
				mk_sense_invalid_fld(scp, SDEB_IN_CDB, k, j);
				goto check_cond;
			}
		}
	}
5378
	if (unlikely(!(F_SKIP_UA & flags) &&
D
Douglas Gilbert 已提交
5379 5380
		     find_first_bit(devip->uas_bm,
				    SDEBUG_NUM_UAS) != SDEBUG_NUM_UAS)) {
5381
		errsts = make_ua(scp, devip);
5382 5383 5384
		if (errsts)
			goto check_cond;
	}
5385
	if (unlikely((F_M_ACCESS & flags) && atomic_read(&devip->stopped))) {
5386
		mk_sense_buffer(scp, NOT_READY, LOGICAL_UNIT_NOT_READY, 0x2);
5387
		if (sdebug_verbose)
5388 5389 5390 5391 5392 5393
			sdev_printk(KERN_INFO, sdp, "%s reports: Not ready: "
				    "%s\n", my_name, "initializing command "
				    "required");
		errsts = check_condition_result;
		goto fini;
	}
5394
	if (sdebug_fake_rw && (F_FAKE_RW & flags))
5395
		goto fini;
5396
	if (unlikely(sdebug_every_nth)) {
5397
		if (fake_timeout(scp))
5398 5399
			return 0;	/* ignore command: make trouble */
	}
5400 5401
	if (likely(oip->pfp))
		errsts = oip->pfp(scp, devip);	/* calls a resp_* function */
5402 5403 5404 5405 5406
	else if (r_pfp)	/* if leaf function ptr NULL, try the root's */
		errsts = r_pfp(scp, devip);

fini:
	return schedule_resp(scp, devip, errsts,
5407
			     ((F_DELAY_OVERR & flags) ? 0 : sdebug_jdelay));
5408 5409
check_cond:
	return schedule_resp(scp, devip, check_condition_result, 0);
5410 5411
err_out:
	return schedule_resp(scp, NULL, DID_NO_CONNECT << 16, 0);
5412 5413
}

5414
static struct scsi_host_template sdebug_driver_template = {
A
Al Viro 已提交
5415 5416
	.show_info =		scsi_debug_show_info,
	.write_info =		scsi_debug_write_info,
5417 5418 5419 5420 5421 5422 5423
	.proc_name =		sdebug_proc_name,
	.name =			"SCSI DEBUG",
	.info =			scsi_debug_info,
	.slave_alloc =		scsi_debug_slave_alloc,
	.slave_configure =	scsi_debug_slave_configure,
	.slave_destroy =	scsi_debug_slave_destroy,
	.ioctl =		scsi_debug_ioctl,
5424
	.queuecommand =		scsi_debug_queuecommand,
5425
	.change_queue_depth =	sdebug_change_qdepth,
5426 5427
	.eh_abort_handler =	scsi_debug_abort,
	.eh_device_reset_handler = scsi_debug_device_reset,
5428 5429
	.eh_target_reset_handler = scsi_debug_target_reset,
	.eh_bus_reset_handler = scsi_debug_bus_reset,
5430
	.eh_host_reset_handler = scsi_debug_host_reset,
5431
	.can_queue =		SDEBUG_CANQUEUE,
5432
	.this_id =		7,
5433
	.sg_tablesize =		SG_MAX_SEGMENTS,
5434
	.cmd_per_lun =		DEF_CMD_PER_LUN,
5435
	.max_sectors =		-1U,
5436 5437
	.use_clustering = 	DISABLE_CLUSTERING,
	.module =		THIS_MODULE,
5438
	.track_queue_depth =	1,
5439 5440
};

L
Linus Torvalds 已提交
5441 5442
static int sdebug_driver_probe(struct device * dev)
{
5443 5444 5445
	int error = 0;
	struct sdebug_host_info *sdbg_host;
	struct Scsi_Host *hpnt;
5446
	int hprot;
L
Linus Torvalds 已提交
5447 5448 5449

	sdbg_host = to_sdebug_host(dev);

5450 5451
	sdebug_driver_template.can_queue = sdebug_max_queue;
	if (sdebug_clustering)
5452
		sdebug_driver_template.use_clustering = ENABLE_CLUSTERING;
5453 5454
	hpnt = scsi_host_alloc(&sdebug_driver_template, sizeof(sdbg_host));
	if (NULL == hpnt) {
5455
		pr_err("scsi_host_alloc failed\n");
5456
		error = -ENODEV;
L
Linus Torvalds 已提交
5457
		return error;
5458
	}
5459 5460 5461 5462 5463 5464 5465 5466 5467 5468
	if (submit_queues > nr_cpu_ids) {
		pr_warn("%s: trim submit_queues (was %d) to nr_cpu_ids=%d\n",
			my_name, submit_queues, nr_cpu_ids);
		submit_queues = nr_cpu_ids;
	}
	/* Decide whether to tell scsi subsystem that we want mq */
	/* Following should give the same answer for each host */
	sdebug_mq_active = shost_use_blk_mq(hpnt) && (submit_queues > 1);
	if (sdebug_mq_active)
		hpnt->nr_hw_queues = submit_queues;
L
Linus Torvalds 已提交
5469 5470 5471

        sdbg_host->shost = hpnt;
	*((struct sdebug_host_info **)hpnt->hostdata) = sdbg_host;
5472 5473
	if ((hpnt->this_id >= 0) && (sdebug_num_tgts > hpnt->this_id))
		hpnt->max_id = sdebug_num_tgts + 1;
L
Linus Torvalds 已提交
5474
	else
5475 5476
		hpnt->max_id = sdebug_num_tgts;
	/* = sdebug_max_luns; */
5477
	hpnt->max_lun = SCSI_W_LUN_REPORT_LUNS + 1;
L
Linus Torvalds 已提交
5478

5479
	hprot = 0;
5480

5481
	switch (sdebug_dif) {
5482 5483

	case SD_DIF_TYPE1_PROTECTION:
5484
		hprot = SHOST_DIF_TYPE1_PROTECTION;
5485
		if (sdebug_dix)
5486
			hprot |= SHOST_DIX_TYPE1_PROTECTION;
5487 5488 5489
		break;

	case SD_DIF_TYPE2_PROTECTION:
5490
		hprot = SHOST_DIF_TYPE2_PROTECTION;
5491
		if (sdebug_dix)
5492
			hprot |= SHOST_DIX_TYPE2_PROTECTION;
5493 5494 5495
		break;

	case SD_DIF_TYPE3_PROTECTION:
5496
		hprot = SHOST_DIF_TYPE3_PROTECTION;
5497
		if (sdebug_dix)
5498
			hprot |= SHOST_DIX_TYPE3_PROTECTION;
5499 5500 5501
		break;

	default:
5502
		if (sdebug_dix)
5503
			hprot |= SHOST_DIX_TYPE0_PROTECTION;
5504 5505 5506
		break;
	}

5507
	scsi_host_set_prot(hpnt, hprot);
5508

5509 5510 5511 5512 5513 5514 5515 5516 5517
	if (have_dif_prot || sdebug_dix)
		pr_info("host protection%s%s%s%s%s%s%s\n",
			(hprot & SHOST_DIF_TYPE1_PROTECTION) ? " DIF1" : "",
			(hprot & SHOST_DIF_TYPE2_PROTECTION) ? " DIF2" : "",
			(hprot & SHOST_DIF_TYPE3_PROTECTION) ? " DIF3" : "",
			(hprot & SHOST_DIX_TYPE0_PROTECTION) ? " DIX0" : "",
			(hprot & SHOST_DIX_TYPE1_PROTECTION) ? " DIX1" : "",
			(hprot & SHOST_DIX_TYPE2_PROTECTION) ? " DIX2" : "",
			(hprot & SHOST_DIX_TYPE3_PROTECTION) ? " DIX3" : "");
5518

5519
	if (sdebug_guard == 1)
5520 5521 5522 5523
		scsi_host_set_guard(hpnt, SHOST_DIX_GUARD_IP);
	else
		scsi_host_set_guard(hpnt, SHOST_DIX_GUARD_CRC);

5524 5525
	sdebug_verbose = !!(SDEBUG_OPT_NOISE & sdebug_opts);
	sdebug_any_injecting_opt = !!(SDEBUG_OPT_ALL_INJECTING & sdebug_opts);
5526 5527
	if (sdebug_every_nth)	/* need stats counters for every_nth */
		sdebug_statistics = true;
L
Linus Torvalds 已提交
5528 5529
        error = scsi_add_host(hpnt, &sdbg_host->dev);
        if (error) {
5530
		pr_err("scsi_add_host failed\n");
L
Linus Torvalds 已提交
5531 5532 5533 5534 5535
                error = -ENODEV;
		scsi_host_put(hpnt);
        } else
		scsi_scan_host(hpnt);

5536
	return error;
L
Linus Torvalds 已提交
5537 5538 5539 5540 5541
}

static int sdebug_driver_remove(struct device * dev)
{
        struct sdebug_host_info *sdbg_host;
5542
	struct sdebug_dev_info *sdbg_devinfo, *tmp;
L
Linus Torvalds 已提交
5543 5544 5545 5546

	sdbg_host = to_sdebug_host(dev);

	if (!sdbg_host) {
5547
		pr_err("Unable to locate host info\n");
L
Linus Torvalds 已提交
5548 5549 5550 5551 5552
		return -ENODEV;
	}

        scsi_remove_host(sdbg_host->shost);

5553 5554
	list_for_each_entry_safe(sdbg_devinfo, tmp, &sdbg_host->dev_info_list,
				 dev_list) {
L
Linus Torvalds 已提交
5555 5556 5557 5558 5559 5560 5561 5562
                list_del(&sdbg_devinfo->dev_list);
                kfree(sdbg_devinfo);
        }

        scsi_host_put(sdbg_host->shost);
        return 0;
}

5563 5564
static int pseudo_lld_bus_match(struct device *dev,
				struct device_driver *dev_driver)
L
Linus Torvalds 已提交
5565
{
5566
	return 1;
L
Linus Torvalds 已提交
5567
}
5568 5569 5570 5571 5572 5573

static struct bus_type pseudo_lld_bus = {
	.name = "pseudo",
	.match = pseudo_lld_bus_match,
	.probe = sdebug_driver_probe,
	.remove = sdebug_driver_remove,
5574
	.drv_groups = sdebug_drv_groups,
5575
};