mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
ahci: add workaround for on-board 5723s on some gigabyte boards
Some gigabytes have on-board SIMG5723s connected to JMB ahcis. These are used to implement hardware raid. Unfortunately some firmware revisions on these 5723s don't bring the link down when all the downstream ports are unoccupied while not responding to reset protocol which makes libata think that there's device attached to the port but is not responding and retry. This results in painfully wrong boot detection time for these ports when they're empty. This patch quirks those boards such that ahci gives up after the initial timeout. Combined with parallel probing, this gives quick enough probing and also is safe because SIMG5723 will respond to the first try if any of the downstream ports is occupied. Signed-off-by: Tejun Heo <tj@kernel.org> Reported-by: Marc Bowes <marcbowes@gmail.com> Reported-by: Nicolas Mailhot <Nicolas.Mailhot@LaPoste.net> Signed-off-by: Jeff Garzik <jgarzik@redhat.com>
This commit is contained in:
parent
b6931c1fba
commit
5594639aab
1 changed files with 72 additions and 4 deletions
|
@ -219,6 +219,8 @@ enum {
|
||||||
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 */
|
AHCI_HFLAG_NO_SUSPEND = (1 << 10), /* don't suspend */
|
||||||
|
AHCI_HFLAG_SRST_TOUT_IS_OFFLINE = (1 << 11), /* treat SRST timeout as
|
||||||
|
link offline */
|
||||||
|
|
||||||
/* ap->flags bits */
|
/* ap->flags bits */
|
||||||
|
|
||||||
|
@ -1663,6 +1665,7 @@ static int ahci_do_softreset(struct ata_link *link, unsigned int *class,
|
||||||
int (*check_ready)(struct ata_link *link))
|
int (*check_ready)(struct ata_link *link))
|
||||||
{
|
{
|
||||||
struct ata_port *ap = link->ap;
|
struct ata_port *ap = link->ap;
|
||||||
|
struct ahci_host_priv *hpriv = ap->host->private_data;
|
||||||
const char *reason = NULL;
|
const char *reason = NULL;
|
||||||
unsigned long now, msecs;
|
unsigned long now, msecs;
|
||||||
struct ata_taskfile tf;
|
struct ata_taskfile tf;
|
||||||
|
@ -1701,12 +1704,21 @@ static int ahci_do_softreset(struct ata_link *link, unsigned int *class,
|
||||||
|
|
||||||
/* wait for link to become ready */
|
/* wait for link to become ready */
|
||||||
rc = ata_wait_after_reset(link, deadline, check_ready);
|
rc = ata_wait_after_reset(link, deadline, check_ready);
|
||||||
/* link occupied, -ENODEV too is an error */
|
if (rc == -EBUSY && hpriv->flags & AHCI_HFLAG_SRST_TOUT_IS_OFFLINE) {
|
||||||
if (rc) {
|
/*
|
||||||
|
* Workaround for cases where link online status can't
|
||||||
|
* be trusted. Treat device readiness timeout as link
|
||||||
|
* offline.
|
||||||
|
*/
|
||||||
|
ata_link_printk(link, KERN_INFO,
|
||||||
|
"device not ready, treating as offline\n");
|
||||||
|
*class = ATA_DEV_NONE;
|
||||||
|
} else if (rc) {
|
||||||
|
/* link occupied, -ENODEV too is an error */
|
||||||
reason = "device not ready";
|
reason = "device not ready";
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
} else
|
||||||
*class = ahci_dev_classify(ap);
|
*class = ahci_dev_classify(ap);
|
||||||
|
|
||||||
DPRINTK("EXIT, class=%u\n", *class);
|
DPRINTK("EXIT, class=%u\n", *class);
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -2727,6 +2739,56 @@ static bool ahci_broken_suspend(struct pci_dev *pdev)
|
||||||
return !ver || strcmp(ver, dmi->driver_data) < 0;
|
return !ver || strcmp(ver, dmi->driver_data) < 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool ahci_broken_online(struct pci_dev *pdev)
|
||||||
|
{
|
||||||
|
#define ENCODE_BUSDEVFN(bus, slot, func) \
|
||||||
|
(void *)(unsigned long)(((bus) << 8) | PCI_DEVFN((slot), (func)))
|
||||||
|
static const struct dmi_system_id sysids[] = {
|
||||||
|
/*
|
||||||
|
* There are several gigabyte boards which use
|
||||||
|
* SIMG5723s configured as hardware RAID. Certain
|
||||||
|
* 5723 firmware revisions shipped there keep the link
|
||||||
|
* online but fail to answer properly to SRST or
|
||||||
|
* IDENTIFY when no device is attached downstream
|
||||||
|
* causing libata to retry quite a few times leading
|
||||||
|
* to excessive detection delay.
|
||||||
|
*
|
||||||
|
* As these firmwares respond to the second reset try
|
||||||
|
* with invalid device signature, considering unknown
|
||||||
|
* sig as offline works around the problem acceptably.
|
||||||
|
*/
|
||||||
|
{
|
||||||
|
.ident = "EP45-DQ6",
|
||||||
|
.matches = {
|
||||||
|
DMI_MATCH(DMI_BOARD_VENDOR,
|
||||||
|
"Gigabyte Technology Co., Ltd."),
|
||||||
|
DMI_MATCH(DMI_BOARD_NAME, "EP45-DQ6"),
|
||||||
|
},
|
||||||
|
.driver_data = ENCODE_BUSDEVFN(0x0a, 0x00, 0),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
.ident = "EP45-DS5",
|
||||||
|
.matches = {
|
||||||
|
DMI_MATCH(DMI_BOARD_VENDOR,
|
||||||
|
"Gigabyte Technology Co., Ltd."),
|
||||||
|
DMI_MATCH(DMI_BOARD_NAME, "EP45-DS5"),
|
||||||
|
},
|
||||||
|
.driver_data = ENCODE_BUSDEVFN(0x03, 0x00, 0),
|
||||||
|
},
|
||||||
|
{ } /* terminate list */
|
||||||
|
};
|
||||||
|
#undef ENCODE_BUSDEVFN
|
||||||
|
const struct dmi_system_id *dmi = dmi_first_match(sysids);
|
||||||
|
unsigned int val;
|
||||||
|
|
||||||
|
if (!dmi)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
val = (unsigned long)dmi->driver_data;
|
||||||
|
|
||||||
|
return pdev->bus->number == (val >> 8) && pdev->devfn == (val & 0xff);
|
||||||
|
}
|
||||||
|
|
||||||
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;
|
||||||
|
@ -2842,6 +2904,12 @@ static int ahci_init_one(struct pci_dev *pdev, const struct pci_device_id *ent)
|
||||||
"BIOS update required for suspend/resume\n");
|
"BIOS update required for suspend/resume\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (ahci_broken_online(pdev)) {
|
||||||
|
hpriv->flags |= AHCI_HFLAG_SRST_TOUT_IS_OFFLINE;
|
||||||
|
dev_info(&pdev->dev,
|
||||||
|
"online status unreliable, applying workaround\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
|
||||||
|
|
Loading…
Reference in a new issue