📄 it821x.c
字号:
struct it821x_dev *itdev = ide_get_hwifdata(hwif); int unit = drive->select.b.unit; if(itdev->mwdma[unit] != MWDMA_OFF) it821x_program(drive, itdev->mwdma[unit]); else if(itdev->udma[unit] != UDMA_OFF && itdev->timing10) it821x_program_udma(drive, itdev->udma[unit]); ide_dma_start(drive);}/** * it821x_dma_write - DMA hook * @drive: drive for DMA stop * * The IT821x has a single timing register for MWDMA and for PIO * operations. As we flip back and forth we have to reload the * clock. */static int it821x_dma_end(ide_drive_t *drive){ ide_hwif_t *hwif = drive->hwif; int unit = drive->select.b.unit; struct it821x_dev *itdev = ide_get_hwifdata(hwif); int ret = __ide_dma_end(drive); if(itdev->mwdma[unit] != MWDMA_OFF) it821x_program(drive, itdev->pio[unit]); return ret;}/** * it821x_tune_chipset - set controller timings * @drive: Drive to set up * @xferspeed: speed we want to achieve * * Tune the ITE chipset for the desired mode. If we can't achieve * the desired mode then tune for a lower one, but ultimately * make the thing work. */static int it821x_tune_chipset (ide_drive_t *drive, byte xferspeed){ ide_hwif_t *hwif = drive->hwif; struct it821x_dev *itdev = ide_get_hwifdata(hwif); u8 speed = ide_rate_filter(it821x_ratemask(drive), xferspeed); if(!itdev->smart) { switch(speed) { case XFER_PIO_4: case XFER_PIO_3: case XFER_PIO_2: case XFER_PIO_1: case XFER_PIO_0: it821x_tuneproc(drive, (speed - XFER_PIO_0)); break; /* MWDMA tuning is really hard because our MWDMA and PIO timings are kept in the same place. We can switch in the host dma on/off callbacks */ case XFER_MW_DMA_2: case XFER_MW_DMA_1: case XFER_MW_DMA_0: it821x_tune_mwdma(drive, (speed - XFER_MW_DMA_0)); break; case XFER_UDMA_6: case XFER_UDMA_5: case XFER_UDMA_4: case XFER_UDMA_3: case XFER_UDMA_2: case XFER_UDMA_1: case XFER_UDMA_0: it821x_tune_udma(drive, (speed - XFER_UDMA_0)); break; default: return 1; } } /* * In smart mode the clocking is done by the host controller * snooping the mode we picked. The rest of it is not our problem */ return ide_config_drive_speed(drive, speed);}/** * config_chipset_for_dma - configure for DMA * @drive: drive to configure * * Called by the IDE layer when it wants the timings set up. */static int config_chipset_for_dma (ide_drive_t *drive){ u8 speed = ide_dma_speed(drive, it821x_ratemask(drive)); config_it821x_chipset_for_pio(drive, !speed); it821x_tune_chipset(drive, speed); return ide_dma_enable(drive);}/** * it821x_configure_drive_for_dma - set up for DMA transfers * @drive: drive we are going to set up * * Set up the drive for DMA, tune the controller and drive as * required. If the drive isn't suitable for DMA or we hit * other problems then we will drop down to PIO and set up * PIO appropriately */static int it821x_config_drive_for_dma (ide_drive_t *drive){ ide_hwif_t *hwif = drive->hwif; if (ide_use_dma(drive)) { if (config_chipset_for_dma(drive)) return hwif->ide_dma_on(drive); } config_it821x_chipset_for_pio(drive, 1); return hwif->ide_dma_off_quietly(drive);}/** * ata66_it821x - check for 80 pin cable * @hwif: interface to check * * Check for the presence of an ATA66 capable cable on the * interface. Problematic as it seems some cards don't have * the needed logic onboard. */static unsigned int __devinit ata66_it821x(ide_hwif_t *hwif){ /* The reference driver also only does disk side */ return 1;}/** * it821x_fixup - post init callback * @hwif: interface * * This callback is run after the drives have been probed but * before anything gets attached. It allows drivers to do any * final tuning that is needed, or fixups to work around bugs. */static void __devinit it821x_fixups(ide_hwif_t *hwif){ struct it821x_dev *itdev = ide_get_hwifdata(hwif); int i; if(!itdev->smart) { /* * If we are in pass through mode then not much * needs to be done, but we do bother to clear the * IRQ mask as we may well be in PIO (eg rev 0x10) * for now and we know unmasking is safe on this chipset. */ for (i = 0; i < 2; i++) { ide_drive_t *drive = &hwif->drives[i]; if(drive->present) drive->unmask = 1; } return; } /* * Perform fixups on smart mode. We need to "lose" some * capabilities the firmware lacks but does not filter, and * also patch up some capability bits that it forgets to set * in RAID mode. */ for(i = 0; i < 2; i++) { ide_drive_t *drive = &hwif->drives[i]; struct hd_driveid *id; u16 *idbits; if(!drive->present) continue; id = drive->id; idbits = (u16 *)drive->id; /* Check for RAID v native */ if(strstr(id->model, "Integrated Technology Express")) { /* In raid mode the ident block is slightly buggy We need to set the bits so that the IDE layer knows LBA28. LBA48 and DMA ar valid */ id->capability |= 3; /* LBA28, DMA */ id->command_set_2 |= 0x0400; /* LBA48 valid */ id->cfs_enable_2 |= 0x0400; /* LBA48 on */ /* Reporting logic */ printk(KERN_INFO "%s: IT8212 %sRAID %d volume", drive->name, idbits[147] ? "Bootable ":"", idbits[129]); if(idbits[129] != 1) printk("(%dK stripe)", idbits[146]); printk(".\n"); /* Now the core code will have wrongly decided no DMA so we need to fix this */ hwif->ide_dma_off_quietly(drive);#ifdef CONFIG_IDEDMA_ONLYDISK if (drive->media == ide_disk)#endif hwif->ide_dma_check(drive); } else { /* Non RAID volume. Fixups to stop the core code doing unsupported things */ id->field_valid &= 1; id->queue_depth = 0; id->command_set_1 = 0; id->command_set_2 &= 0xC400; id->cfsse &= 0xC000; id->cfs_enable_1 = 0; id->cfs_enable_2 &= 0xC400; id->csf_default &= 0xC000; id->word127 = 0; id->dlf = 0; id->csfo = 0; id->cfa_power = 0; printk(KERN_INFO "%s: Performing identify fixups.\n", drive->name); } }}/** * init_hwif_it821x - set up hwif structs * @hwif: interface to set up * * We do the basic set up of the interface structure. The IT8212 * requires several custom handlers so we override the default * ide DMA handlers appropriately */static void __devinit init_hwif_it821x(ide_hwif_t *hwif){ struct it821x_dev *idev = kzalloc(sizeof(struct it821x_dev), GFP_KERNEL); u8 conf; if(idev == NULL) { printk(KERN_ERR "it821x: out of memory, falling back to legacy behaviour.\n"); goto fallback; } ide_set_hwifdata(hwif, idev); pci_read_config_byte(hwif->pci_dev, 0x50, &conf); if(conf & 1) { idev->smart = 1; hwif->atapi_dma = 0; /* Long I/O's although allowed in LBA48 space cause the onboard firmware to enter the twighlight zone */ hwif->rqsize = 256; } /* Pull the current clocks from 0x50 also */ if (conf & (1 << (1 + hwif->channel))) idev->clock_mode = ATA_50; else idev->clock_mode = ATA_66; idev->want[0][1] = ATA_ANY; idev->want[1][1] = ATA_ANY; /* * Not in the docs but according to the reference driver * this is neccessary. */ pci_read_config_byte(hwif->pci_dev, 0x08, &conf); if(conf == 0x10) { idev->timing10 = 1; hwif->atapi_dma = 0; if(!idev->smart) printk(KERN_WARNING "it821x: Revision 0x10, workarounds activated.\n"); } hwif->speedproc = &it821x_tune_chipset; hwif->tuneproc = &it821x_tuneproc; /* MWDMA/PIO clock switching for pass through mode */ if(!idev->smart) { hwif->dma_start = &it821x_dma_start; hwif->ide_dma_end = &it821x_dma_end; } hwif->drives[0].autotune = 1; hwif->drives[1].autotune = 1; if (!hwif->dma_base) goto fallback; hwif->ultra_mask = 0x7f; hwif->mwdma_mask = 0x07; hwif->swdma_mask = 0x07; hwif->ide_dma_check = &it821x_config_drive_for_dma; if (!(hwif->udma_four)) hwif->udma_four = ata66_it821x(hwif); /* * The BIOS often doesn't set up DMA on this controller * so we always do it. */ hwif->autodma = 1; hwif->drives[0].autodma = hwif->autodma; hwif->drives[1].autodma = hwif->autodma; return;fallback: hwif->autodma = 0; return;}static void __devinit it8212_disable_raid(struct pci_dev *dev){ /* Reset local CPU, and set BIOS not ready */ pci_write_config_byte(dev, 0x5E, 0x01); /* Set to bypass mode, and reset PCI bus */ pci_write_config_byte(dev, 0x50, 0x00); pci_write_config_word(dev, PCI_COMMAND, PCI_COMMAND_PARITY | PCI_COMMAND_IO | PCI_COMMAND_MEMORY | PCI_COMMAND_MASTER); pci_write_config_word(dev, 0x40, 0xA0F3); pci_write_config_dword(dev,0x4C, 0x02040204); pci_write_config_byte(dev, 0x42, 0x36); pci_write_config_byte(dev, PCI_LATENCY_TIMER, 0);}static unsigned int __devinit init_chipset_it821x(struct pci_dev *dev, const char *name){ u8 conf; static char *mode[2] = { "pass through", "smart" }; /* Force the card into bypass mode if so requested */ if (it8212_noraid) { printk(KERN_INFO "it8212: forcing bypass mode.\n"); it8212_disable_raid(dev); } pci_read_config_byte(dev, 0x50, &conf); printk(KERN_INFO "it821x: controller in %s mode.\n", mode[conf & 1]); return 0;}#define DECLARE_ITE_DEV(name_str) \ { \ .name = name_str, \ .init_chipset = init_chipset_it821x, \ .init_hwif = init_hwif_it821x, \ .channels = 2, \ .autodma = AUTODMA, \ .bootable = ON_BOARD, \ .fixup = it821x_fixups \ }static ide_pci_device_t it821x_chipsets[] __devinitdata = { /* 0 */ DECLARE_ITE_DEV("IT8212"),};/** * it821x_init_one - pci layer discovery entry * @dev: PCI device * @id: ident table entry * * Called by the PCI code when it finds an ITE821x controller. * We then use the IDE PCI generic helper to do most of the work. */static int __devinit it821x_init_one(struct pci_dev *dev, const struct pci_device_id *id){ ide_setup_pci_device(dev, &it821x_chipsets[id->driver_data]); return 0;}static struct pci_device_id it821x_pci_tbl[] = { { PCI_VENDOR_ID_ITE, PCI_DEVICE_ID_ITE_8211, PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0}, { PCI_VENDOR_ID_ITE, PCI_DEVICE_ID_ITE_8212, PCI_ANY_ID, PCI_ANY_ID, 0, 0, 0}, { 0, },};MODULE_DEVICE_TABLE(pci, it821x_pci_tbl);static struct pci_driver driver = { .name = "ITE821x IDE", .id_table = it821x_pci_tbl, .probe = it821x_init_one,};static int __init it821x_ide_init(void){ return ide_pci_register_driver(&driver);}module_init(it821x_ide_init);module_param_named(noraid, it8212_noraid, int, S_IRUGO);MODULE_PARM_DESC(it8212_noraid, "Force card into bypass mode");MODULE_AUTHOR("Alan Cox");MODULE_DESCRIPTION("PCI driver module for the ITE 821x");MODULE_LICENSE("GPL");
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -