Commit 398e0782 authored by Tejun Heo's avatar Tejun Heo Committed by Jeff Garzik

libata-acpi: implement dev->gtf_cache and evaluate _GTF right after _STM during resume

On certain implementations, _GTF evaluation depends on preceding _STM
and both can be pretty picky about the configuration.  Using _GTM
result cached during controller initialization satisfies the most
neurotic _STM implementation.  However, libata evaluates _GTF after
reset during device configuration and the hardware state can be
different from what _GTF expects and can cause evaluation failure.

This patch adds dev->gtf_cache and updates ata_dev_get_GTF() such that
it uses the cached value if available.  Cache is cleared with a call
to ata_acpi_clear_gtf().

Because for SATA ACPI nodes _GTF must be evaluated after _SDD which
can't be done till IDENTIFY is complete, _GTF caching from
ata_acpi_on_resume() is used only for IDE ACPI nodes.
Signed-off-by: default avatarTejun Heo <htejun@gmail.com>
Signed-off-by: default avatarJeff Garzik <jeff@garzik.org>
parent c05e6ff0
...@@ -41,6 +41,12 @@ static int is_pci_dev(struct device *dev) ...@@ -41,6 +41,12 @@ static int is_pci_dev(struct device *dev)
return (dev->bus == &pci_bus_type); return (dev->bus == &pci_bus_type);
} }
static void ata_acpi_clear_gtf(struct ata_device *dev)
{
kfree(dev->gtf_cache);
dev->gtf_cache = NULL;
}
/** /**
* ata_acpi_associate_sata_port - associate SATA port with ACPI objects * ata_acpi_associate_sata_port - associate SATA port with ACPI objects
* @ap: target SATA port * @ap: target SATA port
...@@ -327,7 +333,6 @@ EXPORT_SYMBOL_GPL(ata_acpi_stm); ...@@ -327,7 +333,6 @@ EXPORT_SYMBOL_GPL(ata_acpi_stm);
* ata_dev_get_GTF - get the drive bootup default taskfile settings * ata_dev_get_GTF - get the drive bootup default taskfile settings
* @dev: target ATA device * @dev: target ATA device
* @gtf: output parameter for buffer containing _GTF taskfile arrays * @gtf: output parameter for buffer containing _GTF taskfile arrays
* @ptr_to_free: pointer which should be freed
* *
* This applies to both PATA and SATA drives. * This applies to both PATA and SATA drives.
* *
...@@ -344,8 +349,7 @@ EXPORT_SYMBOL_GPL(ata_acpi_stm); ...@@ -344,8 +349,7 @@ EXPORT_SYMBOL_GPL(ata_acpi_stm);
* Number of taskfiles on success, 0 if _GTF doesn't exist or doesn't * Number of taskfiles on success, 0 if _GTF doesn't exist or doesn't
* contain valid data. * contain valid data.
*/ */
static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf, static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf)
void **ptr_to_free)
{ {
struct ata_port *ap = dev->link->ap; struct ata_port *ap = dev->link->ap;
acpi_status status; acpi_status status;
...@@ -353,6 +357,12 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf, ...@@ -353,6 +357,12 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf,
union acpi_object *out_obj; union acpi_object *out_obj;
int rc = 0; int rc = 0;
/* if _GTF is cached, use the cached value */
if (dev->gtf_cache) {
out_obj = dev->gtf_cache;
goto done;
}
/* set up output buffer */ /* set up output buffer */
output.length = ACPI_ALLOCATE_BUFFER; output.length = ACPI_ALLOCATE_BUFFER;
output.pointer = NULL; /* ACPI-CA sets this; save/free it later */ output.pointer = NULL; /* ACPI-CA sets this; save/free it later */
...@@ -363,6 +373,7 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf, ...@@ -363,6 +373,7 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf,
/* _GTF has no input parameters */ /* _GTF has no input parameters */
status = acpi_evaluate_object(dev->acpi_handle, "_GTF", NULL, &output); status = acpi_evaluate_object(dev->acpi_handle, "_GTF", NULL, &output);
out_obj = dev->gtf_cache = output.pointer;
if (ACPI_FAILURE(status)) { if (ACPI_FAILURE(status)) {
if (status != AE_NOT_FOUND) { if (status != AE_NOT_FOUND) {
...@@ -383,7 +394,6 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf, ...@@ -383,7 +394,6 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf,
goto out_free; goto out_free;
} }
out_obj = output.pointer;
if (out_obj->type != ACPI_TYPE_BUFFER) { if (out_obj->type != ACPI_TYPE_BUFFER) {
ata_dev_printk(dev, KERN_WARNING, ata_dev_printk(dev, KERN_WARNING,
"_GTF unexpected object type 0x%x\n", "_GTF unexpected object type 0x%x\n",
...@@ -398,18 +408,19 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf, ...@@ -398,18 +408,19 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf,
goto out_free; goto out_free;
} }
*ptr_to_free = out_obj; done:
*gtf = (void *)out_obj->buffer.pointer;
rc = out_obj->buffer.length / REGS_PER_GTF; rc = out_obj->buffer.length / REGS_PER_GTF;
if (gtf) {
if (ata_msg_probe(ap)) *gtf = (void *)out_obj->buffer.pointer;
ata_dev_printk(dev, KERN_DEBUG, "%s: returning " if (ata_msg_probe(ap))
"gtf=%p, gtf_count=%d, ptr_to_free=%p\n", ata_dev_printk(dev, KERN_DEBUG,
__FUNCTION__, *gtf, rc, *ptr_to_free); "%s: returning gtf=%p, gtf_count=%d\n",
__FUNCTION__, *gtf, rc);
}
return rc; return rc;
out_free: out_free:
kfree(output.pointer); ata_acpi_clear_gtf(dev);
return rc; return rc;
} }
...@@ -533,11 +544,10 @@ static int taskfile_load_raw(struct ata_device *dev, ...@@ -533,11 +544,10 @@ static int taskfile_load_raw(struct ata_device *dev,
static int ata_acpi_exec_tfs(struct ata_device *dev) static int ata_acpi_exec_tfs(struct ata_device *dev)
{ {
struct ata_acpi_gtf *gtf = NULL; struct ata_acpi_gtf *gtf = NULL;
void *ptr_to_free = NULL;
int gtf_count, i, rc; int gtf_count, i, rc;
/* get taskfiles */ /* get taskfiles */
gtf_count = ata_dev_get_GTF(dev, &gtf, &ptr_to_free); gtf_count = ata_dev_get_GTF(dev, &gtf);
/* execute them */ /* execute them */
for (i = 0, rc = 0; i < gtf_count; i++) { for (i = 0, rc = 0; i < gtf_count; i++) {
...@@ -551,7 +561,7 @@ static int ata_acpi_exec_tfs(struct ata_device *dev) ...@@ -551,7 +561,7 @@ static int ata_acpi_exec_tfs(struct ata_device *dev)
rc = tmp; rc = tmp;
} }
kfree(ptr_to_free); ata_acpi_clear_gtf(dev);
if (rc == 0) if (rc == 0)
return gtf_count; return gtf_count;
...@@ -644,13 +654,31 @@ void ata_acpi_on_resume(struct ata_port *ap) ...@@ -644,13 +654,31 @@ void ata_acpi_on_resume(struct ata_port *ap)
const struct ata_acpi_gtm *gtm = ata_acpi_init_gtm(ap); const struct ata_acpi_gtm *gtm = ata_acpi_init_gtm(ap);
struct ata_device *dev; struct ata_device *dev;
/* restore timing parameters */ if (ap->acpi_handle && gtm) {
if (ap->acpi_handle && gtm) /* _GTM valid */
/* restore timing parameters */
ata_acpi_stm(ap, gtm); ata_acpi_stm(ap, gtm);
/* schedule _GTF */ /* _GTF should immediately follow _STM so that it can
ata_link_for_each_dev(dev, &ap->link) * use values set by _STM. Cache _GTF result and
dev->flags |= ATA_DFLAG_ACPI_PENDING; * schedule _GTF.
*/
ata_link_for_each_dev(dev, &ap->link) {
ata_acpi_clear_gtf(dev);
if (ata_dev_get_GTF(dev, NULL) >= 0)
dev->flags |= ATA_DFLAG_ACPI_PENDING;
}
} else {
/* SATA _GTF needs to be evaulated after _SDD and
* there's no reason to evaluate IDE _GTF early
* without _STM. Clear cache and schedule _GTF.
*/
ata_link_for_each_dev(dev, &ap->link) {
ata_acpi_clear_gtf(dev);
dev->flags |= ATA_DFLAG_ACPI_PENDING;
}
}
} }
/** /**
...@@ -735,4 +763,5 @@ int ata_acpi_on_devcfg(struct ata_device *dev) ...@@ -735,4 +763,5 @@ int ata_acpi_on_devcfg(struct ata_device *dev)
*/ */
void ata_acpi_on_disable(struct ata_device *dev) void ata_acpi_on_disable(struct ata_device *dev)
{ {
ata_acpi_clear_gtf(dev);
} }
...@@ -498,6 +498,7 @@ struct ata_device { ...@@ -498,6 +498,7 @@ struct ata_device {
struct scsi_device *sdev; /* attached SCSI device */ struct scsi_device *sdev; /* attached SCSI device */
#ifdef CONFIG_ATA_ACPI #ifdef CONFIG_ATA_ACPI
acpi_handle acpi_handle; acpi_handle acpi_handle;
union acpi_object *gtf_cache;
#endif #endif
/* n_sector is used as CLEAR_OFFSET, read comment above CLEAR_OFFSET */ /* n_sector is used as CLEAR_OFFSET, read comment above CLEAR_OFFSET */
u64 n_sectors; /* size of device, if ATA */ u64 n_sectors; /* size of device, if ATA */
......
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