aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/scsi/scsi_ioctl.c
diff options
context:
space:
mode:
authorDouglas Gilbert <dgilbert@interlog.com>2014-10-18 16:11:21 -0400
committerChristoph Hellwig <hch@lst.de>2014-11-12 05:15:54 -0500
commit26cf591e6dfc0d07495b7bcf20a557b316811f00 (patch)
treeaaebbfe8c9764f981f2fa540b6ae20c79dce55af /drivers/scsi/scsi_ioctl.c
parent678e27573237a0b065defdf99e5070c9b0c403c3 (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.c20
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 */
291int scsi_nonblockable_ioctl(struct scsi_device *sdev, int cmd, 292int 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;