Blame view

block/scsi_ioctl.c 16.9 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
  /*
   * Copyright (C) 2001 Jens Axboe <axboe@suse.de>
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
   *
   * This program is distributed in the hope that it will be useful,
   * but WITHOUT ANY WARRANTY; without even the implied warranty of
   *
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   * GNU General Public License for more details.
   *
   * You should have received a copy of the GNU General Public Licens
   * along with this program; if not, write to the Free Software
   * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-
   *
   */
  #include <linux/kernel.h>
  #include <linux/errno.h>
  #include <linux/string.h>
  #include <linux/module.h>
  #include <linux/blkdev.h>
c59ede7b7   Randy.Dunlap   [PATCH] move capa...
24
  #include <linux/capability.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
  #include <linux/completion.h>
  #include <linux/cdrom.h>
  #include <linux/slab.h>
  #include <linux/times.h>
  #include <asm/uaccess.h>
  
  #include <scsi/scsi.h>
  #include <scsi/scsi_ioctl.h>
  #include <scsi/scsi_cmnd.h>
  
  /* Command group 3 is reserved and should never be used.  */
  const unsigned char scsi_command_size[8] =
  {
  	6, 10, 10, 12,
  	16, 12, 10, 10
  };
  
  EXPORT_SYMBOL(scsi_command_size);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
43
44
45
46
  #include <scsi/sg.h>
  
  static int sg_get_version(int __user *p)
  {
64100099e   Arjan van de Ven   [BLOCK] mark some...
47
  	static const int sg_version_num = 30527;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
48
49
  	return put_user(sg_version_num, p);
  }
165125e1e   Jens Axboe   [BLOCK] Get rid o...
50
  static int scsi_get_idlun(struct request_queue *q, int __user *p)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
51
52
53
  {
  	return put_user(0, p);
  }
165125e1e   Jens Axboe   [BLOCK] Get rid o...
54
  static int scsi_get_bus(struct request_queue *q, int __user *p)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
55
56
57
  {
  	return put_user(0, p);
  }
165125e1e   Jens Axboe   [BLOCK] Get rid o...
58
  static int sg_get_timeout(struct request_queue *q)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
59
60
61
  {
  	return q->sg_timeout / (HZ / USER_HZ);
  }
165125e1e   Jens Axboe   [BLOCK] Get rid o...
62
  static int sg_set_timeout(struct request_queue *q, int __user *p)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
63
64
65
66
67
68
69
70
  {
  	int timeout, err = get_user(timeout, p);
  
  	if (!err)
  		q->sg_timeout = timeout * (HZ / USER_HZ);
  
  	return err;
  }
165125e1e   Jens Axboe   [BLOCK] Get rid o...
71
  static int sg_get_reserved_size(struct request_queue *q, int __user *p)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
72
  {
44ec95425   Alan Stern   [SCSI] sg: cap re...
73
74
75
  	unsigned val = min(q->sg_reserved_size, q->max_sectors << 9);
  
  	return put_user(val, p);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
76
  }
165125e1e   Jens Axboe   [BLOCK] Get rid o...
77
  static int sg_set_reserved_size(struct request_queue *q, int __user *p)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
  {
  	int size, err = get_user(size, p);
  
  	if (err)
  		return err;
  
  	if (size < 0)
  		return -EINVAL;
  	if (size > (q->max_sectors << 9))
  		size = q->max_sectors << 9;
  
  	q->sg_reserved_size = size;
  	return 0;
  }
  
  /*
   * will always return that we are ATAPI even for a real SCSI drive, I'm not
   * so sure this is worth doing anything about (why would you care??)
   */
165125e1e   Jens Axboe   [BLOCK] Get rid o...
97
  static int sg_emulated_host(struct request_queue *q, int __user *p)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
98
99
100
101
102
103
104
105
106
  {
  	return put_user(1, p);
  }
  
  #define CMD_READ_SAFE	0x01
  #define CMD_WRITE_SAFE	0x02
  #define CMD_WARNED	0x04
  #define safe_for_read(cmd)	[cmd] = CMD_READ_SAFE
  #define safe_for_write(cmd)	[cmd] = CMD_WRITE_SAFE
337ad41de   FUJITA Tomonori   block: export blk...
107
  int blk_verify_command(unsigned char *cmd, int has_write_perm)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
108
109
110
111
112
113
114
115
116
117
118
  {
  	static unsigned char cmd_type[256] = {
  
  		/* Basic read-only commands */
  		safe_for_read(TEST_UNIT_READY),
  		safe_for_read(REQUEST_SENSE),
  		safe_for_read(READ_6),
  		safe_for_read(READ_10),
  		safe_for_read(READ_12),
  		safe_for_read(READ_16),
  		safe_for_read(READ_BUFFER),
942fc2fb7   Douglas Gilbert   [SCSI] permit REA...
119
  		safe_for_read(READ_DEFECT_DATA),
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
  		safe_for_read(READ_LONG),
  		safe_for_read(INQUIRY),
  		safe_for_read(MODE_SENSE),
  		safe_for_read(MODE_SENSE_10),
  		safe_for_read(LOG_SENSE),
  		safe_for_read(START_STOP),
  		safe_for_read(GPCMD_VERIFY_10),
  		safe_for_read(VERIFY_16),
  
  		/* Audio CD commands */
  		safe_for_read(GPCMD_PLAY_CD),
  		safe_for_read(GPCMD_PLAY_AUDIO_10),
  		safe_for_read(GPCMD_PLAY_AUDIO_MSF),
  		safe_for_read(GPCMD_PLAY_AUDIO_TI),
  		safe_for_read(GPCMD_PAUSE_RESUME),
  
  		/* CD/DVD data reading */
  		safe_for_read(GPCMD_READ_BUFFER_CAPACITY),
  		safe_for_read(GPCMD_READ_CD),
  		safe_for_read(GPCMD_READ_CD_MSF),
  		safe_for_read(GPCMD_READ_DISC_INFO),
  		safe_for_read(GPCMD_READ_CDVD_CAPACITY),
  		safe_for_read(GPCMD_READ_DVD_STRUCTURE),
  		safe_for_read(GPCMD_READ_HEADER),
  		safe_for_read(GPCMD_READ_TRACK_RZONE_INFO),
  		safe_for_read(GPCMD_READ_SUBCHANNEL),
  		safe_for_read(GPCMD_READ_TOC_PMA_ATIP),
  		safe_for_read(GPCMD_REPORT_KEY),
  		safe_for_read(GPCMD_SCAN),
  		safe_for_read(GPCMD_GET_CONFIGURATION),
  		safe_for_read(GPCMD_READ_FORMAT_CAPACITIES),
  		safe_for_read(GPCMD_GET_EVENT_STATUS_NOTIFICATION),
  		safe_for_read(GPCMD_GET_PERFORMANCE),
  		safe_for_read(GPCMD_SEEK),
  		safe_for_read(GPCMD_STOP_PLAY_SCAN),
  
  		/* Basic writing commands */
  		safe_for_write(WRITE_6),
  		safe_for_write(WRITE_10),
  		safe_for_write(WRITE_VERIFY),
  		safe_for_write(WRITE_12),
  		safe_for_write(WRITE_VERIFY_12),
  		safe_for_write(WRITE_16),
  		safe_for_write(WRITE_LONG),
0faf3d3d0   Thomas Maguin   [PATCH] scsi_ioct...
164
  		safe_for_write(WRITE_LONG_2),
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
  		safe_for_write(ERASE),
  		safe_for_write(GPCMD_MODE_SELECT_10),
  		safe_for_write(MODE_SELECT),
  		safe_for_write(LOG_SELECT),
  		safe_for_write(GPCMD_BLANK),
  		safe_for_write(GPCMD_CLOSE_TRACK),
  		safe_for_write(GPCMD_FLUSH_CACHE),
  		safe_for_write(GPCMD_FORMAT_UNIT),
  		safe_for_write(GPCMD_REPAIR_RZONE_TRACK),
  		safe_for_write(GPCMD_RESERVE_RZONE_TRACK),
  		safe_for_write(GPCMD_SEND_DVD_STRUCTURE),
  		safe_for_write(GPCMD_SEND_EVENT),
  		safe_for_write(GPCMD_SEND_KEY),
  		safe_for_write(GPCMD_SEND_OPC),
  		safe_for_write(GPCMD_SEND_CUE_SHEET),
  		safe_for_write(GPCMD_SET_SPEED),
  		safe_for_write(GPCMD_PREVENT_ALLOW_MEDIUM_REMOVAL),
  		safe_for_write(GPCMD_LOAD_UNLOAD),
  		safe_for_write(GPCMD_SET_STREAMING),
  	};
  	unsigned char type = cmd_type[cmd[0]];
  
  	/* Anybody who can open the device can do a read-safe command */
  	if (type & CMD_READ_SAFE)
  		return 0;
  
  	/* Write-safe commands just require a writable open.. */
5a57be8d1   Jens Axboe   [BLOCK] scsi_ioct...
192
193
  	if ((type & CMD_WRITE_SAFE) && has_write_perm)
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
194

3b0e77bd1   Jens Axboe   [PATCH] scsi_ioct...
195
196
197
  	/* And root can do any command.. */
  	if (capable(CAP_SYS_RAWIO))
  		return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
198
199
200
201
202
  	if (!type) {
  		cmd_type[cmd[0]] = CMD_WARNED;
  		printk(KERN_WARNING "scsi: unknown opcode 0x%02x
  ", cmd[0]);
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
203
204
205
  	/* Otherwise fail it with an "Operation not permitted" */
  	return -EPERM;
  }
337ad41de   FUJITA Tomonori   block: export blk...
206
  EXPORT_SYMBOL_GPL(blk_verify_command);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
207

165125e1e   Jens Axboe   [BLOCK] Get rid o...
208
  static int blk_fill_sghdr_rq(struct request_queue *q, struct request *rq,
41e1703b9   FUJITA Tomonori   [SCSI] bsg: unexp...
209
  			     struct sg_io_hdr *hdr, int has_write_perm)
3d6392cfb   Jens Axboe   bsg: support for ...
210
211
212
213
214
  {
  	memset(rq->cmd, 0, BLK_MAX_CDB); /* ATAPI hates garbage after CDB */
  
  	if (copy_from_user(rq->cmd, hdr->cmdp, hdr->cmd_len))
  		return -EFAULT;
337ad41de   FUJITA Tomonori   block: export blk...
215
  	if (blk_verify_command(rq->cmd, has_write_perm))
3d6392cfb   Jens Axboe   bsg: support for ...
216
217
218
219
220
221
222
  		return -EPERM;
  
  	/*
  	 * fill in request structure
  	 */
  	rq->cmd_len = hdr->cmd_len;
  	rq->cmd_type = REQ_TYPE_BLOCK_PC;
24bb8fb99   Tejun Heo   block: use jiffie...
223
  	rq->timeout = msecs_to_jiffies(hdr->timeout);
3d6392cfb   Jens Axboe   bsg: support for ...
224
225
226
227
228
229
230
  	if (!rq->timeout)
  		rq->timeout = q->sg_timeout;
  	if (!rq->timeout)
  		rq->timeout = BLK_DEFAULT_SG_TIMEOUT;
  
  	return 0;
  }
3d6392cfb   Jens Axboe   bsg: support for ...
231
232
233
234
235
  
  /*
   * unmap a request that was previously mapped to this sg_io_hdr. handles
   * both sg and non-sg sg_io_hdr.
   */
41e1703b9   FUJITA Tomonori   [SCSI] bsg: unexp...
236
  static int blk_unmap_sghdr_rq(struct request *rq, struct sg_io_hdr *hdr)
3d6392cfb   Jens Axboe   bsg: support for ...
237
  {
ac6b91b80   FUJITA Tomonori   block: changes fo...
238
  	blk_rq_unmap_user(rq->bio);
3d6392cfb   Jens Axboe   bsg: support for ...
239
240
241
  	blk_put_request(rq);
  	return 0;
  }
3d6392cfb   Jens Axboe   bsg: support for ...
242

41e1703b9   FUJITA Tomonori   [SCSI] bsg: unexp...
243
244
  static int blk_complete_sghdr_rq(struct request *rq, struct sg_io_hdr *hdr,
  				 struct bio *bio)
3d6392cfb   Jens Axboe   bsg: support for ...
245
246
247
248
249
250
251
252
253
254
255
256
257
258
  {
  	int r, ret = 0;
  
  	/*
  	 * fill in all the output members
  	 */
  	hdr->status = rq->errors & 0xff;
  	hdr->masked_status = status_byte(rq->errors);
  	hdr->msg_status = msg_byte(rq->errors);
  	hdr->host_status = host_byte(rq->errors);
  	hdr->driver_status = driver_byte(rq->errors);
  	hdr->info = 0;
  	if (hdr->masked_status || hdr->host_status || hdr->driver_status)
  		hdr->info |= SG_INFO_CHECK;
7a85f8896   FUJITA Tomonori   block: restore th...
259
  	hdr->resid = rq->data_len;
3d6392cfb   Jens Axboe   bsg: support for ...
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
  	hdr->sb_len_wr = 0;
  
  	if (rq->sense_len && hdr->sbp) {
  		int len = min((unsigned int) hdr->mx_sb_len, rq->sense_len);
  
  		if (!copy_to_user(hdr->sbp, rq->sense, len))
  			hdr->sb_len_wr = len;
  		else
  			ret = -EFAULT;
  	}
  
  	rq->bio = bio;
  	r = blk_unmap_sghdr_rq(rq, hdr);
  	if (ret)
  		r = ret;
  
  	return r;
  }
3d6392cfb   Jens Axboe   bsg: support for ...
278

165125e1e   Jens Axboe   [BLOCK] Get rid o...
279
  static int sg_io(struct file *file, struct request_queue *q,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
280
281
  		struct gendisk *bd_disk, struct sg_io_hdr *hdr)
  {
3d6392cfb   Jens Axboe   bsg: support for ...
282
283
  	unsigned long start_time;
  	int writing = 0, ret = 0, has_write_perm = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
284
  	struct request *rq;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
285
  	char sense[SCSI_SENSE_BUFFERSIZE];
77d172ce2   FUJITA Tomonori   [PATCH] fix SG_IO...
286
  	struct bio *bio;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
287
288
289
290
291
  
  	if (hdr->interface_id != 'S')
  		return -EINVAL;
  	if (hdr->cmd_len > BLK_MAX_CDB)
  		return -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
292

defd94b75   Mike Christie   [SCSI] seperate m...
293
  	if (hdr->dxfer_len > (q->max_hw_sectors << 9))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
294
  		return -EIO;
f1970baf6   James Bottomley   [PATCH] Add scatt...
295
  	if (hdr->dxfer_len)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
296
297
298
  		switch (hdr->dxfer_direction) {
  		default:
  			return -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
299
300
301
  		case SG_DXFER_TO_DEV:
  			writing = 1;
  			break;
616e8a091   Jens Axboe   [PATCH] Fix bad d...
302
  		case SG_DXFER_TO_FROM_DEV:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
303
  		case SG_DXFER_FROM_DEV:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
304
305
  			break;
  		}
dd1cab95f   Jens Axboe   [PATCH] Cleanup b...
306
307
308
  	rq = blk_get_request(q, writing ? WRITE : READ, GFP_KERNEL);
  	if (!rq)
  		return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
309

3d6392cfb   Jens Axboe   bsg: support for ...
310
311
  	if (file)
  		has_write_perm = file->f_mode & FMODE_WRITE;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
312

3d6392cfb   Jens Axboe   bsg: support for ...
313
  	if (blk_fill_sghdr_rq(q, rq, hdr, has_write_perm)) {
3d6392cfb   Jens Axboe   bsg: support for ...
314
315
316
  		blk_put_request(rq);
  		return -EFAULT;
  	}
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
317

0e75f9063   Mike Christie   [PATCH] block: su...
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
  	if (hdr->iovec_count) {
  		const int size = sizeof(struct sg_iovec) * hdr->iovec_count;
  		struct sg_iovec *iov;
  
  		iov = kmalloc(size, GFP_KERNEL);
  		if (!iov) {
  			ret = -ENOMEM;
  			goto out;
  		}
  
  		if (copy_from_user(iov, hdr->dxferp, size)) {
  			kfree(iov);
  			ret = -EFAULT;
  			goto out;
  		}
  
  		ret = blk_rq_map_user_iov(q, rq, iov, hdr->iovec_count,
  					  hdr->dxfer_len);
  		kfree(iov);
  	} else if (hdr->dxfer_len)
  		ret = blk_rq_map_user(q, rq, hdr->dxferp, hdr->dxfer_len);
  
  	if (ret)
  		goto out;
77d172ce2   FUJITA Tomonori   [PATCH] fix SG_IO...
342
  	bio = rq->bio;
3d6392cfb   Jens Axboe   bsg: support for ...
343
344
345
  	memset(sense, 0, sizeof(sense));
  	rq->sense = sense;
  	rq->sense_len = 0;
01840f9c9   Jens Axboe   [PATCH] blk: Fix ...
346
  	rq->retries = 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
347
348
349
350
351
352
  	start_time = jiffies;
  
  	/* ignore return value. All information is passed back to caller
  	 * (if he doesn't check that is his problem).
  	 * N.B. a non-zero SCSI status is _not_ necessarily an error.
  	 */
994ca9a19   James Bottomley   [PATCH] update bl...
353
  	blk_execute_rq(q, bd_disk, rq, 0);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
354

24bb8fb99   Tejun Heo   block: use jiffie...
355
  	hdr->duration = jiffies_to_msecs(jiffies - start_time);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
356

3d6392cfb   Jens Axboe   bsg: support for ...
357
  	return blk_complete_sghdr_rq(rq, hdr, bio);
dd1cab95f   Jens Axboe   [PATCH] Cleanup b...
358
359
360
  out:
  	blk_put_request(rq);
  	return ret;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
361
  }
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
362
363
364
365
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
  /**
   * sg_scsi_ioctl  --  handle deprecated SCSI_IOCTL_SEND_COMMAND ioctl
   * @file:	file this ioctl operates on (optional)
   * @q:		request queue to send scsi commands down
   * @disk:	gendisk to operate on (option)
   * @sic:	userspace structure describing the command to perform
   *
   * Send down the scsi command described by @sic to the device below
   * the request queue @q.  If @file is non-NULL it's used to perform
   * fine-grained permission checks that allow users to send down
   * non-destructive SCSI commands.  If the caller has a struct gendisk
   * available it should be passed in as @disk to allow the low level
   * driver to use the information contained in it.  A non-NULL @disk
   * is only allowed if the caller knows that the low level driver doesn't
   * need it (e.g. in the scsi subsystem).
   *
   * Notes:
   *   -  This interface is deprecated - users should use the SG_IO
   *      interface instead, as this is a more flexible approach to
   *      performing SCSI commands on a device.
   *   -  The SCSI command length is determined by examining the 1st byte
   *      of the given command. There is no way to override this.
   *   -  Data transfers are limited to PAGE_SIZE
   *   -  The length (x + y) must be at least OMAX_SB_LEN bytes long to
   *      accommodate the sense buffer when an error occurs.
   *      The sense buffer is truncated to OMAX_SB_LEN (16) bytes so that
   *      old code will not be surprised.
   *   -  If a Unix error occurs (e.g. ENOMEM) then the user will receive
   *      a negative return and the Unix error code in 'errno'.
   *      If the SCSI command succeeds then 0 is returned.
   *      Positive numbers returned are the compacted SCSI error codes (4
   *      bytes in one int) where the lowest byte is the SCSI status.
   */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
395
  #define OMAX_SB_LEN 16          /* For backward compatibility */
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
396
397
  int sg_scsi_ioctl(struct file *file, struct request_queue *q,
  		  struct gendisk *disk, struct scsi_ioctl_command __user *sic)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
398
399
400
401
402
  {
  	struct request *rq;
  	int err;
  	unsigned int in_len, out_len, bytes, opcode, cmdlen;
  	char *buffer = NULL, sense[SCSI_SENSE_BUFFERSIZE];
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
403
404
  	if (!sic)
  		return -EINVAL;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
405
406
407
408
409
410
411
412
413
414
415
416
417
418
  	/*
  	 * get in an out lengths, verify they don't exceed a page worth of data
  	 */
  	if (get_user(in_len, &sic->inlen))
  		return -EFAULT;
  	if (get_user(out_len, &sic->outlen))
  		return -EFAULT;
  	if (in_len > PAGE_SIZE || out_len > PAGE_SIZE)
  		return -EINVAL;
  	if (get_user(opcode, sic->data))
  		return -EFAULT;
  
  	bytes = max(in_len, out_len);
  	if (bytes) {
dd00cc486   Yoann Padioleau   some kmalloc/mems...
419
  		buffer = kzalloc(bytes, q->bounce_gfp | GFP_USER| __GFP_NOWARN);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
420
421
  		if (!buffer)
  			return -ENOMEM;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
422
423
424
425
426
427
428
429
430
431
432
433
434
  	}
  
  	rq = blk_get_request(q, in_len ? WRITE : READ, __GFP_WAIT);
  
  	cmdlen = COMMAND_SIZE(opcode);
  
  	/*
  	 * get command and data to send to device, if any
  	 */
  	err = -EFAULT;
  	rq->cmd_len = cmdlen;
  	if (copy_from_user(rq->cmd, sic->data, cmdlen))
  		goto error;
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
435
  	if (in_len && copy_from_user(buffer, sic->data + cmdlen, in_len))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
436
  		goto error;
337ad41de   FUJITA Tomonori   block: export blk...
437
  	err = blk_verify_command(rq->cmd, file->f_mode & FMODE_WRITE);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
438
439
  	if (err)
  		goto error;
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
440
441
  	/* default.  possible overriden later */
  	rq->retries = 5;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
442
  	switch (opcode) {
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
  	case SEND_DIAGNOSTIC:
  	case FORMAT_UNIT:
  		rq->timeout = FORMAT_UNIT_TIMEOUT;
  		rq->retries = 1;
  		break;
  	case START_STOP:
  		rq->timeout = START_STOP_TIMEOUT;
  		break;
  	case MOVE_MEDIUM:
  		rq->timeout = MOVE_MEDIUM_TIMEOUT;
  		break;
  	case READ_ELEMENT_STATUS:
  		rq->timeout = READ_ELEMENT_STATUS_TIMEOUT;
  		break;
  	case READ_DEFECT_DATA:
  		rq->timeout = READ_DEFECT_DATA_TIMEOUT;
  		rq->retries = 1;
  		break;
  	default:
3d6392cfb   Jens Axboe   bsg: support for ...
462
  		rq->timeout = BLK_DEFAULT_SG_TIMEOUT;
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
463
464
465
466
467
468
  		break;
  	}
  
  	if (bytes && blk_rq_map_kern(q, rq, buffer, bytes, __GFP_WAIT)) {
  		err = DRIVER_ERROR << 24;
  		goto out;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
469
470
471
472
473
  	}
  
  	memset(sense, 0, sizeof(sense));
  	rq->sense = sense;
  	rq->sense_len = 0;
4aff5e233   Jens Axboe   [PATCH] Split str...
474
  	rq->cmd_type = REQ_TYPE_BLOCK_PC;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
475

21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
476
477
478
  	blk_execute_rq(q, disk, rq, 0);
  
  out:
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
  	err = rq->errors & 0xff;	/* only 8 bit SCSI status */
  	if (err) {
  		if (rq->sense_len && rq->sense) {
  			bytes = (OMAX_SB_LEN > rq->sense_len) ?
  				rq->sense_len : OMAX_SB_LEN;
  			if (copy_to_user(sic->data, rq->sense, bytes))
  				err = -EFAULT;
  		}
  	} else {
  		if (copy_to_user(sic->data, buffer, out_len))
  			err = -EFAULT;
  	}
  	
  error:
  	kfree(buffer);
  	blk_put_request(rq);
  	return err;
  }
21b2f0c80   Christoph Hellwig   [SCSI] unify SCSI...
497
  EXPORT_SYMBOL_GPL(sg_scsi_ioctl);
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
498
499
  
  /* Send basic block requests */
165125e1e   Jens Axboe   [BLOCK] Get rid o...
500
501
  static int __blk_send_generic(struct request_queue *q, struct gendisk *bd_disk,
  			      int cmd, int data)
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
502
503
504
505
506
  {
  	struct request *rq;
  	int err;
  
  	rq = blk_get_request(q, WRITE, __GFP_WAIT);
4aff5e233   Jens Axboe   [PATCH] Split str...
507
  	rq->cmd_type = REQ_TYPE_BLOCK_PC;
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
508
509
  	rq->data = NULL;
  	rq->data_len = 0;
7a85f8896   FUJITA Tomonori   block: restore th...
510
  	rq->extra_len = 0;
3d6392cfb   Jens Axboe   bsg: support for ...
511
  	rq->timeout = BLK_DEFAULT_SG_TIMEOUT;
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
512
513
514
515
516
517
518
519
520
  	memset(rq->cmd, 0, sizeof(rq->cmd));
  	rq->cmd[0] = cmd;
  	rq->cmd[4] = data;
  	rq->cmd_len = 6;
  	err = blk_execute_rq(q, bd_disk, rq, 0);
  	blk_put_request(rq);
  
  	return err;
  }
165125e1e   Jens Axboe   [BLOCK] Get rid o...
521
522
  static inline int blk_send_start_stop(struct request_queue *q,
  				      struct gendisk *bd_disk, int data)
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
523
524
525
  {
  	return __blk_send_generic(q, bd_disk, GPCMD_START_STOP_UNIT, data);
  }
45e79a3ac   FUJITA Tomonori   bsg: add a reques...
526
527
  int scsi_cmd_ioctl(struct file *file, struct request_queue *q,
  		   struct gendisk *bd_disk, unsigned int cmd, void __user *arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
528
  {
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
529
  	int err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
530

45e79a3ac   FUJITA Tomonori   bsg: add a reques...
531
  	if (!q || blk_get_queue(q))
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
  		return -ENXIO;
  
  	switch (cmd) {
  		/*
  		 * new sgv3 interface
  		 */
  		case SG_GET_VERSION_NUM:
  			err = sg_get_version(arg);
  			break;
  		case SCSI_IOCTL_GET_IDLUN:
  			err = scsi_get_idlun(q, arg);
  			break;
  		case SCSI_IOCTL_GET_BUS_NUMBER:
  			err = scsi_get_bus(q, arg);
  			break;
  		case SG_SET_TIMEOUT:
  			err = sg_set_timeout(q, arg);
  			break;
  		case SG_GET_TIMEOUT:
  			err = sg_get_timeout(q);
  			break;
  		case SG_GET_RESERVED_SIZE:
  			err = sg_get_reserved_size(q, arg);
  			break;
  		case SG_SET_RESERVED_SIZE:
  			err = sg_set_reserved_size(q, arg);
  			break;
  		case SG_EMULATED_HOST:
  			err = sg_emulated_host(q, arg);
  			break;
  		case SG_IO: {
  			struct sg_io_hdr hdr;
  
  			err = -EFAULT;
  			if (copy_from_user(&hdr, arg, sizeof(hdr)))
  				break;
  			err = sg_io(file, q, bd_disk, &hdr);
  			if (err == -EFAULT)
  				break;
  
  			if (copy_to_user(arg, &hdr, sizeof(hdr)))
  				err = -EFAULT;
  			break;
  		}
  		case CDROM_SEND_PACKET: {
  			struct cdrom_generic_command cgc;
  			struct sg_io_hdr hdr;
  
  			err = -EFAULT;
  			if (copy_from_user(&cgc, arg, sizeof(cgc)))
  				break;
  			cgc.timeout = clock_t_to_jiffies(cgc.timeout);
  			memset(&hdr, 0, sizeof(hdr));
  			hdr.interface_id = 'S';
  			hdr.cmd_len = sizeof(cgc.cmd);
  			hdr.dxfer_len = cgc.buflen;
  			err = 0;
  			switch (cgc.data_direction) {
  				case CGC_DATA_UNKNOWN:
  					hdr.dxfer_direction = SG_DXFER_UNKNOWN;
  					break;
  				case CGC_DATA_WRITE:
  					hdr.dxfer_direction = SG_DXFER_TO_DEV;
  					break;
  				case CGC_DATA_READ:
  					hdr.dxfer_direction = SG_DXFER_FROM_DEV;
  					break;
  				case CGC_DATA_NONE:
  					hdr.dxfer_direction = SG_DXFER_NONE;
  					break;
  				default:
  					err = -EINVAL;
  			}
  			if (err)
  				break;
  
  			hdr.dxferp = cgc.buffer;
  			hdr.sbp = cgc.sense;
  			if (hdr.sbp)
  				hdr.mx_sb_len = sizeof(struct request_sense);
  			hdr.timeout = cgc.timeout;
  			hdr.cmdp = ((struct cdrom_generic_command __user*) arg)->cmd;
  			hdr.cmd_len = sizeof(cgc.cmd);
  
  			err = sg_io(file, q, bd_disk, &hdr);
  			if (err == -EFAULT)
  				break;
  
  			if (hdr.status)
  				err = -EIO;
  
  			cgc.stat = err;
  			cgc.buflen = hdr.resid;
  			if (copy_to_user(arg, &cgc, sizeof(cgc)))
  				err = -EFAULT;
  
  			break;
  		}
  
  		/*
  		 * old junk scsi send command ioctl
  		 */
  		case SCSI_IOCTL_SEND_COMMAND:
  			printk(KERN_WARNING "program %s is using a deprecated SCSI ioctl, please convert it to SG_IO
  ", current->comm);
  			err = -EINVAL;
  			if (!arg)
  				break;
  
  			err = sg_scsi_ioctl(file, q, bd_disk, arg);
  			break;
  		case CDROMCLOSETRAY:
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
644
645
  			err = blk_send_start_stop(q, bd_disk, 0x03);
  			break;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
646
  		case CDROMEJECT:
f98d2dfd0   Ben Collins   [PATCH] block: Cl...
647
  			err = blk_send_start_stop(q, bd_disk, 0x02);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
648
649
650
651
652
653
654
655
656
657
  			break;
  		default:
  			err = -ENOTTY;
  	}
  
  	blk_put_queue(q);
  	return err;
  }
  
  EXPORT_SYMBOL(scsi_cmd_ioctl);