[IDE] pdc202xx_old.c: sanitize 66MHz clock use Sanitize 66MHz clock use: "enable" 66MHz clock before starting UDMA3/4/5 read/write transfer and "disable" it after finishing transfer. - fixes timings for non-UDMA3/4/5 operations (correct 33MHz timings are used) - allows using UDMA3/4/5 modes on a capable drive even if non-UDMA3/4/5 drive is present on the same channel - fixes corner case when one drive on the channel was using UDMA66/100 + LBA48 (so clock was enabled/disabled for each read/write) and other one was using UDMA66/100 + LBA28, it could happen that request on LBA48 drive disabled 66MHz clock and it was not enabled for the next transfer on LBA28 drive drivers/ide/pci/pdc202xx_old.c | 66 ++++++++++++++++++----------------------- 1 files changed, 30 insertions(+), 36 deletions(-) diff -puN drivers/ide/pci/pdc202xx_old.c~ide-pdc_old-66mhz_clock-fix drivers/ide/pci/pdc202xx_old.c --- linux-2.6.0-test11/drivers/ide/pci/pdc202xx_old.c~ide-pdc_old-66mhz_clock-fix 2003-12-07 20:14:26.105438168 +0100 +++ linux-2.6.0-test11-root/drivers/ide/pci/pdc202xx_old.c 2003-12-07 20:17:26.124071176 +0100 @@ -361,16 +361,38 @@ static u8 pdc202xx_old_cable_detect (ide return ((u8)(CIS & mask)); } +/* + * Set the control register to use the 66MHz system + * clock for UDMA 3/4/5 mode operation when necessary. + * + * It may also be possible to leave the 66MHz clock on + * and readjust the timing parameters. + */ +static void pdc_old_enable_66MHz_clock(ide_hwif_t *hwif) +{ + unsigned long clock_reg = hwif->dma_master + 0x11; + u8 clock = hwif->INB(clock_reg); + + hwif->OUTB(clock | (hwif->channel ? 0x08 : 0x02), clock_reg); +} + +static void pdc_old_disable_66MHz_clock(ide_hwif_t *hwif) +{ + unsigned long clock_reg = hwif->dma_master + 0x11; + u8 clock = hwif->INB(clock_reg); + + hwif->OUTB(clock & ~(hwif->channel ? 0x08 : 0x02), clock_reg); +} + static int config_chipset_for_dma (ide_drive_t *drive) { struct hd_driveid *id = drive->id; ide_hwif_t *hwif = HWIF(drive); struct pci_dev *dev = hwif->pci_dev; u32 drive_conf = 0; - u8 mask = hwif->channel ? 0x08 : 0x02; u8 drive_pci = 0x60 + (drive->dn << 2); u8 test1 = 0, test2 = 0, speed = -1; - u8 AP = 0, CLKSPD = 0, cable = 0; + u8 AP = 0, cable = 0; u8 ultra_66 = ((id->dma_ultra & 0x0010) || (id->dma_ultra & 0x0008)) ? 1 : 0; @@ -394,21 +416,6 @@ static int config_chipset_for_dma (ide_d BUG(); } - CLKSPD = hwif->INB(hwif->dma_master + 0x11); - - /* - * Set the control register to use the 66Mhz system - * clock for UDMA 3/4 mode operation. If one drive on - * a channel is U66 capable but the other isn't we - * fall back to U33 mode. The BIOS INT 13 hooks turn - * the clock on then off for each read/write issued. I don't - * do that here because it would require modifying the - * kernel, separating the fop routines from the kernel or - * somehow hooking the fops calls. It may also be possible to - * leave the 66Mhz clock on and readjust the timing - * parameters. - */ - if ((ultra_66) && (cable)) { #ifdef DEBUG printk(KERN_DEBUG "ULTRA 66/100/133: %s channel of Ultra 66/100/133 " @@ -416,29 +423,12 @@ static int config_chipset_for_dma (ide_d hwif->channel ? "Secondary" : "Primary"); printk(KERN_DEBUG " Switching to Ultra33 mode.\n"); #endif /* DEBUG */ - /* Primary : zero out second bit */ - /* Secondary : zero out fourth bit */ - hwif->OUTB(CLKSPD & ~mask, (hwif->dma_master + 0x11)); printk(KERN_WARNING "Warning: %s channel requires an 80-pin cable for operation.\n", hwif->channel ? "Secondary":"Primary"); printk(KERN_WARNING "%s reduced to Ultra33 mode.\n", drive->name); - } else { - if (ultra_66) { - /* - * Check to make sure drive on the same channel - * is UDMA3 or higher capable. Ignore empty slots. - */ - if (hwif->drives[!(drive->dn%2)].present) { - if (hwif->drives[!(drive->dn%2)].id->dma_ultra & 0x0078) { - hwif->OUTB(CLKSPD | mask, (hwif->dma_master + 0x11)); - } else { - hwif->OUTB(CLKSPD & ~mask, (hwif->dma_master + 0x11)); - } - } else { /* udma4 drive by itself */ - hwif->OUTB(CLKSPD | mask, (hwif->dma_master + 0x11)); - } - } } + pdc_old_disable_66MHz_clock(drive->hwif); + drive_pci = 0x60 + (drive->dn << 2); pci_read_config_dword(dev, drive_pci, &drive_conf); if ((drive_conf != 0x004ff304) && (drive_conf != 0x004ff3c4)) @@ -536,6 +526,8 @@ static int pdc202xx_quirkproc (ide_drive static int pdc202xx_old_ide_dma_begin(ide_drive_t *drive) { + if (drive->current_speed > XFER_UDMA_2) + pdc_old_enable_66MHz_clock(drive->hwif); if (drive->addressing == 1) { struct request *rq = HWGROUP(drive)->rq; ide_hwif_t *hwif = HWIF(drive); @@ -569,6 +561,8 @@ static int pdc202xx_old_ide_dma_end(ide_ clock = hwif->INB(high_16 + 0x11); hwif->OUTB(clock & ~(hwif->channel ? 0x08:0x02), high_16+0x11); } + if (drive->current_speed > XFER_UDMA_2) + pdc_old_disable_66MHz_clock(drive->hwif); return __ide_dma_end(drive); } _