Merge branch 'upstream-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/jgarzik/libata-dev
* 'upstream-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/jgarzik/libata-dev: ata_piix: Add HP Compaq nc6000 to the broken poweroff list ahci: add warning messages for hp laptops with broken suspend pata_efar: fix PIO2 underclocking pata_legacy: wait for async probing
This commit is contained in:
commit
be94a4ba09
@ -220,6 +220,7 @@ enum {
|
|||||||
AHCI_HFLAG_NO_HOTPLUG = (1 << 7), /* ignore PxSERR.DIAG.N */
|
AHCI_HFLAG_NO_HOTPLUG = (1 << 7), /* ignore PxSERR.DIAG.N */
|
||||||
AHCI_HFLAG_SECT255 = (1 << 8), /* max 255 sectors */
|
AHCI_HFLAG_SECT255 = (1 << 8), /* max 255 sectors */
|
||||||
AHCI_HFLAG_YES_NCQ = (1 << 9), /* force NCQ cap on */
|
AHCI_HFLAG_YES_NCQ = (1 << 9), /* force NCQ cap on */
|
||||||
|
AHCI_HFLAG_NO_SUSPEND = (1 << 10), /* don't suspend */
|
||||||
|
|
||||||
/* ap->flags bits */
|
/* 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)
|
static int ahci_pci_device_suspend(struct pci_dev *pdev, pm_message_t mesg)
|
||||||
{
|
{
|
||||||
struct ata_host *host = dev_get_drvdata(&pdev->dev);
|
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];
|
void __iomem *mmio = host->iomap[AHCI_PCI_BAR];
|
||||||
u32 ctl;
|
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) {
|
if (mesg.event & PM_EVENT_SLEEP) {
|
||||||
/* AHCI spec rev1.1 section 8.3.3:
|
/* AHCI spec rev1.1 section 8.3.3:
|
||||||
* Software must disable interrupts prior to requesting a
|
* Software must disable interrupts prior to requesting a
|
||||||
@ -2610,6 +2619,63 @@ static bool ahci_broken_system_poweroff(struct pci_dev *pdev)
|
|||||||
return false;
|
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 ahci_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||||
{
|
{
|
||||||
static int printed_version;
|
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");
|
"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
|
/* CAP.NP sometimes indicate the index of the last enabled
|
||||||
* port, at other times, that of the last possible port, so
|
* port, at other times, that of the last possible port, so
|
||||||
* determining the maximum port number requires looking at
|
* determining the maximum port number requires looking at
|
||||||
|
@ -1455,6 +1455,15 @@ static bool piix_broken_system_poweroff(struct pci_dev *pdev)
|
|||||||
/* PCI slot number of the controller */
|
/* PCI slot number of the controller */
|
||||||
.driver_data = (void *)0x1FUL,
|
.driver_data = (void *)0x1FUL,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
.ident = "HP Compaq nc6000",
|
||||||
|
.matches = {
|
||||||
|
DMI_MATCH(DMI_SYS_VENDOR, "Hewlett-Packard"),
|
||||||
|
DMI_MATCH(DMI_PRODUCT_NAME, "HP Compaq nc6000"),
|
||||||
|
},
|
||||||
|
/* PCI slot number of the controller */
|
||||||
|
.driver_data = (void *)0x1FUL,
|
||||||
|
},
|
||||||
|
|
||||||
{ } /* terminate list */
|
{ } /* terminate list */
|
||||||
};
|
};
|
||||||
|
@ -22,7 +22,7 @@
|
|||||||
#include <linux/ata.h>
|
#include <linux/ata.h>
|
||||||
|
|
||||||
#define DRV_NAME "pata_efar"
|
#define DRV_NAME "pata_efar"
|
||||||
#define DRV_VERSION "0.4.4"
|
#define DRV_VERSION "0.4.5"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* efar_pre_reset - Enable bits
|
* efar_pre_reset - Enable bits
|
||||||
@ -98,18 +98,17 @@ static void efar_set_piomode (struct ata_port *ap, struct ata_device *adev)
|
|||||||
{ 2, 1 },
|
{ 2, 1 },
|
||||||
{ 2, 3 }, };
|
{ 2, 3 }, };
|
||||||
|
|
||||||
if (pio > 2)
|
if (pio > 1)
|
||||||
control |= 1; /* TIME1 enable */
|
control |= 1; /* TIME */
|
||||||
if (ata_pio_need_iordy(adev)) /* PIO 3/4 require IORDY */
|
if (ata_pio_need_iordy(adev)) /* PIO 3/4 require IORDY */
|
||||||
control |= 2; /* IE enable */
|
control |= 2; /* IE */
|
||||||
/* Intel specifies that the PPE functionality is for disk only */
|
/* Intel specifies that the prefetch/posting is for disk only */
|
||||||
if (adev->class == ATA_DEV_ATA)
|
if (adev->class == ATA_DEV_ATA)
|
||||||
control |= 4; /* PPE enable */
|
control |= 4; /* PPE */
|
||||||
|
|
||||||
pci_read_config_word(dev, idetm_port, &idetm_data);
|
pci_read_config_word(dev, idetm_port, &idetm_data);
|
||||||
|
|
||||||
/* Enable PPE, IE and TIME as appropriate */
|
/* Set PPE, IE, and TIME as appropriate */
|
||||||
|
|
||||||
if (adev->devno == 0) {
|
if (adev->devno == 0) {
|
||||||
idetm_data &= 0xCCF0;
|
idetm_data &= 0xCCF0;
|
||||||
idetm_data |= control;
|
idetm_data |= control;
|
||||||
@ -129,7 +128,7 @@ static void efar_set_piomode (struct ata_port *ap, struct ata_device *adev)
|
|||||||
pci_write_config_byte(dev, 0x44, slave_data);
|
pci_write_config_byte(dev, 0x44, slave_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
idetm_data |= 0x4000; /* Ensure SITRE is enabled */
|
idetm_data |= 0x4000; /* Ensure SITRE is set */
|
||||||
pci_write_config_word(dev, idetm_port, idetm_data);
|
pci_write_config_word(dev, idetm_port, idetm_data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -48,6 +48,7 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <linux/async.h>
|
||||||
#include <linux/kernel.h>
|
#include <linux/kernel.h>
|
||||||
#include <linux/module.h>
|
#include <linux/module.h>
|
||||||
#include <linux/pci.h>
|
#include <linux/pci.h>
|
||||||
@ -1028,6 +1029,7 @@ static __init int legacy_init_one(struct legacy_probe *probe)
|
|||||||
&legacy_sht);
|
&legacy_sht);
|
||||||
if (ret)
|
if (ret)
|
||||||
goto fail;
|
goto fail;
|
||||||
|
async_synchronize_full();
|
||||||
ld->platform_dev = pdev;
|
ld->platform_dev = pdev;
|
||||||
|
|
||||||
/* Nothing found means we drop the port as its probably not there */
|
/* Nothing found means we drop the port as its probably not there */
|
||||||
|
Loading…
Reference in New Issue
Block a user