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);
static void ata_set_mode(struct ata_port *ap);
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 int fgb(unsigned long bitmap);
static int fgb(u32 bitmap);
static int ata_choose_xfer_mode(struct ata_port *ap,
unsigned int *xfer_mode_out,
u8 *xfer_mode_out,
unsigned int *xfer_shift_out);
static unsigned int ata_unique_id = 1;
......@@ -948,7 +948,8 @@ static void ata_dev_identify(struct ata_port *ap, unsigned int device)
{
struct ata_device *dev = &ap->device[device];
unsigned int i;
u16 tmp, xfer_modes;
u16 tmp;
unsigned long xfer_modes;
u8 status;
struct ata_taskfile tf;
unsigned int using_edd;
......@@ -1063,7 +1064,7 @@ static void ata_dev_identify(struct ata_port *ap, unsigned int device)
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];
if (!xfer_modes)
xfer_modes = (dev->id[ATA_ID_MWDMA_MODES]) << ATA_SHIFT_MWDMA;
......@@ -1253,14 +1254,14 @@ void ata_port_disable(struct ata_port *ap)
static struct {
unsigned int shift;
unsigned int base;
u8 base;
} xfer_mode_classes[] = {
{ ATA_SHIFT_UDMA, XFER_UDMA_0 },
{ ATA_SHIFT_MWDMA, XFER_MW_DMA_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;
......@@ -1268,12 +1269,13 @@ static inline unsigned int base_from_shift(unsigned int shift)
if (xfer_mode_classes[i].shift == shift)
return xfer_mode_classes[i].base;
return 0xffffffff;
return 0xff;
}
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))
return;
......@@ -1281,17 +1283,22 @@ static void ata_dev_set_mode(struct ata_port *ap, struct ata_device *dev)
ata_dev_set_xfermode(ap, dev);
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",
ap->id, dev->devno,
xfer_mode_str[offset + dev->xfer_shift]);
ap->id, dev->devno, xfer_mode_str[idx]);
}
static int ata_host_set_pio(struct ata_port *ap)
{
unsigned int mask, base, xfer_mode;
unsigned int mask;
int x, i;
u8 base, xfer_mode;
mask = ata_get_mode_mask(ap, ATA_SHIFT_PIO);
x = fgb(mask);
......@@ -1303,6 +1310,9 @@ static int ata_host_set_pio(struct ata_port *ap)
base = base_from_shift(ATA_SHIFT_PIO);
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++) {
struct ata_device *dev = &ap->device[i];
if (ata_dev_present(dev)) {
......@@ -1317,7 +1327,7 @@ static int ata_host_set_pio(struct ata_port *ap)
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)
{
int i;
......@@ -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)
{
unsigned int i, xfer_mode, xfer_shift;
unsigned int i, xfer_shift;
u8 xfer_mode;
int rc;
/* step 1: always set host PIO timings */
......@@ -1695,11 +1706,12 @@ static unsigned int ata_get_mode_mask(struct ata_port *ap, int shift)
}
/* 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))
x = i;
......@@ -1717,7 +1729,7 @@ static int fgb(unsigned long bitmap)
*/
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 mask, shift;
......@@ -2941,7 +2953,7 @@ int ata_device_add(struct ata_probe_ent *ent)
/* register each port bound to this device */
for (i = 0; i < ent->n_ports; i++) {
struct ata_port *ap;
unsigned int xfer_mode_mask;
unsigned long xfer_mode_mask;
ap = ata_host_add(ent, host_set, i);
if (!ap)
......
......@@ -258,9 +258,9 @@ struct ata_device {
unsigned int class; /* ATA_DEV_xxx */
unsigned int devno; /* 0 or 1 */
u16 id[ATA_ID_WORDS]; /* IDENTIFY xxx DEVICE data */
unsigned int pio_mode;
unsigned int dma_mode;
unsigned int xfer_mode;
u8 pio_mode;
u8 dma_mode;
u8 xfer_mode;
unsigned int xfer_shift; /* ATA_SHIFT_xxx */
/* 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