Commit 1d6fb7da authored by Jeff Garzik's avatar Jeff Garzik

[libata] transfer mode bug fixes and type cleanup

Fix two bugs that causes the recently-added transfer mode code
to break on 64-bit platforms.  Make associated code more type-correct
in the process.
parent 7a3e5b20
...@@ -53,9 +53,9 @@ static void __ata_dev_select (struct ata_port *ap, unsigned int device); ...@@ -53,9 +53,9 @@ static void __ata_dev_select (struct ata_port *ap, unsigned int device);
static void ata_set_mode(struct ata_port *ap); static void ata_set_mode(struct ata_port *ap);
static void ata_dev_set_xfermode(struct ata_port *ap, struct ata_device *dev); static void ata_dev_set_xfermode(struct ata_port *ap, struct ata_device *dev);
static unsigned int ata_get_mode_mask(struct ata_port *ap, int shift); static unsigned int ata_get_mode_mask(struct ata_port *ap, int shift);
static int fgb(unsigned long bitmap); static int fgb(u32 bitmap);
static int ata_choose_xfer_mode(struct ata_port *ap, static int ata_choose_xfer_mode(struct ata_port *ap,
unsigned int *xfer_mode_out, u8 *xfer_mode_out,
unsigned int *xfer_shift_out); unsigned int *xfer_shift_out);
static unsigned int ata_unique_id = 1; static unsigned int ata_unique_id = 1;
...@@ -948,7 +948,8 @@ static void ata_dev_identify(struct ata_port *ap, unsigned int device) ...@@ -948,7 +948,8 @@ static void ata_dev_identify(struct ata_port *ap, unsigned int device)
{ {
struct ata_device *dev = &ap->device[device]; struct ata_device *dev = &ap->device[device];
unsigned int i; unsigned int i;
u16 tmp, xfer_modes; u16 tmp;
unsigned long xfer_modes;
u8 status; u8 status;
struct ata_taskfile tf; struct ata_taskfile tf;
unsigned int using_edd; unsigned int using_edd;
...@@ -1063,7 +1064,7 @@ static void ata_dev_identify(struct ata_port *ap, unsigned int device) ...@@ -1063,7 +1064,7 @@ static void ata_dev_identify(struct ata_port *ap, unsigned int device)
goto err_out_nosup; goto err_out_nosup;
} }
/* we require UDMA support */ /* quick-n-dirty find max transfer mode; for printk only */
xfer_modes = dev->id[ATA_ID_UDMA_MODES]; xfer_modes = dev->id[ATA_ID_UDMA_MODES];
if (!xfer_modes) if (!xfer_modes)
xfer_modes = (dev->id[ATA_ID_MWDMA_MODES]) << ATA_SHIFT_MWDMA; xfer_modes = (dev->id[ATA_ID_MWDMA_MODES]) << ATA_SHIFT_MWDMA;
...@@ -1253,14 +1254,14 @@ void ata_port_disable(struct ata_port *ap) ...@@ -1253,14 +1254,14 @@ void ata_port_disable(struct ata_port *ap)
static struct { static struct {
unsigned int shift; unsigned int shift;
unsigned int base; u8 base;
} xfer_mode_classes[] = { } xfer_mode_classes[] = {
{ ATA_SHIFT_UDMA, XFER_UDMA_0 }, { ATA_SHIFT_UDMA, XFER_UDMA_0 },
{ ATA_SHIFT_MWDMA, XFER_MW_DMA_0 }, { ATA_SHIFT_MWDMA, XFER_MW_DMA_0 },
{ ATA_SHIFT_PIO, XFER_PIO_0 }, { ATA_SHIFT_PIO, XFER_PIO_0 },
}; };
static inline unsigned int base_from_shift(unsigned int shift) static inline u8 base_from_shift(unsigned int shift)
{ {
int i; int i;
...@@ -1268,12 +1269,13 @@ static inline unsigned int base_from_shift(unsigned int shift) ...@@ -1268,12 +1269,13 @@ static inline unsigned int base_from_shift(unsigned int shift)
if (xfer_mode_classes[i].shift == shift) if (xfer_mode_classes[i].shift == shift)
return xfer_mode_classes[i].base; return xfer_mode_classes[i].base;
return 0xffffffff; return 0xff;
} }
static void ata_dev_set_mode(struct ata_port *ap, struct ata_device *dev) static void ata_dev_set_mode(struct ata_port *ap, struct ata_device *dev)
{ {
unsigned int base, offset; int ofs, idx;
u8 base;
if (!ata_dev_present(dev) || (ap->flags & ATA_FLAG_PORT_DISABLED)) if (!ata_dev_present(dev) || (ap->flags & ATA_FLAG_PORT_DISABLED))
return; return;
...@@ -1281,17 +1283,22 @@ static void ata_dev_set_mode(struct ata_port *ap, struct ata_device *dev) ...@@ -1281,17 +1283,22 @@ static void ata_dev_set_mode(struct ata_port *ap, struct ata_device *dev)
ata_dev_set_xfermode(ap, dev); ata_dev_set_xfermode(ap, dev);
base = base_from_shift(dev->xfer_shift); base = base_from_shift(dev->xfer_shift);
offset = dev->xfer_mode - base; ofs = dev->xfer_mode - base;
idx = ofs + dev->xfer_shift;
WARN_ON(idx >= ARRAY_SIZE(xfer_mode_str));
DPRINTK("idx=%d xfer_shift=%u, xfer_mode=0x%x, base=0x%x, offset=%d\n",
idx, dev->xfer_shift, (int)dev->xfer_mode, (int)base, ofs);
printk(KERN_INFO "ata%u: dev %u configured for %s\n", printk(KERN_INFO "ata%u: dev %u configured for %s\n",
ap->id, dev->devno, ap->id, dev->devno, xfer_mode_str[idx]);
xfer_mode_str[offset + dev->xfer_shift]);
} }
static int ata_host_set_pio(struct ata_port *ap) static int ata_host_set_pio(struct ata_port *ap)
{ {
unsigned int mask, base, xfer_mode; unsigned int mask;
int x, i; int x, i;
u8 base, xfer_mode;
mask = ata_get_mode_mask(ap, ATA_SHIFT_PIO); mask = ata_get_mode_mask(ap, ATA_SHIFT_PIO);
x = fgb(mask); x = fgb(mask);
...@@ -1303,6 +1310,9 @@ static int ata_host_set_pio(struct ata_port *ap) ...@@ -1303,6 +1310,9 @@ static int ata_host_set_pio(struct ata_port *ap)
base = base_from_shift(ATA_SHIFT_PIO); base = base_from_shift(ATA_SHIFT_PIO);
xfer_mode = base + x; xfer_mode = base + x;
DPRINTK("base 0x%x xfer_mode 0x%x mask 0x%x x %d\n",
(int)base, (int)xfer_mode, mask, x);
for (i = 0; i < ATA_MAX_DEVICES; i++) { for (i = 0; i < ATA_MAX_DEVICES; i++) {
struct ata_device *dev = &ap->device[i]; struct ata_device *dev = &ap->device[i];
if (ata_dev_present(dev)) { if (ata_dev_present(dev)) {
...@@ -1317,7 +1327,7 @@ static int ata_host_set_pio(struct ata_port *ap) ...@@ -1317,7 +1327,7 @@ static int ata_host_set_pio(struct ata_port *ap)
return 0; return 0;
} }
static void ata_host_set_dma(struct ata_port *ap, unsigned int xfer_mode, static void ata_host_set_dma(struct ata_port *ap, u8 xfer_mode,
unsigned int xfer_shift) unsigned int xfer_shift)
{ {
int i; int i;
...@@ -1343,7 +1353,8 @@ static void ata_host_set_dma(struct ata_port *ap, unsigned int xfer_mode, ...@@ -1343,7 +1353,8 @@ static void ata_host_set_dma(struct ata_port *ap, unsigned int xfer_mode,
*/ */
static void ata_set_mode(struct ata_port *ap) static void ata_set_mode(struct ata_port *ap)
{ {
unsigned int i, xfer_mode, xfer_shift; unsigned int i, xfer_shift;
u8 xfer_mode;
int rc; int rc;
/* step 1: always set host PIO timings */ /* step 1: always set host PIO timings */
...@@ -1695,19 +1706,20 @@ static unsigned int ata_get_mode_mask(struct ata_port *ap, int shift) ...@@ -1695,19 +1706,20 @@ static unsigned int ata_get_mode_mask(struct ata_port *ap, int shift)
} }
/* find greatest bit */ /* find greatest bit */
static int fgb(unsigned long bitmap) static int fgb(u32 bitmap)
{ {
int i, x = -1; unsigned int i;
int x = -1;
for (i = 0; i < (sizeof(unsigned long) * 8); i++) for (i = 0; i < 32; i++)
if (bitmap & (1 << i)) if (bitmap & (1 << i))
x = i; x = i;
return x; return x;
} }
/** /**
* ata_choose_xfer_mode - * ata_choose_xfer_mode -
* @ap: * @ap:
* *
* LOCKING: * LOCKING:
...@@ -1717,7 +1729,7 @@ static int fgb(unsigned long bitmap) ...@@ -1717,7 +1729,7 @@ static int fgb(unsigned long bitmap)
*/ */
static int ata_choose_xfer_mode(struct ata_port *ap, static int ata_choose_xfer_mode(struct ata_port *ap,
unsigned int *xfer_mode_out, u8 *xfer_mode_out,
unsigned int *xfer_shift_out) unsigned int *xfer_shift_out)
{ {
unsigned int mask, shift; unsigned int mask, shift;
...@@ -2941,7 +2953,7 @@ int ata_device_add(struct ata_probe_ent *ent) ...@@ -2941,7 +2953,7 @@ int ata_device_add(struct ata_probe_ent *ent)
/* register each port bound to this device */ /* register each port bound to this device */
for (i = 0; i < ent->n_ports; i++) { for (i = 0; i < ent->n_ports; i++) {
struct ata_port *ap; struct ata_port *ap;
unsigned int xfer_mode_mask; unsigned long xfer_mode_mask;
ap = ata_host_add(ent, host_set, i); ap = ata_host_add(ent, host_set, i);
if (!ap) if (!ap)
......
...@@ -258,9 +258,9 @@ struct ata_device { ...@@ -258,9 +258,9 @@ struct ata_device {
unsigned int class; /* ATA_DEV_xxx */ unsigned int class; /* ATA_DEV_xxx */
unsigned int devno; /* 0 or 1 */ unsigned int devno; /* 0 or 1 */
u16 id[ATA_ID_WORDS]; /* IDENTIFY xxx DEVICE data */ u16 id[ATA_ID_WORDS]; /* IDENTIFY xxx DEVICE data */
unsigned int pio_mode; u8 pio_mode;
unsigned int dma_mode; u8 dma_mode;
unsigned int xfer_mode; u8 xfer_mode;
unsigned int xfer_shift; /* ATA_SHIFT_xxx */ unsigned int xfer_shift; /* ATA_SHIFT_xxx */
/* cache info about current transfer mode */ /* cache info about current transfer mode */
......
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