diff options
author | Douglas Gilbert <dgilbert@interlog.com> | 2014-10-18 16:11:21 -0400 |
---|---|---|
committer | Christoph Hellwig <hch@lst.de> | 2014-11-12 05:15:54 -0500 |
commit | 26cf591e6dfc0d07495b7bcf20a557b316811f00 (patch) | |
tree | aaebbfe8c9764f981f2fa540b6ae20c79dce55af /drivers/scsi/scsi_ioctl.c | |
parent | 678e27573237a0b065defdf99e5070c9b0c403c3 (diff) |
scsi: add SG_SCSI_RESET_NO_ESCALATE flag to SG_SCSI_RESET ioctl
Further to a January 2013 thread titled: "[PATCH] SG_SCSI_RESET ioctl
should only perform requested operation" by Jeremy Linton a patch (v3)
is presented that expands the existing ioctl to include "no_escalate"
versions to the existing resets. This requires no changes to SCSI low
level drivers (LLDs); it adds several more finely tuned reset options
to the user space. For example:
/* This call remains the same, with the same escalating semantics
* if the device (LU) reset fail. That is: on failure to try a
* target reset and if that fails, try a bus reset, and if that fails
* try a host (i.e. LLD) reset. */
val = SG_SCSI_RESET_DEVICE;
res = ioctl(<sg_or_block_fd>, SG_SCSI_RESET, &val);
/* What follows is a new option introduced by this patch series. Only
* a device reset is attempted. If that fails then an appropriate
* error code is provided. N.B. There is no reset escalation. */
val = SG_SCSI_RESET_DEVICE | SG_SCSI_RESET_NO_ESCALATE;
res = ioctl(<sg_or_block_fd>, SG_SCSI_RESET, &val);
Signed-off-by: Douglas Gilbert <dgilbert@interlog.com>
Reviewed-by: Jeremy Linton <jlinton@tributary.com>
Reviewed-by: Hannes Reinecke <hare@suse.de>
Signed-off-by: Christoph Hellwig <hch@lst.de>
Diffstat (limited to 'drivers/scsi/scsi_ioctl.c')
-rw-r--r-- | drivers/scsi/scsi_ioctl.c | 20 |
1 files changed, 13 insertions, 7 deletions
diff --git a/drivers/scsi/scsi_ioctl.c b/drivers/scsi/scsi_ioctl.c index 1aaaf43c6803..12fe676d1343 100644 --- a/drivers/scsi/scsi_ioctl.c +++ b/drivers/scsi/scsi_ioctl.c | |||
@@ -285,13 +285,14 @@ EXPORT_SYMBOL(scsi_ioctl); | |||
285 | * scsi_nonblockable_ioctl() - Handle SG_SCSI_RESET | 285 | * scsi_nonblockable_ioctl() - Handle SG_SCSI_RESET |
286 | * @sdev: scsi device receiving ioctl | 286 | * @sdev: scsi device receiving ioctl |
287 | * @cmd: Must be SC_SCSI_RESET | 287 | * @cmd: Must be SC_SCSI_RESET |
288 | * @arg: pointer to int containing SG_SCSI_RESET_{DEVICE,BUS,HOST} | 288 | * @arg: pointer to int containing SG_SCSI_RESET_{DEVICE,TARGET,BUS,HOST} |
289 | * possibly OR-ed with SG_SCSI_RESET_NO_ESCALATE | ||
289 | * @ndelay: file mode O_NDELAY flag | 290 | * @ndelay: file mode O_NDELAY flag |
290 | */ | 291 | */ |
291 | int scsi_nonblockable_ioctl(struct scsi_device *sdev, int cmd, | 292 | int scsi_nonblockable_ioctl(struct scsi_device *sdev, int cmd, |
292 | void __user *arg, int ndelay) | 293 | void __user *arg, int ndelay) |
293 | { | 294 | { |
294 | int val, result; | 295 | int val, val2, result; |
295 | 296 | ||
296 | /* The first set of iocts may be executed even if we're doing | 297 | /* The first set of iocts may be executed even if we're doing |
297 | * error processing, as long as the device was opened | 298 | * error processing, as long as the device was opened |
@@ -307,27 +308,32 @@ int scsi_nonblockable_ioctl(struct scsi_device *sdev, int cmd, | |||
307 | result = get_user(val, (int __user *)arg); | 308 | result = get_user(val, (int __user *)arg); |
308 | if (result) | 309 | if (result) |
309 | return result; | 310 | return result; |
311 | if (val & SG_SCSI_RESET_NO_ESCALATE) { | ||
312 | val &= ~SG_SCSI_RESET_NO_ESCALATE; | ||
313 | val2 = SCSI_TRY_RESET_NO_ESCALATE; | ||
314 | } else | ||
315 | val2 = 0; | ||
310 | if (val == SG_SCSI_RESET_NOTHING) | 316 | if (val == SG_SCSI_RESET_NOTHING) |
311 | return 0; | 317 | return 0; |
312 | switch (val) { | 318 | switch (val) { |
313 | case SG_SCSI_RESET_DEVICE: | 319 | case SG_SCSI_RESET_DEVICE: |
314 | val = SCSI_TRY_RESET_DEVICE; | 320 | val2 |= SCSI_TRY_RESET_DEVICE; |
315 | break; | 321 | break; |
316 | case SG_SCSI_RESET_TARGET: | 322 | case SG_SCSI_RESET_TARGET: |
317 | val = SCSI_TRY_RESET_TARGET; | 323 | val2 |= SCSI_TRY_RESET_TARGET; |
318 | break; | 324 | break; |
319 | case SG_SCSI_RESET_BUS: | 325 | case SG_SCSI_RESET_BUS: |
320 | val = SCSI_TRY_RESET_BUS; | 326 | val2 |= SCSI_TRY_RESET_BUS; |
321 | break; | 327 | break; |
322 | case SG_SCSI_RESET_HOST: | 328 | case SG_SCSI_RESET_HOST: |
323 | val = SCSI_TRY_RESET_HOST; | 329 | val2 |= SCSI_TRY_RESET_HOST; |
324 | break; | 330 | break; |
325 | default: | 331 | default: |
326 | return -EINVAL; | 332 | return -EINVAL; |
327 | } | 333 | } |
328 | if (!capable(CAP_SYS_ADMIN) || !capable(CAP_SYS_RAWIO)) | 334 | if (!capable(CAP_SYS_ADMIN) || !capable(CAP_SYS_RAWIO)) |
329 | return -EACCES; | 335 | return -EACCES; |
330 | return (scsi_reset_provider(sdev, val) == | 336 | return (scsi_reset_provider(sdev, val2) == |
331 | SUCCESS) ? 0 : -EIO; | 337 | SUCCESS) ? 0 : -EIO; |
332 | } | 338 | } |
333 | return -ENODEV; | 339 | return -ENODEV; |