ide-scsi.c 28.3 KB
Newer Older
Linus Torvalds's avatar
Linus Torvalds committed
1
/*
2
3
 * Copyright (C) 1996-1999  Gadi Oxman <gadio@netvision.net.il>
 * Copyright (C) 2004-2005  Bartlomiej Zolnierkiewicz
Linus Torvalds's avatar
Linus Torvalds committed
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
 */

/*
 * Emulation of a SCSI host adapter for IDE ATAPI devices.
 *
 * With this driver, one can use the Linux SCSI drivers instead of the
 * native IDE ATAPI drivers.
 *
 * Ver 0.1   Dec  3 96   Initial version.
 * Ver 0.2   Jan 26 97   Fixed bug in cleanup_module() and added emulation
 *                        of MODE_SENSE_6/MODE_SELECT_6 for cdroms. Thanks
 *                        to Janos Farkas for pointing this out.
 *                       Avoid using bitfields in structures for m68k.
 *                       Added Scatter/Gather and DMA support.
 * Ver 0.4   Dec  7 97   Add support for ATAPI PD/CD drives.
 *                       Use variable timeout for each command.
 * Ver 0.5   Jan  2 98   Fix previous PD/CD support.
 *                       Allow disabling of SCSI-6 to SCSI-10 transformation.
 * Ver 0.6   Jan 27 98   Allow disabling of SCSI command translation layer
 *                        for access through /dev/sg.
 *                       Fix MODE_SENSE_6/MODE_SELECT_6/INQUIRY translation.
 * Ver 0.7   Dec 04 98   Ignore commands where lun != 0 to avoid multiple
 *                        detection of devices with CONFIG_SCSI_MULTI_LUN
 * Ver 0.8   Feb 05 99   Optical media need translation too. Reverse 0.7.
 * Ver 0.9   Jul 04 99   Fix a bug in SG_SET_TRANSFORM.
 * Ver 0.91  Jun 10 02   Fix "off by one" error in transforms
 * Ver 0.92  Dec 31 02   Implement new SCSI mid level API
 */

#define IDESCSI_VERSION "0.92"

#include <linux/module.h>
#include <linux/types.h>
#include <linux/string.h>
#include <linux/kernel.h>
#include <linux/mm.h>
#include <linux/ioport.h>
#include <linux/blkdev.h>
#include <linux/errno.h>
#include <linux/hdreg.h>
#include <linux/slab.h>
#include <linux/ide.h>
#include <linux/scatterlist.h>
47
#include <linux/delay.h>
48
#include <linux/mutex.h>
Jiri Slaby's avatar
Jiri Slaby committed
49
#include <linux/bitops.h>
Linus Torvalds's avatar
Linus Torvalds committed
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78

#include <asm/io.h>
#include <asm/uaccess.h>

#include <scsi/scsi.h>
#include <scsi/scsi_cmnd.h>
#include <scsi/scsi_device.h>
#include <scsi/scsi_host.h>
#include <scsi/scsi_tcq.h>
#include <scsi/sg.h>

#define IDESCSI_DEBUG_LOG		0

/*
 *	SCSI command transformation layer
 */
#define IDESCSI_SG_TRANSFORM		1	/* /dev/sg transformation */

/*
 *	Log flags
 */
#define IDESCSI_LOG_CMD			0	/* Log SCSI commands */

typedef struct ide_scsi_obj {
	ide_drive_t		*drive;
	ide_driver_t		*driver;
	struct gendisk		*disk;
	struct Scsi_Host	*host;

79
	struct ide_atapi_pc *pc;		/* Current packet command */
Linus Torvalds's avatar
Linus Torvalds committed
80
81
82
83
84
	unsigned long flags;			/* Status/Action flags */
	unsigned long transform;		/* SCSI cmd translation layer */
	unsigned long log;			/* log flags */
} idescsi_scsi_t;

85
static DEFINE_MUTEX(idescsi_ref_mutex);
86
87
/* Set by module param to skip cd */
static int idescsi_nocd;
Linus Torvalds's avatar
Linus Torvalds committed
88
89
90
91
92
93
94
95

#define ide_scsi_g(disk) \
	container_of((disk)->private_data, struct ide_scsi_obj, driver)

static struct ide_scsi_obj *ide_scsi_get(struct gendisk *disk)
{
	struct ide_scsi_obj *scsi = NULL;

96
	mutex_lock(&idescsi_ref_mutex);
Linus Torvalds's avatar
Linus Torvalds committed
97
98
99
	scsi = ide_scsi_g(disk);
	if (scsi)
		scsi_host_get(scsi->host);
100
	mutex_unlock(&idescsi_ref_mutex);
Linus Torvalds's avatar
Linus Torvalds committed
101
102
103
104
105
	return scsi;
}

static void ide_scsi_put(struct ide_scsi_obj *scsi)
{
106
	mutex_lock(&idescsi_ref_mutex);
Linus Torvalds's avatar
Linus Torvalds committed
107
	scsi_host_put(scsi->host);
108
	mutex_unlock(&idescsi_ref_mutex);
Linus Torvalds's avatar
Linus Torvalds committed
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
}

static inline idescsi_scsi_t *scsihost_to_idescsi(struct Scsi_Host *host)
{
	return (idescsi_scsi_t*) (&host[1]);
}

static inline idescsi_scsi_t *drive_to_idescsi(ide_drive_t *ide_drive)
{
	return scsihost_to_idescsi(ide_drive->driver_data);
}

/*
 *	Per ATAPI device status bits.
 */
#define IDESCSI_DRQ_INTERRUPT		0	/* DRQ interrupt device */

/*
 *	ide-scsi requests.
 */
#define IDESCSI_PC_RQ			90

/*
 *	PIO data transfer routines using the scatter gather table.
 */
134
135
static void idescsi_input_buffers(ide_drive_t *drive, struct ide_atapi_pc *pc,
		unsigned int bcount)
Linus Torvalds's avatar
Linus Torvalds committed
136
{
137
	ide_hwif_t *hwif = drive->hwif;
Linus Torvalds's avatar
Linus Torvalds committed
138
139
140
141
142
	int count;
	char *buf;

	while (bcount) {
		count = min(pc->sg->length - pc->b_count, bcount);
143
		if (PageHighMem(sg_page(pc->sg))) {
144
145
146
			unsigned long flags;

			local_irq_save(flags);
147
			buf = kmap_atomic(sg_page(pc->sg), KM_IRQ0) +
148
					pc->sg->offset;
149
			hwif->input_data(drive, NULL, buf + pc->b_count, count);
150
151
152
			kunmap_atomic(buf - pc->sg->offset, KM_IRQ0);
			local_irq_restore(flags);
		} else {
153
			buf = sg_virt(pc->sg);
154
			hwif->input_data(drive, NULL, buf + pc->b_count, count);
155
156
		}
		bcount -= count; pc->b_count += count;
Linus Torvalds's avatar
Linus Torvalds committed
157
		if (pc->b_count == pc->sg->length) {
158
			if (!--pc->sg_cnt)
Jens Axboe's avatar
Jens Axboe committed
159
160
				break;
			pc->sg = sg_next(pc->sg);
Linus Torvalds's avatar
Linus Torvalds committed
161
162
163
			pc->b_count = 0;
		}
	}
Jens Axboe's avatar
Jens Axboe committed
164
165
166

	if (bcount) {
		printk (KERN_ERR "ide-scsi: scatter gather table too small, discarding data\n");
167
		ide_pad_transfer(drive, 0, bcount);
Jens Axboe's avatar
Jens Axboe committed
168
	}
Linus Torvalds's avatar
Linus Torvalds committed
169
170
}

171
172
static void idescsi_output_buffers(ide_drive_t *drive, struct ide_atapi_pc *pc,
		unsigned int bcount)
Linus Torvalds's avatar
Linus Torvalds committed
173
{
174
	ide_hwif_t *hwif = drive->hwif;
Linus Torvalds's avatar
Linus Torvalds committed
175
176
177
178
179
	int count;
	char *buf;

	while (bcount) {
		count = min(pc->sg->length - pc->b_count, bcount);
180
		if (PageHighMem(sg_page(pc->sg))) {
181
182
183
			unsigned long flags;

			local_irq_save(flags);
184
			buf = kmap_atomic(sg_page(pc->sg), KM_IRQ0) +
185
						pc->sg->offset;
186
			hwif->output_data(drive, NULL, buf + pc->b_count, count);
187
188
189
			kunmap_atomic(buf - pc->sg->offset, KM_IRQ0);
			local_irq_restore(flags);
		} else {
190
			buf = sg_virt(pc->sg);
191
			hwif->output_data(drive, NULL, buf + pc->b_count, count);
192
193
		}
		bcount -= count; pc->b_count += count;
Linus Torvalds's avatar
Linus Torvalds committed
194
		if (pc->b_count == pc->sg->length) {
195
			if (!--pc->sg_cnt)
Jens Axboe's avatar
Jens Axboe committed
196
197
				break;
			pc->sg = sg_next(pc->sg);
Linus Torvalds's avatar
Linus Torvalds committed
198
199
200
			pc->b_count = 0;
		}
	}
Jens Axboe's avatar
Jens Axboe committed
201
202
203

	if (bcount) {
		printk (KERN_ERR "ide-scsi: scatter gather table too small, padding with zeros\n");
204
		ide_pad_transfer(drive, 1, bcount);
Jens Axboe's avatar
Jens Axboe committed
205
	}
Linus Torvalds's avatar
Linus Torvalds committed
206
207
}

208
209
210
211
212
static void ide_scsi_hex_dump(u8 *data, int len)
{
	print_hex_dump(KERN_CONT, "", DUMP_PREFIX_NONE, 16, 1, data, len, 0);
}

213
214
static int idescsi_check_condition(ide_drive_t *drive,
		struct request *failed_cmd)
Linus Torvalds's avatar
Linus Torvalds committed
215
216
{
	idescsi_scsi_t *scsi = drive_to_idescsi(drive);
217
	struct ide_atapi_pc   *pc;
Linus Torvalds's avatar
Linus Torvalds committed
218
219
220
221
	struct request *rq;
	u8             *buf;

	/* stuff a sense request in front of our current request */
222
	pc = kzalloc(sizeof(struct ide_atapi_pc), GFP_ATOMIC);
223
224
225
	rq = kmalloc(sizeof(struct request), GFP_ATOMIC);
	buf = kzalloc(SCSI_SENSE_BUFFERSIZE, GFP_ATOMIC);
	if (!pc || !rq || !buf) {
226
227
228
		kfree(buf);
		kfree(rq);
		kfree(pc);
Linus Torvalds's avatar
Linus Torvalds committed
229
230
		return -ENOMEM;
	}
231
	blk_rq_init(NULL, rq);
Linus Torvalds's avatar
Linus Torvalds committed
232
233
	rq->special = (char *) pc;
	pc->rq = rq;
234
	pc->buf = buf;
Linus Torvalds's avatar
Linus Torvalds committed
235
	pc->c[0] = REQUEST_SENSE;
236
	pc->c[4] = pc->req_xfer = pc->buf_size = SCSI_SENSE_BUFFERSIZE;
237
	rq->cmd_type = REQ_TYPE_SENSE;
238
	rq->cmd_flags |= REQ_PREEMPT;
Linus Torvalds's avatar
Linus Torvalds committed
239
240
	pc->timeout = jiffies + WAIT_READY;
	/* NOTE! Save the failed packet command in "rq->buffer" */
241
242
	rq->buffer = (void *) failed_cmd->special;
	pc->scsi_cmd = ((struct ide_atapi_pc *) failed_cmd->special)->scsi_cmd;
Linus Torvalds's avatar
Linus Torvalds committed
243
244
	if (test_bit(IDESCSI_LOG_CMD, &scsi->log)) {
		printk ("ide-scsi: %s: queue cmd = ", drive->name);
245
		ide_scsi_hex_dump(pc->c, 6);
Linus Torvalds's avatar
Linus Torvalds committed
246
247
	}
	rq->rq_disk = scsi->disk;
248
249
	ide_do_drive_cmd(drive, rq);
	return 0;
Linus Torvalds's avatar
Linus Torvalds committed
250
251
252
253
254
255
256
}

static int idescsi_end_request(ide_drive_t *, int, int);

static ide_startstop_t
idescsi_atapi_error(ide_drive_t *drive, struct request *rq, u8 stat, u8 err)
{
257
258
	ide_hwif_t *hwif = drive->hwif;

259
	if (ide_read_status(drive) & (BUSY_STAT | DRQ_STAT))
Linus Torvalds's avatar
Linus Torvalds committed
260
		/* force an abort */
261
		hwif->OUTBSYNC(hwif, WIN_IDLEIMMEDIATE,
262
			       hwif->io_ports.command_addr);
Linus Torvalds's avatar
Linus Torvalds committed
263
264
265
266
267
268
269
270
271
272
273
274
275

	rq->errors++;

	idescsi_end_request(drive, 0, 0);

	return ide_stopped;
}

static ide_startstop_t
idescsi_atapi_abort(ide_drive_t *drive, struct request *rq)
{
#if IDESCSI_DEBUG_LOG
	printk(KERN_WARNING "idescsi_atapi_abort called for %lu\n",
276
		((struct ide_atapi_pc *) rq->special)->scsi_cmd->serial_number);
Linus Torvalds's avatar
Linus Torvalds committed
277
278
279
280
281
282
283
284
285
286
287
288
#endif
	rq->errors |= ERROR_MAX;

	idescsi_end_request(drive, 0, 0);

	return ide_stopped;
}

static int idescsi_end_request (ide_drive_t *drive, int uptodate, int nrsecs)
{
	idescsi_scsi_t *scsi = drive_to_idescsi(drive);
	struct request *rq = HWGROUP(drive)->rq;
289
	struct ide_atapi_pc *pc = (struct ide_atapi_pc *) rq->special;
Linus Torvalds's avatar
Linus Torvalds committed
290
291
	int log = test_bit(IDESCSI_LOG_CMD, &scsi->log);
	struct Scsi_Host *host;
292
	int errors = rq->errors;
Linus Torvalds's avatar
Linus Torvalds committed
293
294
	unsigned long flags;

295
	if (!blk_special_request(rq) && !blk_sense_request(rq)) {
Linus Torvalds's avatar
Linus Torvalds committed
296
297
298
299
		ide_end_request(drive, uptodate, nrsecs);
		return 0;
	}
	ide_end_drive_cmd (drive, 0, 0);
300
	if (blk_sense_request(rq)) {
301
		struct ide_atapi_pc *opc = (struct ide_atapi_pc *) rq->buffer;
Linus Torvalds's avatar
Linus Torvalds committed
302
303
		if (log) {
			printk ("ide-scsi: %s: wrap up check %lu, rst = ", drive->name, opc->scsi_cmd->serial_number);
304
			ide_scsi_hex_dump(pc->buf, 16);
Linus Torvalds's avatar
Linus Torvalds committed
305
		}
306
307
308
		memcpy((void *) opc->scsi_cmd->sense_buffer, pc->buf,
			SCSI_SENSE_BUFFERSIZE);
		kfree(pc->buf);
Linus Torvalds's avatar
Linus Torvalds committed
309
310
311
312
313
		kfree(pc);
		kfree(rq);
		pc = opc;
		rq = pc->rq;
		pc->scsi_cmd->result = (CHECK_CONDITION << 1) |
314
315
316
317
				(((pc->flags & PC_FLAG_TIMEDOUT) ?
				  DID_TIME_OUT :
				  DID_OK) << 16);
	} else if (pc->flags & PC_FLAG_TIMEDOUT) {
Linus Torvalds's avatar
Linus Torvalds committed
318
319
320
321
		if (log)
			printk (KERN_WARNING "ide-scsi: %s: timed out for %lu\n",
					drive->name, pc->scsi_cmd->serial_number);
		pc->scsi_cmd->result = DID_TIME_OUT << 16;
322
	} else if (errors >= ERROR_MAX) {
Linus Torvalds's avatar
Linus Torvalds committed
323
324
325
		pc->scsi_cmd->result = DID_ERROR << 16;
		if (log)
			printk ("ide-scsi: %s: I/O error for %lu\n", drive->name, pc->scsi_cmd->serial_number);
326
	} else if (errors) {
Linus Torvalds's avatar
Linus Torvalds committed
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
		if (log)
			printk ("ide-scsi: %s: check condition for %lu\n", drive->name, pc->scsi_cmd->serial_number);
		if (!idescsi_check_condition(drive, rq))
			/* we started a request sense, so we'll be back, exit for now */
			return 0;
		pc->scsi_cmd->result = (CHECK_CONDITION << 1) | (DID_OK << 16);
	} else {
		pc->scsi_cmd->result = DID_OK << 16;
	}
	host = pc->scsi_cmd->device->host;
	spin_lock_irqsave(host->host_lock, flags);
	pc->done(pc->scsi_cmd);
	spin_unlock_irqrestore(host->host_lock, flags);
	kfree(pc);
	kfree(rq);
	scsi->pc = NULL;
	return 0;
}

346
static inline unsigned long get_timeout(struct ide_atapi_pc *pc)
Linus Torvalds's avatar
Linus Torvalds committed
347
348
349
350
351
352
{
	return max_t(unsigned long, WAIT_CMD, pc->timeout - jiffies);
}

static int idescsi_expiry(ide_drive_t *drive)
{
353
	idescsi_scsi_t *scsi = drive_to_idescsi(drive);
354
	struct ide_atapi_pc   *pc   = scsi->pc;
Linus Torvalds's avatar
Linus Torvalds committed
355
356
357
358

#if IDESCSI_DEBUG_LOG
	printk(KERN_WARNING "idescsi_expiry called for %lu at %lu\n", pc->scsi_cmd->serial_number, jiffies);
#endif
359
	pc->flags |= PC_FLAG_TIMEDOUT;
Linus Torvalds's avatar
Linus Torvalds committed
360
361
362
363
364
365
366
367
368
369

	return 0;					/* we do not want the ide subsystem to retry */
}

/*
 *	Our interrupt handler.
 */
static ide_startstop_t idescsi_pc_intr (ide_drive_t *drive)
{
	idescsi_scsi_t *scsi = drive_to_idescsi(drive);
370
	ide_hwif_t *hwif = drive->hwif;
371
	struct ide_atapi_pc *pc = scsi->pc;
Linus Torvalds's avatar
Linus Torvalds committed
372
373
	struct request *rq = pc->rq;
	unsigned int temp;
374
	u16 bcount;
375
	u8 stat, ireason;
Linus Torvalds's avatar
Linus Torvalds committed
376
377
378
379
380

#if IDESCSI_DEBUG_LOG
	printk (KERN_INFO "ide-scsi: Reached idescsi_pc_intr interrupt handler\n");
#endif /* IDESCSI_DEBUG_LOG */

381
	if (pc->flags & PC_FLAG_TIMEDOUT) {
Linus Torvalds's avatar
Linus Torvalds committed
382
383
384
385
386
387
388
389
#if IDESCSI_DEBUG_LOG
		printk(KERN_WARNING "idescsi_pc_intr: got timed out packet  %lu at %lu\n",
				pc->scsi_cmd->serial_number, jiffies);
#endif
		/* end this request now - scsi should retry it*/
		idescsi_end_request (drive, 1, 0);
		return ide_stopped;
	}
390
391
	if (pc->flags & PC_FLAG_DMA_IN_PROGRESS) {
		pc->flags &= ~PC_FLAG_DMA_IN_PROGRESS;
Linus Torvalds's avatar
Linus Torvalds committed
392
393
394
#if IDESCSI_DEBUG_LOG
		printk ("ide-scsi: %s: DMA complete\n", drive->name);
#endif /* IDESCSI_DEBUG_LOG */
395
		pc->xferred = pc->req_xfer;
396
		(void)hwif->dma_ops->dma_end(drive);
Linus Torvalds's avatar
Linus Torvalds committed
397
398
399
	}

	/* Clear the interrupt */
400
	stat = ide_read_status(drive);
Linus Torvalds's avatar
Linus Torvalds committed
401

402
	if ((stat & DRQ_STAT) == 0) {
Linus Torvalds's avatar
Linus Torvalds committed
403
404
		/* No more interrupts */
		if (test_bit(IDESCSI_LOG_CMD, &scsi->log))
405
406
			printk(KERN_INFO "Packet command completed, %d bytes"
					" transferred\n", pc->xferred);
407
		local_irq_enable_in_hardirq();
408
		if (stat & ERR_STAT)
Linus Torvalds's avatar
Linus Torvalds committed
409
410
411
412
			rq->errors++;
		idescsi_end_request (drive, 1, 0);
		return ide_stopped;
	}
413
414
415
	bcount = (hwif->INB(hwif->io_ports.lbah_addr) << 8) |
		  hwif->INB(hwif->io_ports.lbam_addr);
	ireason = hwif->INB(hwif->io_ports.nsect_addr);
Linus Torvalds's avatar
Linus Torvalds committed
416

417
	if (ireason & CD) {
Linus Torvalds's avatar
Linus Torvalds committed
418
419
420
		printk(KERN_ERR "ide-scsi: CoD != 0 in idescsi_pc_intr\n");
		return ide_do_reset (drive);
	}
421
	if (ireason & IO) {
422
423
424
		temp = pc->xferred + bcount;
		if (temp > pc->req_xfer) {
			if (temp > pc->buf_size) {
Linus Torvalds's avatar
Linus Torvalds committed
425
426
427
				printk(KERN_ERR "ide-scsi: The scsi wants to "
					"send us more data than expected "
					"- discarding data\n");
428
				temp = pc->buf_size - pc->xferred;
Linus Torvalds's avatar
Linus Torvalds committed
429
				if (temp) {
430
					pc->flags &= ~PC_FLAG_WRITING;
Linus Torvalds's avatar
Linus Torvalds committed
431
					if (pc->sg)
432
433
						idescsi_input_buffers(drive, pc,
									temp);
Linus Torvalds's avatar
Linus Torvalds committed
434
					else
435
436
						hwif->input_data(drive, NULL,
							pc->cur_pos, temp);
437
438
439
					printk(KERN_ERR "ide-scsi: transferred"
							" %d of %d bytes\n",
							temp, bcount);
Linus Torvalds's avatar
Linus Torvalds committed
440
				}
441
442
				pc->xferred += temp;
				pc->cur_pos += temp;
443
				ide_pad_transfer(drive, 0, bcount - temp);
Linus Torvalds's avatar
Linus Torvalds committed
444
445
446
447
448
449
450
451
				ide_set_handler(drive, &idescsi_pc_intr, get_timeout(pc), idescsi_expiry);
				return ide_started;
			}
#if IDESCSI_DEBUG_LOG
			printk (KERN_NOTICE "ide-scsi: The scsi wants to send us more data than expected - allowing transfer\n");
#endif /* IDESCSI_DEBUG_LOG */
		}
	}
452
	if (ireason & IO) {
453
		pc->flags &= ~PC_FLAG_WRITING;
Linus Torvalds's avatar
Linus Torvalds committed
454
		if (pc->sg)
455
			idescsi_input_buffers(drive, pc, bcount);
Linus Torvalds's avatar
Linus Torvalds committed
456
		else
457
			hwif->input_data(drive, NULL, pc->cur_pos, bcount);
Linus Torvalds's avatar
Linus Torvalds committed
458
	} else {
459
		pc->flags |= PC_FLAG_WRITING;
Linus Torvalds's avatar
Linus Torvalds committed
460
		if (pc->sg)
461
			idescsi_output_buffers(drive, pc, bcount);
Linus Torvalds's avatar
Linus Torvalds committed
462
		else
463
			hwif->output_data(drive, NULL, pc->cur_pos, bcount);
Linus Torvalds's avatar
Linus Torvalds committed
464
465
	}
	/* Update the current position */
466
467
	pc->xferred += bcount;
	pc->cur_pos += bcount;
Linus Torvalds's avatar
Linus Torvalds committed
468
469
470
471
472
473
474
475
476
477

	/* And set the interrupt handler again */
	ide_set_handler(drive, &idescsi_pc_intr, get_timeout(pc), idescsi_expiry);
	return ide_started;
}

static ide_startstop_t idescsi_transfer_pc(ide_drive_t *drive)
{
	ide_hwif_t *hwif = drive->hwif;
	idescsi_scsi_t *scsi = drive_to_idescsi(drive);
478
	struct ide_atapi_pc *pc = scsi->pc;
Linus Torvalds's avatar
Linus Torvalds committed
479
	ide_startstop_t startstop;
480
	u8 ireason;
Linus Torvalds's avatar
Linus Torvalds committed
481
482
483
484
485
486

	if (ide_wait_stat(&startstop,drive,DRQ_STAT,BUSY_STAT,WAIT_READY)) {
		printk(KERN_ERR "ide-scsi: Strange, packet command "
			"initiated yet DRQ isn't asserted\n");
		return startstop;
	}
487
	ireason = hwif->INB(hwif->io_ports.nsect_addr);
488
	if ((ireason & CD) == 0 || (ireason & IO)) {
Linus Torvalds's avatar
Linus Torvalds committed
489
490
491
492
		printk(KERN_ERR "ide-scsi: (IO,CoD) != (0,1) while "
				"issuing a packet command\n");
		return ide_do_reset (drive);
	}
493
	BUG_ON(HWGROUP(drive)->handler != NULL);
Linus Torvalds's avatar
Linus Torvalds committed
494
495
	/* Set the interrupt routine */
	ide_set_handler(drive, &idescsi_pc_intr, get_timeout(pc), idescsi_expiry);
496

497
498
	if (pc->flags & PC_FLAG_DMA_OK) {
		pc->flags |= PC_FLAG_DMA_IN_PROGRESS;
499
		hwif->dma_ops->dma_start(drive);
Linus Torvalds's avatar
Linus Torvalds committed
500
	}
501
502
503
504

	/* Send the actual packet */
	hwif->output_data(drive, NULL, scsi->pc->c, 12);

Linus Torvalds's avatar
Linus Torvalds committed
505
506
507
	return ide_started;
}

508
static inline int idescsi_set_direction(struct ide_atapi_pc *pc)
Linus Torvalds's avatar
Linus Torvalds committed
509
510
511
{
	switch (pc->c[0]) {
		case READ_6: case READ_10: case READ_12:
512
			pc->flags &= ~PC_FLAG_WRITING;
Linus Torvalds's avatar
Linus Torvalds committed
513
514
			return 0;
		case WRITE_6: case WRITE_10: case WRITE_12:
515
			pc->flags |= PC_FLAG_WRITING;
Linus Torvalds's avatar
Linus Torvalds committed
516
517
518
519
520
521
			return 0;
		default:
			return 1;
	}
}

522
static int idescsi_map_sg(ide_drive_t *drive, struct ide_atapi_pc *pc)
Linus Torvalds's avatar
Linus Torvalds committed
523
524
525
526
527
{
	ide_hwif_t *hwif = drive->hwif;
	struct scatterlist *sg, *scsi_sg;
	int segments;

528
	if (!pc->req_xfer || pc->req_xfer % 1024)
Linus Torvalds's avatar
Linus Torvalds committed
529
530
531
532
533
534
		return 1;

	if (idescsi_set_direction(pc))
		return 1;

	sg = hwif->sg_table;
535
536
	scsi_sg = scsi_sglist(pc->scsi_cmd);
	segments = scsi_sg_count(pc->scsi_cmd);
Linus Torvalds's avatar
Linus Torvalds committed
537
538
539
540

	if (segments > hwif->sg_max_nents)
		return 1;

541
542
	hwif->sg_nents = segments;
	memcpy(sg, scsi_sg, sizeof(*sg) * segments);
Linus Torvalds's avatar
Linus Torvalds committed
543
544
545
546

	return 0;
}

547
548
static ide_startstop_t idescsi_issue_pc(ide_drive_t *drive,
		struct ide_atapi_pc *pc)
Linus Torvalds's avatar
Linus Torvalds committed
549
550
551
{
	idescsi_scsi_t *scsi = drive_to_idescsi(drive);
	ide_hwif_t *hwif = drive->hwif;
552
	u16 bcount;
553
	u8 dma = 0;
Linus Torvalds's avatar
Linus Torvalds committed
554

555
556
557
558
559
	/* Set the current packet command */
	scsi->pc = pc;
	/* We haven't transferred any data yet */
	pc->xferred = 0;
	pc->cur_pos = pc->buf;
560
	/* Request to transfer the entire buffer at once */
561
	bcount = min(pc->req_xfer, 63 * 1024);
Linus Torvalds's avatar
Linus Torvalds committed
562
563
564

	if (drive->using_dma && !idescsi_map_sg(drive, pc)) {
		hwif->sg_mapped = 1;
565
		dma = !hwif->dma_ops->dma_setup(drive);
Linus Torvalds's avatar
Linus Torvalds committed
566
567
568
		hwif->sg_mapped = 0;
	}

569
	ide_pktcmd_tf_load(drive, 0, bcount, dma);
Linus Torvalds's avatar
Linus Torvalds committed
570

571
	if (dma)
572
		pc->flags |= PC_FLAG_DMA_OK;
Linus Torvalds's avatar
Linus Torvalds committed
573
574

	if (test_bit(IDESCSI_DRQ_INTERRUPT, &scsi->flags)) {
575
576
		ide_execute_command(drive, WIN_PACKETCMD, &idescsi_transfer_pc,
				    get_timeout(pc), idescsi_expiry);
Linus Torvalds's avatar
Linus Torvalds committed
577
578
579
		return ide_started;
	} else {
		/* Issue the packet command */
580
		ide_execute_pkt_cmd(drive);
Linus Torvalds's avatar
Linus Torvalds committed
581
582
583
584
585
586
587
588
589
590
		return idescsi_transfer_pc(drive);
	}
}

/*
 *	idescsi_do_request is our request handling function.
 */
static ide_startstop_t idescsi_do_request (ide_drive_t *drive, struct request *rq, sector_t block)
{
#if IDESCSI_DEBUG_LOG
591
	printk (KERN_INFO "dev: %s, cmd: %x, errors: %d\n", rq->rq_disk->disk_name,rq->cmd[0],rq->errors);
Linus Torvalds's avatar
Linus Torvalds committed
592
593
594
	printk (KERN_INFO "sector: %ld, nr_sectors: %ld, current_nr_sectors: %d\n",rq->sector,rq->nr_sectors,rq->current_nr_sectors);
#endif /* IDESCSI_DEBUG_LOG */

595
	if (blk_sense_request(rq) || blk_special_request(rq)) {
596
597
		return idescsi_issue_pc(drive,
				(struct ide_atapi_pc *) rq->special);
Linus Torvalds's avatar
Linus Torvalds committed
598
599
600
601
602
603
	}
	blk_dump_rq_flags(rq, "ide-scsi: unsup command");
	idescsi_end_request (drive, 0, 0);
	return ide_stopped;
}

604
#ifdef CONFIG_IDE_PROC_FS
Linus Torvalds's avatar
Linus Torvalds committed
605
606
607
608
609
static void idescsi_add_settings(ide_drive_t *drive)
{
	idescsi_scsi_t *scsi = drive_to_idescsi(drive);

/*
610
 *			drive	setting name	read/write	data type	min	max	mul_factor	div_factor	data pointer		set function
Linus Torvalds's avatar
Linus Torvalds committed
611
 */
612
613
614
615
616
	ide_add_setting(drive,	"bios_cyl",	SETTING_RW,	TYPE_INT,	0,	1023,	1,		1,		&drive->bios_cyl,	NULL);
	ide_add_setting(drive,	"bios_head",	SETTING_RW,	TYPE_BYTE,	0,	255,	1,		1,		&drive->bios_head,	NULL);
	ide_add_setting(drive,	"bios_sect",	SETTING_RW,	TYPE_BYTE,	0,	63,	1,		1,		&drive->bios_sect,	NULL);
	ide_add_setting(drive,	"transform",	SETTING_RW,	TYPE_INT,	0,	3,	1,		1,		&scsi->transform,	NULL);
	ide_add_setting(drive,	"log",		SETTING_RW,	TYPE_INT,	0,	1,	1,		1,		&scsi->log,		NULL);
Linus Torvalds's avatar
Linus Torvalds committed
617
}
618
619
620
#else
static inline void idescsi_add_settings(ide_drive_t *drive) { ; }
#endif
Linus Torvalds's avatar
Linus Torvalds committed
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635

/*
 *	Driver initialization.
 */
static void idescsi_setup (ide_drive_t *drive, idescsi_scsi_t *scsi)
{
	if (drive->id && (drive->id->config & 0x0060) == 0x20)
		set_bit (IDESCSI_DRQ_INTERRUPT, &scsi->flags);
	clear_bit(IDESCSI_SG_TRANSFORM, &scsi->transform);
#if IDESCSI_DEBUG_LOG
	set_bit(IDESCSI_LOG_CMD, &scsi->log);
#endif /* IDESCSI_DEBUG_LOG */
	idescsi_add_settings(drive);
}

636
static void ide_scsi_remove(ide_drive_t *drive)
Linus Torvalds's avatar
Linus Torvalds committed
637
638
639
640
641
{
	struct Scsi_Host *scsihost = drive->driver_data;
	struct ide_scsi_obj *scsi = scsihost_to_idescsi(scsihost);
	struct gendisk *g = scsi->disk;

642
	scsi_remove_host(scsihost);
643
	ide_proc_unregister_driver(drive, scsi->driver);
Linus Torvalds's avatar
Linus Torvalds committed
644
645
646
647
648
649
650
651
652
653

	ide_unregister_region(g);

	drive->driver_data = NULL;
	g->private_data = NULL;
	put_disk(g);

	ide_scsi_put(scsi);
}

654
static int ide_scsi_probe(ide_drive_t *);
Linus Torvalds's avatar
Linus Torvalds committed
655

656
#ifdef CONFIG_IDE_PROC_FS
Linus Torvalds's avatar
Linus Torvalds committed
657
658
659
660
661
662
663
static ide_proc_entry_t idescsi_proc[] = {
	{ "capacity", S_IFREG|S_IRUGO, proc_ide_read_capacity, NULL },
	{ NULL, 0, NULL, NULL }
};
#endif

static ide_driver_t idescsi_driver = {
664
	.gen_driver = {
665
		.owner		= THIS_MODULE,
666
667
668
		.name		= "ide-scsi",
		.bus		= &ide_bus_type,
	},
669
670
	.probe			= ide_scsi_probe,
	.remove			= ide_scsi_remove,
Linus Torvalds's avatar
Linus Torvalds committed
671
672
673
674
675
676
677
	.version		= IDESCSI_VERSION,
	.media			= ide_scsi,
	.supports_dsc_overlap	= 0,
	.do_request		= idescsi_do_request,
	.end_request		= idescsi_end_request,
	.error                  = idescsi_atapi_error,
	.abort                  = idescsi_atapi_abort,
678
679
680
#ifdef CONFIG_IDE_PROC_FS
	.proc			= idescsi_proc,
#endif
Linus Torvalds's avatar
Linus Torvalds committed
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
};

static int idescsi_ide_open(struct inode *inode, struct file *filp)
{
	struct gendisk *disk = inode->i_bdev->bd_disk;
	struct ide_scsi_obj *scsi;

	if (!(scsi = ide_scsi_get(disk)))
		return -ENXIO;

	return 0;
}

static int idescsi_ide_release(struct inode *inode, struct file *filp)
{
	struct gendisk *disk = inode->i_bdev->bd_disk;
	struct ide_scsi_obj *scsi = ide_scsi_g(disk);

	ide_scsi_put(scsi);

	return 0;
}

static int idescsi_ide_ioctl(struct inode *inode, struct file *file,
			unsigned int cmd, unsigned long arg)
{
	struct block_device *bdev = inode->i_bdev;
	struct ide_scsi_obj *scsi = ide_scsi_g(bdev->bd_disk);
	return generic_ide_ioctl(scsi->drive, file, bdev, cmd, arg);
}

static struct block_device_operations idescsi_ops = {
	.owner		= THIS_MODULE,
	.open		= idescsi_ide_open,
	.release	= idescsi_ide_release,
	.ioctl		= idescsi_ide_ioctl,
};

static int idescsi_slave_configure(struct scsi_device * sdp)
{
	/* Configure detected device */
722
723
	sdp->use_10_for_rw = 1;
	sdp->use_10_for_ms = 1;
Linus Torvalds's avatar
Linus Torvalds committed
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
	scsi_adjust_queue_depth(sdp, MSG_SIMPLE_TAG, sdp->host->cmd_per_lun);
	return 0;
}

static const char *idescsi_info (struct Scsi_Host *host)
{
	return "SCSI host adapter emulation for IDE ATAPI devices";
}

static int idescsi_ioctl (struct scsi_device *dev, int cmd, void __user *arg)
{
	idescsi_scsi_t *scsi = scsihost_to_idescsi(dev->host);

	if (cmd == SG_SET_TRANSFORM) {
		if (arg)
			set_bit(IDESCSI_SG_TRANSFORM, &scsi->transform);
		else
			clear_bit(IDESCSI_SG_TRANSFORM, &scsi->transform);
		return 0;
	} else if (cmd == SG_GET_TRANSFORM)
		return put_user(test_bit(IDESCSI_SG_TRANSFORM, &scsi->transform), (int __user *) arg);
	return -EINVAL;
}

static int idescsi_queue (struct scsi_cmnd *cmd,
		void (*done)(struct scsi_cmnd *))
{
	struct Scsi_Host *host = cmd->device->host;
	idescsi_scsi_t *scsi = scsihost_to_idescsi(host);
	ide_drive_t *drive = scsi->drive;
	struct request *rq = NULL;
755
	struct ide_atapi_pc *pc = NULL;
Linus Torvalds's avatar
Linus Torvalds committed
756
757

	if (!drive) {
758
		scmd_printk (KERN_ERR, cmd, "drive not present\n");
Linus Torvalds's avatar
Linus Torvalds committed
759
760
761
		goto abort;
	}
	scsi = drive_to_idescsi(drive);
762
763
	pc = kmalloc(sizeof(struct ide_atapi_pc), GFP_ATOMIC);
	rq = kmalloc(sizeof(struct request), GFP_ATOMIC);
Linus Torvalds's avatar
Linus Torvalds committed
764
765
766
767
768
769
770
771
772
	if (rq == NULL || pc == NULL) {
		printk (KERN_ERR "ide-scsi: %s: out of memory\n", drive->name);
		goto abort;
	}

	memset (pc->c, 0, 12);
	pc->flags = 0;
	pc->rq = rq;
	memcpy (pc->c, cmd->cmnd, cmd->cmd_len);
773
	pc->buf = NULL;
774
	pc->sg = scsi_sglist(cmd);
775
	pc->sg_cnt = scsi_sg_count(cmd);
Linus Torvalds's avatar
Linus Torvalds committed
776
	pc->b_count = 0;
777
	pc->req_xfer = pc->buf_size = scsi_bufflen(cmd);
Linus Torvalds's avatar
Linus Torvalds committed
778
779
780
781
782
783
	pc->scsi_cmd = cmd;
	pc->done = done;
	pc->timeout = jiffies + cmd->timeout_per_command;

	if (test_bit(IDESCSI_LOG_CMD, &scsi->log)) {
		printk ("ide-scsi: %s: que %lu, cmd = ", drive->name, cmd->serial_number);
784
		ide_scsi_hex_dump(cmd->cmnd, cmd->cmd_len);
Linus Torvalds's avatar
Linus Torvalds committed
785
786
		if (memcmp(pc->c, cmd->cmnd, cmd->cmd_len)) {
			printk ("ide-scsi: %s: que %lu, tsl = ", drive->name, cmd->serial_number);
787
			ide_scsi_hex_dump(pc->c, 12);
Linus Torvalds's avatar
Linus Torvalds committed
788
789
790
		}
	}

791
	blk_rq_init(NULL, rq);
Linus Torvalds's avatar
Linus Torvalds committed
792
	rq->special = (char *) pc;
793
	rq->cmd_type = REQ_TYPE_SPECIAL;
Linus Torvalds's avatar
Linus Torvalds committed
794
	spin_unlock_irq(host->host_lock);
795
	blk_execute_rq_nowait(drive->queue, scsi->disk, rq, 0, NULL);
Linus Torvalds's avatar
Linus Torvalds committed
796
797
798
	spin_lock_irq(host->host_lock);
	return 0;
abort:
799
800
	kfree (pc);
	kfree (rq);
Linus Torvalds's avatar
Linus Torvalds committed
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
	cmd->result = DID_ERROR << 16;
	done(cmd);
	return 0;
}

static int idescsi_eh_abort (struct scsi_cmnd *cmd)
{
	idescsi_scsi_t *scsi  = scsihost_to_idescsi(cmd->device->host);
	ide_drive_t    *drive = scsi->drive;
	int		busy;
	int             ret   = FAILED;

	/* In idescsi_eh_abort we try to gently pry our command from the ide subsystem */

	if (test_bit(IDESCSI_LOG_CMD, &scsi->log))
		printk (KERN_WARNING "ide-scsi: abort called for %lu\n", cmd->serial_number);

	if (!drive) {
		printk (KERN_WARNING "ide-scsi: Drive not set in idescsi_eh_abort\n");
		WARN_ON(1);
		goto no_drive;
	}

	/* First give it some more time, how much is "right" is hard to say :-( */

	busy = ide_wait_not_busy(HWIF(drive), 100);	/* FIXME - uses mdelay which causes latency? */
	if (test_bit(IDESCSI_LOG_CMD, &scsi->log))
		printk (KERN_WARNING "ide-scsi: drive did%s become ready\n", busy?" not":"");

	spin_lock_irq(&ide_lock);

	/* If there is no pc running we're done (our interrupt took care of it) */
	if (!scsi->pc) {
		ret = SUCCESS;
		goto ide_unlock;
	}

	/* It's somewhere in flight. Does ide subsystem agree? */
	if (scsi->pc->scsi_cmd->serial_number == cmd->serial_number && !busy &&
	    elv_queue_empty(drive->queue) && HWGROUP(drive)->rq != scsi->pc->rq) {
		/*
		 * FIXME - not sure this condition can ever occur
		 */
		printk (KERN_ERR "ide-scsi: cmd aborted!\n");

846
		if (blk_sense_request(scsi->pc->rq))
847
			kfree(scsi->pc->buf);
Linus Torvalds's avatar
Linus Torvalds committed
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
		kfree(scsi->pc->rq);
		kfree(scsi->pc);
		scsi->pc = NULL;

		ret = SUCCESS;
	}

ide_unlock:
	spin_unlock_irq(&ide_lock);
no_drive:
	if (test_bit(IDESCSI_LOG_CMD, &scsi->log))
		printk (KERN_WARNING "ide-scsi: abort returns %s\n", ret == SUCCESS?"success":"failed");

	return ret;
}

static int idescsi_eh_reset (struct scsi_cmnd *cmd)
{
	struct request *req;
	idescsi_scsi_t *scsi  = scsihost_to_idescsi(cmd->device->host);
	ide_drive_t    *drive = scsi->drive;
	int             ready = 0;
	int             ret   = SUCCESS;

	/* In idescsi_eh_reset we forcefully remove the command from the ide subsystem and reset the device. */

	if (test_bit(IDESCSI_LOG_CMD, &scsi->log))
		printk (KERN_WARNING "ide-scsi: reset called for %lu\n", cmd->serial_number);

	if (!drive) {
		printk (KERN_WARNING "ide-scsi: Drive not set in idescsi_eh_reset\n");
		WARN_ON(1);
		return FAILED;
	}

883
884
	spin_lock_irq(cmd->device->host->host_lock);
	spin_lock(&ide_lock);
Linus Torvalds's avatar
Linus Torvalds committed
885
886
887
888

	if (!scsi->pc || (req = scsi->pc->rq) != HWGROUP(drive)->rq || !HWGROUP(drive)->handler) {
		printk (KERN_WARNING "ide-scsi: No active request in idescsi_eh_reset\n");
		spin_unlock(&ide_lock);
889
		spin_unlock_irq(cmd->device->host->host_lock);
Linus Torvalds's avatar
Linus Torvalds committed
890
891
892
893
		return FAILED;
	}

	/* kill current request */
894
895
	if (__blk_end_request(req, -EIO, 0))
		BUG();
896
	if (blk_sense_request(req))
897
		kfree(scsi->pc->buf);
Linus Torvalds's avatar
Linus Torvalds committed
898
899
900
901
902
903
	kfree(scsi->pc);
	scsi->pc = NULL;
	kfree(req);

	/* now nuke the drive queue */
	while ((req = elv_next_request(drive->queue))) {
904
905
		if (__blk_end_request(req, -EIO, 0))
			BUG();
Linus Torvalds's avatar
Linus Torvalds committed
906
907
908
909
910
	}

	HWGROUP(drive)->rq = NULL;
	HWGROUP(drive)->handler = NULL;
	HWGROUP(drive)->busy = 1;		/* will set this to zero when ide reset finished */
911
	spin_unlock(&ide_lock);
Linus Torvalds's avatar
Linus Torvalds committed
912
913
914
915
916
917
918

	ide_do_reset(drive);

	/* ide_do_reset starts a polling handler which restarts itself every 50ms until the reset finishes */

	do {
		spin_unlock_irq(cmd->device->host->host_lock);
919
		msleep(50);
Linus Torvalds's avatar
Linus Torvalds committed
920
921
922
923
924
925
926
927
928
929
		spin_lock_irq(cmd->device->host->host_lock);
	} while ( HWGROUP(drive)->handler );

	ready = drive_is_ready(drive);
	HWGROUP(drive)->busy--;
	if (!ready) {
		printk (KERN_ERR "ide-scsi: reset failed!\n");
		ret = FAILED;
	}

930
	spin_unlock_irq(cmd->device->host->host_lock);
Linus Torvalds's avatar
Linus Torvalds committed
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
	return ret;
}

static int idescsi_bios(struct scsi_device *sdev, struct block_device *bdev,
		sector_t capacity, int *parm)
{
	idescsi_scsi_t *idescsi = scsihost_to_idescsi(sdev->host);
	ide_drive_t *drive = idescsi->drive;

	if (drive->bios_cyl && drive->bios_head && drive->bios_sect) {
		parm[0] = drive->bios_head;
		parm[1] = drive->bios_sect;
		parm[2] = drive->bios_cyl;
	}
	return 0;
}

static struct scsi_host_template idescsi_template = {
	.module			= THIS_MODULE,
	.name			= "idescsi",
	.info			= idescsi_info,
	.slave_configure        = idescsi_slave_configure,
	.ioctl			= idescsi_ioctl,
	.queuecommand		= idescsi_queue,
	.eh_abort_handler	= idescsi_eh_abort,
	.eh_host_reset_handler  = idescsi_eh_reset,
	.bios_param		= idescsi_bios,
	.can_queue		= 40,
	.this_id		= -1,
	.sg_tablesize		= 256,
	.cmd_per_lun		= 5,
	.max_sectors		= 128,
	.use_clustering		= DISABLE_CLUSTERING,
	.emulated		= 1,
	.proc_name		= "ide-scsi",
};

968
static int ide_scsi_probe(ide_drive_t *drive)
Linus Torvalds's avatar
Linus Torvalds committed
969
970
971
972
973
974
975
976
977
978
979
980
{
	idescsi_scsi_t *idescsi;
	struct Scsi_Host *host;
	struct gendisk *g;
	static int warned;
	int err = -ENOMEM;

	if (!warned && drive->media == ide_cdrom) {
		printk(KERN_WARNING "ide-scsi is deprecated for cd burning! Use ide-cd and give dev=/dev/hdX as device\n");
		warned = 1;
	}

981
982
983
	if (idescsi_nocd && drive->media == ide_cdrom)
		return -ENODEV;

Linus Torvalds's avatar
Linus Torvalds committed
984
985
986
987
	if (!strstr("ide-scsi", drive->driver_req) ||
	    !drive->present ||
	    drive->media == ide_disk ||
	    !(host = scsi_host_alloc(&idescsi_template,sizeof(idescsi_scsi_t))))
988
		return -ENODEV;
Linus Torvalds's avatar
Linus Torvalds committed
989
990
991
992
993
994
995
996
997
998
999
1000

	g = alloc_disk(1 << PARTN_BITS);
	if (!g)
		goto out_host_put;

	ide_init_disk(g, drive);

	host->max_id = 1;

#if IDESCSI_DEBUG_LOG
	if (drive->id->last_lun)
		printk(KERN_NOTICE "%s: id->last_lun=%u\n", drive->name, drive->id->last_lun);
For faster browsing, not all history is shown. View entire blame