|
@@ -220,6 +220,7 @@ enum {
|
|
|
AHCI_HFLAG_NO_HOTPLUG = (1 << 7), /* ignore PxSERR.DIAG.N */
|
|
|
AHCI_HFLAG_SECT255 = (1 << 8), /* max 255 sectors */
|
|
|
AHCI_HFLAG_YES_NCQ = (1 << 9), /* force NCQ cap on */
|
|
|
+ AHCI_HFLAG_NO_SUSPEND = (1 << 10), /* don't suspend */
|
|
|
|
|
|
/* ap->flags bits */
|
|
|
|
|
@@ -2316,9 +2317,17 @@ static int ahci_port_suspend(struct ata_port *ap, pm_message_t mesg)
|
|
|
static int ahci_pci_device_suspend(struct pci_dev *pdev, pm_message_t mesg)
|
|
|
{
|
|
|
struct ata_host *host = dev_get_drvdata(&pdev->dev);
|
|
|
+ struct ahci_host_priv *hpriv = host->private_data;
|
|
|
void __iomem *mmio = host->iomap[AHCI_PCI_BAR];
|
|
|
u32 ctl;
|
|
|
|
|
|
+ if (mesg.event & PM_EVENT_SUSPEND &&
|
|
|
+ hpriv->flags & AHCI_HFLAG_NO_SUSPEND) {
|
|
|
+ dev_printk(KERN_ERR, &pdev->dev,
|
|
|
+ "BIOS update required for suspend/resume\n");
|
|
|
+ return -EIO;
|
|
|
+ }
|
|
|
+
|
|
|
if (mesg.event & PM_EVENT_SLEEP) {
|
|
|
/* AHCI spec rev1.1 section 8.3.3:
|
|
|
* Software must disable interrupts prior to requesting a
|
|
@@ -2610,6 +2619,63 @@ static bool ahci_broken_system_poweroff(struct pci_dev *pdev)
|
|
|
return false;
|
|
|
}
|
|
|
|
|
|
+static bool ahci_broken_suspend(struct pci_dev *pdev)
|
|
|
+{
|
|
|
+ static const struct dmi_system_id sysids[] = {
|
|
|
+ /*
|
|
|
+ * On HP dv[4-6] and HDX18 with earlier BIOSen, link
|
|
|
+ * to the harddisk doesn't become online after
|
|
|
+ * resuming from STR. Warn and fail suspend.
|
|
|
+ */
|
|
|
+ {
|
|
|
+ .ident = "dv4",
|
|
|
+ .matches = {
|
|
|
+ DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
|
|
+ DMI_MATCH(DMI_PRODUCT_NAME,
|
|
|
+ "HP Pavilion dv4 Notebook PC"),
|
|
|
+ },
|
|
|
+ .driver_data = "F.30", /* cutoff BIOS version */
|
|
|
+ },
|
|
|
+ {
|
|
|
+ .ident = "dv5",
|
|
|
+ .matches = {
|
|
|
+ DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
|
|
+ DMI_MATCH(DMI_PRODUCT_NAME,
|
|
|
+ "HP Pavilion dv5 Notebook PC"),
|
|
|
+ },
|
|
|
+ .driver_data = "F.16", /* cutoff BIOS version */
|
|
|
+ },
|
|
|
+ {
|
|
|
+ .ident = "dv6",
|
|
|
+ .matches = {
|
|
|
+ DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
|
|
+ DMI_MATCH(DMI_PRODUCT_NAME,
|
|
|
+ "HP Pavilion dv6 Notebook PC"),
|
|
|
+ },
|
|
|
+ .driver_data = "F.21", /* cutoff BIOS version */
|
|
|
+ },
|
|
|
+ {
|
|
|
+ .ident = "HDX18",
|
|
|
+ .matches = {
|
|
|
+ DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
|
|
+ DMI_MATCH(DMI_PRODUCT_NAME,
|
|
|
+ "HP HDX18 Notebook PC"),
|
|
|
+ },
|
|
|
+ .driver_data = "F.23", /* cutoff BIOS version */
|
|
|
+ },
|
|
|
+ { } /* terminate list */
|
|
|
+ };
|
|
|
+ const struct dmi_system_id *dmi = dmi_first_match(sysids);
|
|
|
+ const char *ver;
|
|
|
+
|
|
|
+ if (!dmi || pdev->bus->number || pdev->devfn != PCI_DEVFN(0x1f, 2))
|
|
|
+ return false;
|
|
|
+
|
|
|
+ ver = dmi_get_system_info(DMI_BIOS_VERSION);
|
|
|
+
|
|
|
+ return !ver || strcmp(ver, dmi->driver_data) < 0;
|
|
|
+}
|
|
|
+
|
|
|
static int ahci_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|
|
{
|
|
|
static int printed_version;
|
|
@@ -2715,6 +2781,12 @@ static int ahci_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
|
|
"quirky BIOS, skipping spindown on poweroff\n");
|
|
|
}
|
|
|
|
|
|
+ if (ahci_broken_suspend(pdev)) {
|
|
|
+ hpriv->flags |= AHCI_HFLAG_NO_SUSPEND;
|
|
|
+ dev_printk(KERN_WARNING, &pdev->dev,
|
|
|
+ "BIOS update required for suspend/resume\n");
|
|
|
+ }
|
|
|
+
|
|
|
/* CAP.NP sometimes indicate the index of the last enabled
|
|
|
* port, at other times, that of the last possible port, so
|
|
|
* determining the maximum port number requires looking at
|