Commit 47005f25 authored by Tejun Heo's avatar Tejun Heo Committed by Jeff Garzik

[PATCH] libata: implement per-dev EH action mask eh_info->dev_action[]

Currently, the only per-dev EH action is REVALIDATE.  EH used to
exploit ehi->dev to do selective revalidation on a ATA bus.  However,
this is a bit hacky and makes it impossible to request selective
revalidation from outside of EH or add another per-dev EH action.

This patch adds per-dev EH action mask eh_info->dev_action[] and
update EH to use this field for REVALIDATE.  Note that per-dev actions
can still be specified at port-level and it has the same effect of
specifying the action for all devices on the port.
Signed-off-by: default avatarTejun Heo <htejun@gmail.com>
Signed-off-by: default avatarJeff Garzik <jeff@garzik.org>
parent 3a778275
...@@ -706,9 +706,35 @@ static void ata_eh_detach_dev(struct ata_device *dev) ...@@ -706,9 +706,35 @@ static void ata_eh_detach_dev(struct ata_device *dev)
spin_unlock_irqrestore(&ap->host_set->lock, flags); spin_unlock_irqrestore(&ap->host_set->lock, flags);
} }
static void ata_eh_clear_action(struct ata_device *dev,
struct ata_eh_info *ehi, unsigned int action)
{
int i;
if (!dev) {
ehi->action &= ~action;
for (i = 0; i < ATA_MAX_DEVICES; i++)
ehi->dev_action[i] &= ~action;
} else {
/* doesn't make sense for port-wide EH actions */
WARN_ON(!(action & ATA_EH_PERDEV_MASK));
/* break ehi->action into ehi->dev_action */
if (ehi->action & action) {
for (i = 0; i < ATA_MAX_DEVICES; i++)
ehi->dev_action[i] |= ehi->action & action;
ehi->action &= ~action;
}
/* turn off the specified per-dev action */
ehi->dev_action[dev->devno] &= ~action;
}
}
/** /**
* ata_eh_about_to_do - about to perform eh_action * ata_eh_about_to_do - about to perform eh_action
* @ap: target ATA port * @ap: target ATA port
* @dev: target ATA dev for per-dev action (can be NULL)
* @action: action about to be performed * @action: action about to be performed
* *
* Called just before performing EH actions to clear related bits * Called just before performing EH actions to clear related bits
...@@ -718,16 +744,35 @@ static void ata_eh_detach_dev(struct ata_device *dev) ...@@ -718,16 +744,35 @@ static void ata_eh_detach_dev(struct ata_device *dev)
* LOCKING: * LOCKING:
* None. * None.
*/ */
static void ata_eh_about_to_do(struct ata_port *ap, unsigned int action) static void ata_eh_about_to_do(struct ata_port *ap, struct ata_device *dev,
unsigned int action)
{ {
unsigned long flags; unsigned long flags;
spin_lock_irqsave(&ap->host_set->lock, flags); spin_lock_irqsave(&ap->host_set->lock, flags);
ap->eh_info.action &= ~action; ata_eh_clear_action(dev, &ap->eh_info, action);
ap->flags |= ATA_FLAG_RECOVERED; ap->flags |= ATA_FLAG_RECOVERED;
spin_unlock_irqrestore(&ap->host_set->lock, flags); spin_unlock_irqrestore(&ap->host_set->lock, flags);
} }
/**
* ata_eh_done - EH action complete
* @ap: target ATA port
* @dev: target ATA dev for per-dev action (can be NULL)
* @action: action just completed
*
* Called right after performing EH actions to clear related bits
* in @ap->eh_context.
*
* LOCKING:
* None.
*/
static void ata_eh_done(struct ata_port *ap, struct ata_device *dev,
unsigned int action)
{
ata_eh_clear_action(dev, &ap->eh_context.i, action);
}
/** /**
* ata_err_string - convert err_mask to descriptive string * ata_err_string - convert err_mask to descriptive string
* @err_mask: error mask to convert to string * @err_mask: error mask to convert to string
...@@ -1271,10 +1316,6 @@ static void ata_eh_autopsy(struct ata_port *ap) ...@@ -1271,10 +1316,6 @@ static void ata_eh_autopsy(struct ata_port *ap)
is_io = 1; is_io = 1;
} }
/* speed down iff command was in progress */
if (failed_dev)
action |= ata_eh_speed_down(failed_dev, is_io, all_err_mask);
/* enforce default EH actions */ /* enforce default EH actions */
if (ap->flags & ATA_FLAG_FROZEN || if (ap->flags & ATA_FLAG_FROZEN ||
all_err_mask & (AC_ERR_HSM | AC_ERR_TIMEOUT)) all_err_mask & (AC_ERR_HSM | AC_ERR_TIMEOUT))
...@@ -1282,6 +1323,17 @@ static void ata_eh_autopsy(struct ata_port *ap) ...@@ -1282,6 +1323,17 @@ static void ata_eh_autopsy(struct ata_port *ap)
else if (all_err_mask) else if (all_err_mask)
action |= ATA_EH_REVALIDATE; action |= ATA_EH_REVALIDATE;
/* if we have offending qcs and the associated failed device */
if (failed_dev) {
/* speed down */
action |= ata_eh_speed_down(failed_dev, is_io, all_err_mask);
/* perform per-dev EH action only on the offending device */
ehc->i.dev_action[failed_dev->devno] |=
action & ATA_EH_PERDEV_MASK;
action &= ~ATA_EH_PERDEV_MASK;
}
/* record autopsy result */ /* record autopsy result */
ehc->i.dev = failed_dev; ehc->i.dev = failed_dev;
ehc->i.action = action; ehc->i.action = action;
...@@ -1457,7 +1509,7 @@ static int ata_eh_reset(struct ata_port *ap, int classify, ...@@ -1457,7 +1509,7 @@ static int ata_eh_reset(struct ata_port *ap, int classify,
reset == softreset ? "soft" : "hard"); reset == softreset ? "soft" : "hard");
/* reset */ /* reset */
ata_eh_about_to_do(ap, ATA_EH_RESET_MASK); ata_eh_about_to_do(ap, NULL, ATA_EH_RESET_MASK);
ehc->i.flags |= ATA_EHI_DID_RESET; ehc->i.flags |= ATA_EHI_DID_RESET;
rc = ata_do_reset(ap, reset, classes); rc = ata_do_reset(ap, reset, classes);
...@@ -1476,7 +1528,7 @@ static int ata_eh_reset(struct ata_port *ap, int classify, ...@@ -1476,7 +1528,7 @@ static int ata_eh_reset(struct ata_port *ap, int classify,
return -EINVAL; return -EINVAL;
} }
ata_eh_about_to_do(ap, ATA_EH_RESET_MASK); ata_eh_about_to_do(ap, NULL, ATA_EH_RESET_MASK);
rc = ata_do_reset(ap, reset, classes); rc = ata_do_reset(ap, reset, classes);
if (rc == 0 && classify && if (rc == 0 && classify &&
...@@ -1520,8 +1572,7 @@ static int ata_eh_reset(struct ata_port *ap, int classify, ...@@ -1520,8 +1572,7 @@ static int ata_eh_reset(struct ata_port *ap, int classify,
postreset(ap, classes); postreset(ap, classes);
/* reset successful, schedule revalidation */ /* reset successful, schedule revalidation */
ehc->i.dev = NULL; ata_eh_done(ap, NULL, ATA_EH_RESET_MASK);
ehc->i.action &= ~ATA_EH_RESET_MASK;
ehc->i.action |= ATA_EH_REVALIDATE; ehc->i.action |= ATA_EH_REVALIDATE;
} }
...@@ -1539,21 +1590,25 @@ static int ata_eh_revalidate_and_attach(struct ata_port *ap, ...@@ -1539,21 +1590,25 @@ static int ata_eh_revalidate_and_attach(struct ata_port *ap,
DPRINTK("ENTER\n"); DPRINTK("ENTER\n");
for (i = 0; i < ATA_MAX_DEVICES; i++) { for (i = 0; i < ATA_MAX_DEVICES; i++) {
unsigned int action;
dev = &ap->device[i]; dev = &ap->device[i];
action = ehc->i.action | ehc->i.dev_action[dev->devno];
if (ehc->i.action & ATA_EH_REVALIDATE && ata_dev_enabled(dev) && if (action & ATA_EH_REVALIDATE && ata_dev_enabled(dev)) {
(!ehc->i.dev || ehc->i.dev == dev)) {
if (ata_port_offline(ap)) { if (ata_port_offline(ap)) {
rc = -EIO; rc = -EIO;
break; break;
} }
ata_eh_about_to_do(ap, ATA_EH_REVALIDATE); ata_eh_about_to_do(ap, dev, ATA_EH_REVALIDATE);
rc = ata_dev_revalidate(dev, rc = ata_dev_revalidate(dev,
ehc->i.flags & ATA_EHI_DID_RESET); ehc->i.flags & ATA_EHI_DID_RESET);
if (rc) if (rc)
break; break;
ata_eh_done(ap, dev, ATA_EH_REVALIDATE);
/* schedule the scsi_rescan_device() here */ /* schedule the scsi_rescan_device() here */
queue_work(ata_aux_wq, &(ap->scsi_rescan_task)); queue_work(ata_aux_wq, &(ap->scsi_rescan_task));
} else if (dev->class == ATA_DEV_UNKNOWN && } else if (dev->class == ATA_DEV_UNKNOWN &&
...@@ -1576,9 +1631,7 @@ static int ata_eh_revalidate_and_attach(struct ata_port *ap, ...@@ -1576,9 +1631,7 @@ static int ata_eh_revalidate_and_attach(struct ata_port *ap,
} }
} }
if (rc == 0) if (rc)
ehc->i.action &= ~ATA_EH_REVALIDATE;
else
*r_failed_dev = dev; *r_failed_dev = dev;
DPRINTK("EXIT\n"); DPRINTK("EXIT\n");
......
...@@ -249,6 +249,7 @@ enum { ...@@ -249,6 +249,7 @@ enum {
ATA_EH_HARDRESET = (1 << 2), ATA_EH_HARDRESET = (1 << 2),
ATA_EH_RESET_MASK = ATA_EH_SOFTRESET | ATA_EH_HARDRESET, ATA_EH_RESET_MASK = ATA_EH_SOFTRESET | ATA_EH_HARDRESET,
ATA_EH_PERDEV_MASK = ATA_EH_REVALIDATE,
/* ata_eh_info->flags */ /* ata_eh_info->flags */
ATA_EHI_HOTPLUGGED = (1 << 0), /* could have been hotplugged */ ATA_EHI_HOTPLUGGED = (1 << 0), /* could have been hotplugged */
...@@ -462,6 +463,7 @@ struct ata_eh_info { ...@@ -462,6 +463,7 @@ struct ata_eh_info {
u32 serror; /* SError from LLDD */ u32 serror; /* SError from LLDD */
unsigned int err_mask; /* port-wide err_mask */ unsigned int err_mask; /* port-wide err_mask */
unsigned int action; /* ATA_EH_* action mask */ unsigned int action; /* ATA_EH_* action mask */
unsigned int dev_action[ATA_MAX_DEVICES]; /* dev EH action */
unsigned int flags; /* ATA_EHI_* flags */ unsigned int flags; /* ATA_EHI_* flags */
unsigned long hotplug_timestamp; unsigned long hotplug_timestamp;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment