libata-acpi: implement _GTF command filtering
Implement _GTF command filtering which can be controlled by libata.acpi_filter kernel parameter. Currently SETXFER and LOCK commands are filtered. libata configures transfer mode by itself and _GTF SETXFER commands can potentially disrupt device configuration. _GTM/_STM mechanism can't handle hotplugging too well and when _GTF is executed, controller is in PIO0 rather than the mode _STM configured. Note that detecting SET MAX LOCK requires looking at the previous command. This adds a bit to code complexity. Signed-off-by: Tejun Heo <htejun@gmail.com> Signed-off-by: Jeff Garzik <jeff@garzik.org>
This commit is contained in:
parent
0e8634bf8e
commit
3264a8d8f9
|
@ -6,6 +6,7 @@
|
||||||
* Copyright (C) 2006 Randy Dunlap
|
* Copyright (C) 2006 Randy Dunlap
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <linux/module.h>
|
||||||
#include <linux/ata.h>
|
#include <linux/ata.h>
|
||||||
#include <linux/delay.h>
|
#include <linux/delay.h>
|
||||||
#include <linux/device.h>
|
#include <linux/device.h>
|
||||||
|
@ -25,6 +26,18 @@
|
||||||
#include <acpi/acmacros.h>
|
#include <acpi/acmacros.h>
|
||||||
#include <acpi/actypes.h>
|
#include <acpi/actypes.h>
|
||||||
|
|
||||||
|
enum {
|
||||||
|
ATA_ACPI_FILTER_SETXFER = 1 << 0,
|
||||||
|
ATA_ACPI_FILTER_LOCK = 1 << 1,
|
||||||
|
|
||||||
|
ATA_ACPI_FILTER_DEFAULT = ATA_ACPI_FILTER_SETXFER |
|
||||||
|
ATA_ACPI_FILTER_LOCK,
|
||||||
|
};
|
||||||
|
|
||||||
|
static unsigned int ata_acpi_gtf_filter = ATA_ACPI_FILTER_DEFAULT;
|
||||||
|
module_param_named(acpi_gtf_filter, ata_acpi_gtf_filter, int, 0644);
|
||||||
|
MODULE_PARM_DESC(acpi_gtf_filter, "filter mask for ACPI _GTF commands, set to filter out (0x1=set xfermode, 0x2=lock/freeze lock)");
|
||||||
|
|
||||||
#define NO_PORT_MULT 0xffff
|
#define NO_PORT_MULT 0xffff
|
||||||
#define SATA_ADR(root, pmp) (((root) << 16) | (pmp))
|
#define SATA_ADR(root, pmp) (((root) << 16) | (pmp))
|
||||||
|
|
||||||
|
@ -465,6 +478,60 @@ int ata_acpi_cbl_80wire(struct ata_port *ap)
|
||||||
|
|
||||||
EXPORT_SYMBOL_GPL(ata_acpi_cbl_80wire);
|
EXPORT_SYMBOL_GPL(ata_acpi_cbl_80wire);
|
||||||
|
|
||||||
|
static void ata_acpi_gtf_to_tf(struct ata_device *dev,
|
||||||
|
const struct ata_acpi_gtf *gtf,
|
||||||
|
struct ata_taskfile *tf)
|
||||||
|
{
|
||||||
|
ata_tf_init(dev, tf);
|
||||||
|
|
||||||
|
tf->flags |= ATA_TFLAG_ISADDR | ATA_TFLAG_DEVICE;
|
||||||
|
tf->protocol = ATA_PROT_NODATA;
|
||||||
|
tf->feature = gtf->tf[0]; /* 0x1f1 */
|
||||||
|
tf->nsect = gtf->tf[1]; /* 0x1f2 */
|
||||||
|
tf->lbal = gtf->tf[2]; /* 0x1f3 */
|
||||||
|
tf->lbam = gtf->tf[3]; /* 0x1f4 */
|
||||||
|
tf->lbah = gtf->tf[4]; /* 0x1f5 */
|
||||||
|
tf->device = gtf->tf[5]; /* 0x1f6 */
|
||||||
|
tf->command = gtf->tf[6]; /* 0x1f7 */
|
||||||
|
}
|
||||||
|
|
||||||
|
static int ata_acpi_filter_tf(const struct ata_taskfile *tf,
|
||||||
|
const struct ata_taskfile *ptf)
|
||||||
|
{
|
||||||
|
if (ata_acpi_gtf_filter & ATA_ACPI_FILTER_SETXFER) {
|
||||||
|
/* libata doesn't use ACPI to configure transfer mode.
|
||||||
|
* It will only confuse device configuration. Skip.
|
||||||
|
*/
|
||||||
|
if (tf->command == ATA_CMD_SET_FEATURES &&
|
||||||
|
tf->feature == SETFEATURES_XFER)
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ata_acpi_gtf_filter & ATA_ACPI_FILTER_LOCK) {
|
||||||
|
/* BIOS writers, sorry but we don't wanna lock
|
||||||
|
* features unless the user explicitly said so.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* DEVICE CONFIGURATION FREEZE LOCK */
|
||||||
|
if (tf->command == ATA_CMD_CONF_OVERLAY &&
|
||||||
|
tf->feature == ATA_DCO_FREEZE_LOCK)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
/* SECURITY FREEZE LOCK */
|
||||||
|
if (tf->command == ATA_CMD_SEC_FREEZE_LOCK)
|
||||||
|
return 1;
|
||||||
|
|
||||||
|
/* SET MAX LOCK and SET MAX FREEZE LOCK */
|
||||||
|
if ((!ptf || ptf->command != ATA_CMD_READ_NATIVE_MAX) &&
|
||||||
|
tf->command == ATA_CMD_SET_MAX &&
|
||||||
|
(tf->feature == ATA_SET_MAX_LOCK ||
|
||||||
|
tf->feature == ATA_SET_MAX_FREEZE_LOCK))
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* ata_acpi_run_tf - send taskfile registers to host controller
|
* ata_acpi_run_tf - send taskfile registers to host controller
|
||||||
* @dev: target ATA device
|
* @dev: target ATA device
|
||||||
|
@ -485,13 +552,15 @@ EXPORT_SYMBOL_GPL(ata_acpi_cbl_80wire);
|
||||||
* EH context.
|
* EH context.
|
||||||
*
|
*
|
||||||
* RETURNS:
|
* RETURNS:
|
||||||
* 1 if command is executed successfully. 0 if ignored or rejected,
|
* 1 if command is executed successfully. 0 if ignored, rejected or
|
||||||
* -errno on other errors.
|
* filtered out, -errno on other errors.
|
||||||
*/
|
*/
|
||||||
static int ata_acpi_run_tf(struct ata_device *dev,
|
static int ata_acpi_run_tf(struct ata_device *dev,
|
||||||
const struct ata_acpi_gtf *gtf)
|
const struct ata_acpi_gtf *gtf,
|
||||||
|
const struct ata_acpi_gtf *prev_gtf)
|
||||||
{
|
{
|
||||||
struct ata_taskfile tf, rtf;
|
struct ata_taskfile *pptf = NULL;
|
||||||
|
struct ata_taskfile tf, ptf, rtf;
|
||||||
unsigned int err_mask;
|
unsigned int err_mask;
|
||||||
const char *level;
|
const char *level;
|
||||||
char msg[60];
|
char msg[60];
|
||||||
|
@ -502,44 +571,44 @@ static int ata_acpi_run_tf(struct ata_device *dev,
|
||||||
&& (gtf->tf[6] == 0))
|
&& (gtf->tf[6] == 0))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
ata_tf_init(dev, &tf);
|
ata_acpi_gtf_to_tf(dev, gtf, &tf);
|
||||||
|
if (prev_gtf) {
|
||||||
|
ata_acpi_gtf_to_tf(dev, prev_gtf, &ptf);
|
||||||
|
pptf = &ptf;
|
||||||
|
}
|
||||||
|
|
||||||
/* convert gtf to tf */
|
if (!ata_acpi_filter_tf(&tf, pptf)) {
|
||||||
tf.flags |= ATA_TFLAG_ISADDR | ATA_TFLAG_DEVICE; /* TBD */
|
rtf = tf;
|
||||||
tf.protocol = ATA_PROT_NODATA;
|
err_mask = ata_exec_internal(dev, &rtf, NULL,
|
||||||
tf.feature = gtf->tf[0]; /* 0x1f1 */
|
DMA_NONE, NULL, 0, 0);
|
||||||
tf.nsect = gtf->tf[1]; /* 0x1f2 */
|
|
||||||
tf.lbal = gtf->tf[2]; /* 0x1f3 */
|
|
||||||
tf.lbam = gtf->tf[3]; /* 0x1f4 */
|
|
||||||
tf.lbah = gtf->tf[4]; /* 0x1f5 */
|
|
||||||
tf.device = gtf->tf[5]; /* 0x1f6 */
|
|
||||||
tf.command = gtf->tf[6]; /* 0x1f7 */
|
|
||||||
|
|
||||||
rtf = tf;
|
switch (err_mask) {
|
||||||
err_mask = ata_exec_internal(dev, &rtf, NULL, DMA_NONE, NULL, 0, 0);
|
case 0:
|
||||||
|
level = KERN_DEBUG;
|
||||||
|
snprintf(msg, sizeof(msg), "succeeded");
|
||||||
|
rc = 1;
|
||||||
|
break;
|
||||||
|
|
||||||
switch (err_mask) {
|
case AC_ERR_DEV:
|
||||||
case 0:
|
level = KERN_INFO;
|
||||||
level = KERN_DEBUG;
|
snprintf(msg, sizeof(msg),
|
||||||
snprintf(msg, sizeof(msg), "succeeded");
|
"rejected by device (Stat=0x%02x Err=0x%02x)",
|
||||||
rc = 1;
|
rtf.command, rtf.feature);
|
||||||
break;
|
rc = 0;
|
||||||
|
break;
|
||||||
|
|
||||||
case AC_ERR_DEV:
|
default:
|
||||||
|
level = KERN_ERR;
|
||||||
|
snprintf(msg, sizeof(msg),
|
||||||
|
"failed (Emask=0x%x Stat=0x%02x Err=0x%02x)",
|
||||||
|
err_mask, rtf.command, rtf.feature);
|
||||||
|
rc = -EIO;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
level = KERN_INFO;
|
level = KERN_INFO;
|
||||||
snprintf(msg, sizeof(msg),
|
snprintf(msg, sizeof(msg), "filtered out");
|
||||||
"rejected by device (Stat=0x%02x Err=0x%02x)",
|
|
||||||
rtf.command, rtf.feature);
|
|
||||||
rc = 0;
|
rc = 0;
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
level = KERN_ERR;
|
|
||||||
snprintf(msg, sizeof(msg),
|
|
||||||
"failed (Emask=0x%x Stat=0x%02x Err=0x%02x)",
|
|
||||||
err_mask, rtf.command, rtf.feature);
|
|
||||||
rc = -EIO;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ata_dev_printk(dev, level,
|
ata_dev_printk(dev, level,
|
||||||
|
@ -566,7 +635,7 @@ static int ata_acpi_run_tf(struct ata_device *dev,
|
||||||
*/
|
*/
|
||||||
static int ata_acpi_exec_tfs(struct ata_device *dev, int *nr_executed)
|
static int ata_acpi_exec_tfs(struct ata_device *dev, int *nr_executed)
|
||||||
{
|
{
|
||||||
struct ata_acpi_gtf *gtf = NULL;
|
struct ata_acpi_gtf *gtf = NULL, *pgtf = NULL;
|
||||||
int gtf_count, i, rc;
|
int gtf_count, i, rc;
|
||||||
|
|
||||||
/* get taskfiles */
|
/* get taskfiles */
|
||||||
|
@ -576,12 +645,14 @@ static int ata_acpi_exec_tfs(struct ata_device *dev, int *nr_executed)
|
||||||
gtf_count = rc;
|
gtf_count = rc;
|
||||||
|
|
||||||
/* execute them */
|
/* execute them */
|
||||||
for (i = 0; i < gtf_count; i++) {
|
for (i = 0; i < gtf_count; i++, gtf++) {
|
||||||
rc = ata_acpi_run_tf(dev, gtf++);
|
rc = ata_acpi_run_tf(dev, gtf, pgtf);
|
||||||
if (rc < 0)
|
if (rc < 0)
|
||||||
break;
|
break;
|
||||||
if (rc)
|
if (rc) {
|
||||||
(*nr_executed)++;
|
(*nr_executed)++;
|
||||||
|
pgtf = gtf;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ata_acpi_clear_gtf(dev);
|
ata_acpi_clear_gtf(dev);
|
||||||
|
|
Loading…
Reference in New Issue