Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit c05e6ff0 authored by Tejun Heo's avatar Tejun Heo Committed by Jeff Garzik
Browse files

libata-acpi: implement and use ata_acpi_init_gtm()



_GTM fetches currently configured transfer mode while _STM configures
controller according to _GTM parameter and prepares transfer mode
configuration TFs for _GTF.  In many cases _GTM and _STM
implementations are quite brittle and can't cope with configuration
changed by libata.

libata does not depend on ATA ACPI to configure devices.  The only
reason libata performs _GTM and _STM are to make _GTF evaluation
succeed and libata also doesn't care about how _GTF TFs configure
transfer mode.  It overrides that configuration anyway, so from
libata's POV, it doesn't matter what value is feeded to _STM as long
as evaluation succeeds for _STM and following _GTF.

This patch adds dev->__acpi_init_gtm and store initial _GTM values on
host initialization before modified by reset and mode configuration.
If the field is valid, ata_acpi_init_gtm() returns pointer to the
saved _GTM structure; otherwise, NULL.

This saved value is used for _STM during resume and peek at
BIOS/firmware programmed initial timing for later use.  The accessor
is there to make building w/o ACPI easy as dev->__acpi_init doesn't
exist if ACPI is not enabled.

On driver detach, the initial BIOS configuration is restored by
executing _STM with the initial _GTM values such that the next driver
can also use the initial BIOS configured values.

Signed-off-by: default avatarTejun Heo <htejun@gmail.com>
Signed-off-by: default avatarJeff Garzik <jeff@garzik.org>
parent 562f0c2d
Loading
Loading
Loading
Loading
+27 −35
Original line number Original line Diff line number Diff line
@@ -94,6 +94,9 @@ static void ata_acpi_associate_ide_port(struct ata_port *ap)


		dev->acpi_handle = acpi_get_child(ap->acpi_handle, i);
		dev->acpi_handle = acpi_get_child(ap->acpi_handle, i);
	}
	}

	if (ata_acpi_gtm(ap, &ap->__acpi_init_gtm) == 0)
		ap->pflags |= ATA_PFLAG_INIT_GTM_VALID;
}
}


static void ata_acpi_handle_hotplug(struct ata_port *ap, struct kobject *kobj,
static void ata_acpi_handle_hotplug(struct ata_port *ap, struct kobject *kobj,
@@ -199,7 +202,18 @@ void ata_acpi_associate(struct ata_host *host)
 */
 */
void ata_acpi_dissociate(struct ata_host *host)
void ata_acpi_dissociate(struct ata_host *host)
{
{
	/* nada */
	int i;

	/* Restore initial _GTM values so that driver which attaches
	 * afterward can use them too.
	 */
	for (i = 0; i < host->n_ports; i++) {
		struct ata_port *ap = host->ports[i];
		const struct ata_acpi_gtm *gtm = ata_acpi_init_gtm(ap);

		if (ap->acpi_handle && gtm)
			ata_acpi_stm(ap, gtm);
	}
}
}


/**
/**
@@ -409,22 +423,21 @@ static int ata_dev_get_GTF(struct ata_device *dev, struct ata_acpi_gtf **gtf,


int ata_acpi_cbl_80wire(struct ata_port *ap)
int ata_acpi_cbl_80wire(struct ata_port *ap)
{
{
	struct ata_acpi_gtm gtm;
	const struct ata_acpi_gtm *gtm = ata_acpi_init_gtm(ap);
	int valid = 0;
	int valid = 0;


	/* No _GTM data, no information */
	if (!gtm)
	if (ata_acpi_gtm(ap, &gtm) < 0)
		return 0;
		return 0;


	/* Split timing, DMA enabled */
	/* Split timing, DMA enabled */
	if ((gtm.flags & 0x11) == 0x11 && gtm.drive[0].dma < 55)
	if ((gtm->flags & 0x11) == 0x11 && gtm->drive[0].dma < 55)
		valid |= 1;
		valid |= 1;
	if ((gtm.flags & 0x14) == 0x14 && gtm.drive[1].dma < 55)
	if ((gtm->flags & 0x14) == 0x14 && gtm->drive[1].dma < 55)
		valid |= 2;
		valid |= 2;
	/* Shared timing, DMA enabled */
	/* Shared timing, DMA enabled */
	if ((gtm.flags & 0x11) == 0x01 && gtm.drive[0].dma < 55)
	if ((gtm->flags & 0x11) == 0x01 && gtm->drive[0].dma < 55)
		valid |= 1;
		valid |= 1;
	if ((gtm.flags & 0x14) == 0x04 && gtm.drive[0].dma < 55)
	if ((gtm->flags & 0x14) == 0x04 && gtm->drive[0].dma < 55)
		valid |= 2;
		valid |= 2;


	/* Drive check */
	/* Drive check */
@@ -612,27 +625,8 @@ static int ata_acpi_push_id(struct ata_device *dev)
 */
 */
int ata_acpi_on_suspend(struct ata_port *ap)
int ata_acpi_on_suspend(struct ata_port *ap)
{
{
	unsigned long flags;
	/* nada */
	int rc;

	/* proceed iff per-port acpi_handle is valid */
	if (!ap->acpi_handle)
	return 0;
	return 0;
	BUG_ON(ap->flags & ATA_FLAG_ACPI_SATA);

	/* store timing parameters */
	rc = ata_acpi_gtm(ap, &ap->acpi_gtm);

	spin_lock_irqsave(ap->lock, flags);
	if (rc == 0)
		ap->pflags |= ATA_PFLAG_GTM_VALID;
	else
		ap->pflags &= ~ATA_PFLAG_GTM_VALID;
	spin_unlock_irqrestore(ap->lock, flags);

	if (rc == -ENOENT)
		rc = 0;
	return rc;
}
}


/**
/**
@@ -647,14 +641,12 @@ int ata_acpi_on_suspend(struct ata_port *ap)
 */
 */
void ata_acpi_on_resume(struct ata_port *ap)
void ata_acpi_on_resume(struct ata_port *ap)
{
{
	const struct ata_acpi_gtm *gtm = ata_acpi_init_gtm(ap);
	struct ata_device *dev;
	struct ata_device *dev;


	if (ap->acpi_handle && (ap->pflags & ATA_PFLAG_GTM_VALID)) {
		BUG_ON(ap->flags & ATA_FLAG_ACPI_SATA);

	/* restore timing parameters */
	/* restore timing parameters */
		ata_acpi_stm(ap, &ap->acpi_gtm);
	if (ap->acpi_handle && gtm)
	}
		ata_acpi_stm(ap, gtm);


	/* schedule _GTF */
	/* schedule _GTF */
	ata_link_for_each_dev(dev, &ap->link)
	ata_link_for_each_dev(dev, &ap->link)
+12 −2
Original line number Original line Diff line number Diff line
@@ -211,7 +211,7 @@ enum {


	ATA_PFLAG_SUSPENDED	= (1 << 17), /* port is suspended (power) */
	ATA_PFLAG_SUSPENDED	= (1 << 17), /* port is suspended (power) */
	ATA_PFLAG_PM_PENDING	= (1 << 18), /* PM operation pending */
	ATA_PFLAG_PM_PENDING	= (1 << 18), /* PM operation pending */
	ATA_PFLAG_GTM_VALID	= (1 << 19), /* acpi_gtm data valid */
	ATA_PFLAG_INIT_GTM_VALID = (1 << 19), /* initial gtm data valid */


	/* struct ata_queued_cmd flags */
	/* struct ata_queued_cmd flags */
	ATA_QCFLAG_ACTIVE	= (1 << 0), /* cmd not yet ack'd to scsi lyer */
	ATA_QCFLAG_ACTIVE	= (1 << 0), /* cmd not yet ack'd to scsi lyer */
@@ -653,7 +653,7 @@ struct ata_port {


#ifdef CONFIG_ATA_ACPI
#ifdef CONFIG_ATA_ACPI
	acpi_handle		acpi_handle;
	acpi_handle		acpi_handle;
	struct ata_acpi_gtm	acpi_gtm;
	struct ata_acpi_gtm	__acpi_init_gtm; /* use ata_acpi_init_gtm() */
#endif
#endif
	u8			sector_buf[ATA_SECT_SIZE]; /* owned by EH */
	u8			sector_buf[ATA_SECT_SIZE]; /* owned by EH */
};
};
@@ -939,10 +939,20 @@ enum {


/* libata-acpi.c */
/* libata-acpi.c */
#ifdef CONFIG_ATA_ACPI
#ifdef CONFIG_ATA_ACPI
static inline const struct ata_acpi_gtm *ata_acpi_init_gtm(struct ata_port *ap)
{
	if (ap->pflags & ATA_PFLAG_INIT_GTM_VALID)
		return &ap->__acpi_init_gtm;
	return NULL;
}
extern int ata_acpi_cbl_80wire(struct ata_port *ap);
extern int ata_acpi_cbl_80wire(struct ata_port *ap);
int ata_acpi_stm(struct ata_port *ap, const struct ata_acpi_gtm *stm);
int ata_acpi_stm(struct ata_port *ap, const struct ata_acpi_gtm *stm);
int ata_acpi_gtm(struct ata_port *ap, struct ata_acpi_gtm *stm);
int ata_acpi_gtm(struct ata_port *ap, struct ata_acpi_gtm *stm);
#else
#else
static inline const struct ata_acpi_gtm *ata_acpi_init_gtm(struct ata_port *ap)
{
	return NULL;
}
static inline int ata_acpi_cbl_80wire(struct ata_port *ap) { return 0; }
static inline int ata_acpi_cbl_80wire(struct ata_port *ap) { return 0; }
#endif
#endif