Merge branch 'hibern_fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/jgarzik/libata-dev
* 'hibern_fixes' of git://git.kernel.org/pub/scm/linux/kernel/git/jgarzik/libata-dev: SATA PIIX: Blacklist system that spins off disks during ACPI power off SATA Sil: Blacklist system that spins off disks during ACPI power off SATA AHCI: Blacklist system that spins off disks during ACPI power off SATA: Blacklisting of systems that spin off disks during ACPI power off DMI: Introduce dmi_first_match to make the interface more flexible Hibernation: Introduce system_entering_hibernation
This commit is contained in:
@@ -2548,6 +2548,32 @@ static void ahci_p5wdh_workaround(struct ata_host *host)
|
||||
}
|
||||
}
|
||||
|
||||
static bool ahci_broken_system_poweroff(struct pci_dev *pdev)
|
||||
{
|
||||
static const struct dmi_system_id broken_systems[] = {
|
||||
{
|
||||
.ident = "HP Compaq nx6310",
|
||||
.matches = {
|
||||
DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
||||
DMI_MATCH(DMI_PRODUCT_NAME, "HP Compaq nx6310"),
|
||||
},
|
||||
/* PCI slot number of the controller */
|
||||
.driver_data = (void *)0x1FUL,
|
||||
},
|
||||
|
||||
{ } /* terminate list */
|
||||
};
|
||||
const struct dmi_system_id *dmi = dmi_first_match(broken_systems);
|
||||
|
||||
if (dmi) {
|
||||
unsigned long slot = (unsigned long)dmi->driver_data;
|
||||
/* apply the quirk only to on-board controllers */
|
||||
return slot == PCI_SLOT(pdev->devfn);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static int ahci_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
{
|
||||
static int printed_version;
|
||||
@@ -2647,6 +2673,12 @@ static int ahci_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
}
|
||||
}
|
||||
|
||||
if (ahci_broken_system_poweroff(pdev)) {
|
||||
pi.flags |= ATA_FLAG_NO_POWEROFF_SPINDOWN;
|
||||
dev_info(&pdev->dev,
|
||||
"quirky BIOS, skipping spindown on poweroff\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
|
||||
|
@@ -1387,6 +1387,32 @@ static void piix_iocfg_bit18_quirk(struct ata_host *host)
|
||||
}
|
||||
}
|
||||
|
||||
static bool piix_broken_system_poweroff(struct pci_dev *pdev)
|
||||
{
|
||||
static const struct dmi_system_id broken_systems[] = {
|
||||
{
|
||||
.ident = "HP Compaq 2510p",
|
||||
.matches = {
|
||||
DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
||||
DMI_MATCH(DMI_PRODUCT_NAME, "HP Compaq 2510p"),
|
||||
},
|
||||
/* PCI slot number of the controller */
|
||||
.driver_data = (void *)0x1FUL,
|
||||
},
|
||||
|
||||
{ } /* terminate list */
|
||||
};
|
||||
const struct dmi_system_id *dmi = dmi_first_match(broken_systems);
|
||||
|
||||
if (dmi) {
|
||||
unsigned long slot = (unsigned long)dmi->driver_data;
|
||||
/* apply the quirk only to on-board controllers */
|
||||
return slot == PCI_SLOT(pdev->devfn);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* piix_init_one - Register PIIX ATA PCI device with kernel services
|
||||
* @pdev: PCI device to register
|
||||
@@ -1422,6 +1448,14 @@ static int __devinit piix_init_one(struct pci_dev *pdev,
|
||||
if (!in_module_init)
|
||||
return -ENODEV;
|
||||
|
||||
if (piix_broken_system_poweroff(pdev)) {
|
||||
piix_port_info[ent->driver_data].flags |=
|
||||
ATA_FLAG_NO_POWEROFF_SPINDOWN |
|
||||
ATA_FLAG_NO_HIBERNATE_SPINDOWN;
|
||||
dev_info(&pdev->dev, "quirky BIOS, skipping spindown "
|
||||
"on poweroff and hibernation\n");
|
||||
}
|
||||
|
||||
port_info[0] = piix_port_info[ent->driver_data];
|
||||
port_info[1] = piix_port_info[ent->driver_data];
|
||||
|
||||
|
@@ -46,6 +46,7 @@
|
||||
#include <linux/libata.h>
|
||||
#include <linux/hdreg.h>
|
||||
#include <linux/uaccess.h>
|
||||
#include <linux/suspend.h>
|
||||
|
||||
#include "libata.h"
|
||||
|
||||
@@ -1303,6 +1304,17 @@ static unsigned int ata_scsi_start_stop_xlat(struct ata_queued_cmd *qc)
|
||||
|
||||
tf->command = ATA_CMD_VERIFY; /* READ VERIFY */
|
||||
} else {
|
||||
/* Some odd clown BIOSen issue spindown on power off (ACPI S4
|
||||
* or S5) causing some drives to spin up and down again.
|
||||
*/
|
||||
if ((qc->ap->flags & ATA_FLAG_NO_POWEROFF_SPINDOWN) &&
|
||||
system_state == SYSTEM_POWER_OFF)
|
||||
goto skip;
|
||||
|
||||
if ((qc->ap->flags & ATA_FLAG_NO_HIBERNATE_SPINDOWN) &&
|
||||
system_entering_hibernation())
|
||||
goto skip;
|
||||
|
||||
/* XXX: This is for backward compatibility, will be
|
||||
* removed. Read Documentation/feature-removal-schedule.txt
|
||||
* for more info.
|
||||
@@ -1326,8 +1338,7 @@ static unsigned int ata_scsi_start_stop_xlat(struct ata_queued_cmd *qc)
|
||||
scmd->scsi_done = qc->scsidone;
|
||||
qc->scsidone = ata_delayed_done;
|
||||
}
|
||||
scmd->result = SAM_STAT_GOOD;
|
||||
return 1;
|
||||
goto skip;
|
||||
}
|
||||
|
||||
/* Issue ATA STANDBY IMMEDIATE command */
|
||||
@@ -1343,10 +1354,13 @@ static unsigned int ata_scsi_start_stop_xlat(struct ata_queued_cmd *qc)
|
||||
|
||||
return 0;
|
||||
|
||||
invalid_fld:
|
||||
invalid_fld:
|
||||
ata_scsi_set_sense(scmd, ILLEGAL_REQUEST, 0x24, 0x0);
|
||||
/* "Invalid field in cbd" */
|
||||
return 1;
|
||||
skip:
|
||||
scmd->result = SAM_STAT_GOOD;
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
||||
|
@@ -695,11 +695,38 @@ static void sil_init_controller(struct ata_host *host)
|
||||
}
|
||||
}
|
||||
|
||||
static bool sil_broken_system_poweroff(struct pci_dev *pdev)
|
||||
{
|
||||
static const struct dmi_system_id broken_systems[] = {
|
||||
{
|
||||
.ident = "HP Compaq nx6325",
|
||||
.matches = {
|
||||
DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
||||
DMI_MATCH(DMI_PRODUCT_NAME, "HP Compaq nx6325"),
|
||||
},
|
||||
/* PCI slot number of the controller */
|
||||
.driver_data = (void *)0x12UL,
|
||||
},
|
||||
|
||||
{ } /* terminate list */
|
||||
};
|
||||
const struct dmi_system_id *dmi = dmi_first_match(broken_systems);
|
||||
|
||||
if (dmi) {
|
||||
unsigned long slot = (unsigned long)dmi->driver_data;
|
||||
/* apply the quirk only to on-board controllers */
|
||||
return slot == PCI_SLOT(pdev->devfn);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
static int sil_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
{
|
||||
static int printed_version;
|
||||
int board_id = ent->driver_data;
|
||||
const struct ata_port_info *ppi[] = { &sil_port_info[board_id], NULL };
|
||||
struct ata_port_info pi = sil_port_info[board_id];
|
||||
const struct ata_port_info *ppi[] = { &pi, NULL };
|
||||
struct ata_host *host;
|
||||
void __iomem *mmio_base;
|
||||
int n_ports, rc;
|
||||
@@ -713,6 +740,13 @@ static int sil_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||
if (board_id == sil_3114)
|
||||
n_ports = 4;
|
||||
|
||||
if (sil_broken_system_poweroff(pdev)) {
|
||||
pi.flags |= ATA_FLAG_NO_POWEROFF_SPINDOWN |
|
||||
ATA_FLAG_NO_HIBERNATE_SPINDOWN;
|
||||
dev_info(&pdev->dev, "quirky BIOS, skipping spindown "
|
||||
"on poweroff and hibernation\n");
|
||||
}
|
||||
|
||||
host = ata_host_alloc_pinfo(&pdev->dev, ppi, n_ports);
|
||||
if (!host)
|
||||
return -ENOMEM;
|
||||
|
Reference in New Issue
Block a user