aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/ata/libata-core.c
diff options
context:
space:
mode:
authorTejun Heo <htejun@gmail.com>2007-10-30 21:17:07 -0400
committerJeff Garzik <jeff@garzik.org>2007-11-03 08:47:27 -0400
commit5270222f96608818e431b5c4029b1f12020ab719 (patch)
tree564f141932bc4740ebf3ea0a17f2a03427913a8c /drivers/ata/libata-core.c
parentdb64bcf387aae6c7afad122a529d7d0513d3c5db (diff)
libata: don't configure downstream links faster than the upstream link
There's nothing to be gained by configuring downstream links faster than the upstream link and such configurations cause problems on certain PMPs. Limit downstream link speed by the upstream link speed. Signed-off-by: Tejun Heo <htejun@gmail.com> Signed-off-by: Jeff Garzik <jeff@garzik.org>
Diffstat (limited to 'drivers/ata/libata-core.c')
-rw-r--r--drivers/ata/libata-core.c22
1 files changed, 16 insertions, 6 deletions
diff --git a/drivers/ata/libata-core.c b/drivers/ata/libata-core.c
index 3a1ec4e715ed..164c7d9514f9 100644
--- a/drivers/ata/libata-core.c
+++ b/drivers/ata/libata-core.c
@@ -2751,17 +2751,27 @@ int sata_down_spd_limit(struct ata_link *link)
2751 2751
2752static int __sata_set_spd_needed(struct ata_link *link, u32 *scontrol) 2752static int __sata_set_spd_needed(struct ata_link *link, u32 *scontrol)
2753{ 2753{
2754 u32 spd, limit; 2754 struct ata_link *host_link = &link->ap->link;
2755 u32 limit, target, spd;
2755 2756
2756 if (link->sata_spd_limit == UINT_MAX) 2757 limit = link->sata_spd_limit;
2757 limit = 0; 2758
2759 /* Don't configure downstream link faster than upstream link.
2760 * It doesn't speed up anything and some PMPs choke on such
2761 * configuration.
2762 */
2763 if (!ata_is_host_link(link) && host_link->sata_spd)
2764 limit &= (1 << host_link->sata_spd) - 1;
2765
2766 if (limit == UINT_MAX)
2767 target = 0;
2758 else 2768 else
2759 limit = fls(link->sata_spd_limit); 2769 target = fls(limit);
2760 2770
2761 spd = (*scontrol >> 4) & 0xf; 2771 spd = (*scontrol >> 4) & 0xf;
2762 *scontrol = (*scontrol & ~0xf0) | ((limit & 0xf) << 4); 2772 *scontrol = (*scontrol & ~0xf0) | ((target & 0xf) << 4);
2763 2773
2764 return spd != limit; 2774 return spd != target;
2765} 2775}
2766 2776
2767/** 2777/**