siimage.c

来自「linux 内核源代码」· C语言 代码 · 共 911 行 · 第 1/2 页

C
911
字号
/* * linux/drivers/ide/pci/siimage.c		Version 1.19	Nov 16 2007 * * Copyright (C) 2001-2002	Andre Hedrick <andre@linux-ide.org> * Copyright (C) 2003		Red Hat <alan@redhat.com> * Copyright (C) 2007		MontaVista Software, Inc. * Copyright (C) 2007		Bartlomiej Zolnierkiewicz * *  May be copied or modified under the terms of the GNU General Public License * *  Documentation for CMD680: *  http://gkernel.sourceforge.net/specs/sii/sii-0680a-v1.31.pdf.bz2 * *  Documentation for SiI 3112: *  http://gkernel.sourceforge.net/specs/sii/3112A_SiI-DS-0095-B2.pdf.bz2 * *  Errata and other documentation only available under NDA. * * *  FAQ Items: *	If you are using Marvell SATA-IDE adapters with Maxtor drives *	ensure the system is set up for ATA100/UDMA5 not UDMA6. * *	If you are using WD drives with SATA bridges you must set the *	drive to "Single". "Master" will hang * *	If you have strange problems with nVidia chipset systems please *	see the SI support documentation and update your system BIOS *	if necessary * *  The Dell DRAC4 has some interesting features including effectively hot *  unplugging/replugging the virtual CD interface when the DRAC is reset. *  This often causes drivers/ide/siimage to panic but is ok with the rather *  smarter code in libata. * * TODO: * - IORDY fixes * - VDMA support */#include <linux/types.h>#include <linux/module.h>#include <linux/pci.h>#include <linux/delay.h>#include <linux/hdreg.h>#include <linux/ide.h>#include <linux/init.h>#include <asm/io.h>/** *	pdev_is_sata		-	check if device is SATA *	@pdev:	PCI device to check *	 *	Returns true if this is a SATA controller */ static int pdev_is_sata(struct pci_dev *pdev){#ifdef CONFIG_BLK_DEV_IDE_SATA	switch(pdev->device) {		case PCI_DEVICE_ID_SII_3112:		case PCI_DEVICE_ID_SII_1210SA:			return 1;		case PCI_DEVICE_ID_SII_680:			return 0;	}	BUG();#endif	return 0;}/** *	is_sata			-	check if hwif is SATA *	@hwif:	interface to check *	 *	Returns true if this is a SATA controller */ static inline int is_sata(ide_hwif_t *hwif){	return pdev_is_sata(hwif->pci_dev);}/** *	siimage_selreg		-	return register base *	@hwif: interface *	@r: config offset * *	Turn a config register offset into the right address in either *	PCI space or MMIO space to access the control register in question *	Thankfully this is a configuration operation so isnt performance *	criticial.  */ static unsigned long siimage_selreg(ide_hwif_t *hwif, int r){	unsigned long base = (unsigned long)hwif->hwif_data;	base += 0xA0 + r;	if(hwif->mmio)		base += (hwif->channel << 6);	else		base += (hwif->channel << 4);	return base;}	/** *	siimage_seldev		-	return register base *	@hwif: interface *	@r: config offset * *	Turn a config register offset into the right address in either *	PCI space or MMIO space to access the control register in question *	including accounting for the unit shift. */ static inline unsigned long siimage_seldev(ide_drive_t *drive, int r){	ide_hwif_t *hwif	= HWIF(drive);	unsigned long base = (unsigned long)hwif->hwif_data;	base += 0xA0 + r;	if(hwif->mmio)		base += (hwif->channel << 6);	else		base += (hwif->channel << 4);	base |= drive->select.b.unit << drive->select.b.unit;	return base;}/** *	sil_udma_filter		-	compute UDMA mask *	@drive: IDE device * *	Compute the available UDMA speeds for the device on the interface. * *	For the CMD680 this depends on the clocking mode (scsc), for the *	SI3112 SATA controller life is a bit simpler. */static u8 sil_pata_udma_filter(ide_drive_t *drive){	ide_hwif_t *hwif = drive->hwif;	unsigned long base = (unsigned long) hwif->hwif_data;	u8 mask = 0, scsc = 0;	if (hwif->mmio)		scsc = hwif->INB(base + 0x4A);	else		pci_read_config_byte(hwif->pci_dev, 0x8A, &scsc);	if ((scsc & 0x30) == 0x10)	/* 133 */		mask = ATA_UDMA6;	else if ((scsc & 0x30) == 0x20)	/* 2xPCI */		mask = ATA_UDMA6;	else if ((scsc & 0x30) == 0x00)	/* 100 */		mask = ATA_UDMA5;	else 	/* Disabled ? */		BUG();	return mask;}static u8 sil_sata_udma_filter(ide_drive_t *drive){	return strstr(drive->id->model, "Maxtor") ? ATA_UDMA5 : ATA_UDMA6;}/** *	sil_set_pio_mode	-	set host controller for PIO mode *	@drive: drive *	@pio: PIO mode number * *	Load the timing settings for this device mode into the *	controller. If we are in PIO mode 3 or 4 turn on IORDY *	monitoring (bit 9). The TF timing is bits 31:16 */static void sil_set_pio_mode(ide_drive_t *drive, u8 pio){	const u16 tf_speed[]	= { 0x328a, 0x2283, 0x1281, 0x10c3, 0x10c1 };	const u16 data_speed[]	= { 0x328a, 0x2283, 0x1104, 0x10c3, 0x10c1 };	ide_hwif_t *hwif	= HWIF(drive);	ide_drive_t *pair	= ide_get_paired_drive(drive);	u32 speedt		= 0;	u16 speedp		= 0;	unsigned long addr	= siimage_seldev(drive, 0x04);	unsigned long tfaddr	= siimage_selreg(hwif, 0x02);	unsigned long base	= (unsigned long)hwif->hwif_data;	u8 tf_pio		= pio;	u8 addr_mask		= hwif->channel ? (hwif->mmio ? 0xF4 : 0x84)						: (hwif->mmio ? 0xB4 : 0x80);	u8 mode			= 0;	u8 unit			= drive->select.b.unit;	/* trim *taskfile* PIO to the slowest of the master/slave */	if (pair->present) {		u8 pair_pio = ide_get_best_pio_mode(pair, 255, 4);		if (pair_pio < tf_pio)			tf_pio = pair_pio;	}	/* cheat for now and use the docs */	speedp = data_speed[pio];	speedt = tf_speed[tf_pio];	if (hwif->mmio) {		hwif->OUTW(speedp, addr);		hwif->OUTW(speedt, tfaddr);		/* Now set up IORDY */		if (pio > 2)			hwif->OUTW(hwif->INW(tfaddr-2)|0x200, tfaddr-2);		else			hwif->OUTW(hwif->INW(tfaddr-2)&~0x200, tfaddr-2);		mode = hwif->INB(base + addr_mask);		mode &= ~(unit ? 0x30 : 0x03);		mode |= (unit ? 0x10 : 0x01);		hwif->OUTB(mode, base + addr_mask);	} else {		pci_write_config_word(hwif->pci_dev, addr, speedp);		pci_write_config_word(hwif->pci_dev, tfaddr, speedt);		pci_read_config_word(hwif->pci_dev, tfaddr-2, &speedp);		speedp &= ~0x200;		/* Set IORDY for mode 3 or 4 */		if (pio > 2)			speedp |= 0x200;		pci_write_config_word(hwif->pci_dev, tfaddr-2, speedp);		pci_read_config_byte(hwif->pci_dev, addr_mask, &mode);		mode &= ~(unit ? 0x30 : 0x03);		mode |= (unit ? 0x10 : 0x01);		pci_write_config_byte(hwif->pci_dev, addr_mask, mode);	}}/** *	sil_set_dma_mode	-	set host controller for DMA mode *	@drive: drive *	@speed: DMA mode * *	Tune the SiI chipset for the desired DMA mode. */static void sil_set_dma_mode(ide_drive_t *drive, const u8 speed){	u8 ultra6[]		= { 0x0F, 0x0B, 0x07, 0x05, 0x03, 0x02, 0x01 };	u8 ultra5[]		= { 0x0C, 0x07, 0x05, 0x04, 0x02, 0x01 };	u16 dma[]		= { 0x2208, 0x10C2, 0x10C1 };	ide_hwif_t *hwif	= HWIF(drive);	u16 ultra = 0, multi	= 0;	u8 mode = 0, unit	= drive->select.b.unit;	unsigned long base	= (unsigned long)hwif->hwif_data;	u8 scsc = 0, addr_mask	= ((hwif->channel) ?				    ((hwif->mmio) ? 0xF4 : 0x84) :				    ((hwif->mmio) ? 0xB4 : 0x80));				    	unsigned long ma	= siimage_seldev(drive, 0x08);	unsigned long ua	= siimage_seldev(drive, 0x0C);	if (hwif->mmio) {		scsc = hwif->INB(base + 0x4A);		mode = hwif->INB(base + addr_mask);		multi = hwif->INW(ma);		ultra = hwif->INW(ua);	} else {		pci_read_config_byte(hwif->pci_dev, 0x8A, &scsc);		pci_read_config_byte(hwif->pci_dev, addr_mask, &mode);		pci_read_config_word(hwif->pci_dev, ma, &multi);		pci_read_config_word(hwif->pci_dev, ua, &ultra);	}	mode &= ~((unit) ? 0x30 : 0x03);	ultra &= ~0x3F;	scsc = ((scsc & 0x30) == 0x00) ? 0 : 1;	scsc = is_sata(hwif) ? 1 : scsc;	switch(speed) {		case XFER_MW_DMA_2:		case XFER_MW_DMA_1:		case XFER_MW_DMA_0:			multi = dma[speed - XFER_MW_DMA_0];			mode |= ((unit) ? 0x20 : 0x02);			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:			multi = dma[2];			ultra |= ((scsc) ? (ultra6[speed - XFER_UDMA_0]) :					   (ultra5[speed - XFER_UDMA_0]));			mode |= ((unit) ? 0x30 : 0x03);			break;		default:			return;	}	if (hwif->mmio) {		hwif->OUTB(mode, base + addr_mask);		hwif->OUTW(multi, ma);		hwif->OUTW(ultra, ua);	} else {		pci_write_config_byte(hwif->pci_dev, addr_mask, mode);		pci_write_config_word(hwif->pci_dev, ma, multi);		pci_write_config_word(hwif->pci_dev, ua, ultra);	}}/* returns 1 if dma irq issued, 0 otherwise */static int siimage_io_ide_dma_test_irq (ide_drive_t *drive){	ide_hwif_t *hwif	= HWIF(drive);	u8 dma_altstat		= 0;	unsigned long addr	= siimage_selreg(hwif, 1);	/* return 1 if INTR asserted */	if ((hwif->INB(hwif->dma_status) & 4) == 4)		return 1;	/* return 1 if Device INTR asserted */	pci_read_config_byte(hwif->pci_dev, addr, &dma_altstat);	if (dma_altstat & 8)		return 0;	//return 1;	return 0;}/** *	siimage_mmio_ide_dma_test_irq	-	check we caused an IRQ *	@drive: drive we are testing * *	Check if we caused an IDE DMA interrupt. We may also have caused *	SATA status interrupts, if so we clean them up and continue. */ static int siimage_mmio_ide_dma_test_irq (ide_drive_t *drive){	ide_hwif_t *hwif	= HWIF(drive);	unsigned long addr	= siimage_selreg(hwif, 0x1);	if (SATA_ERROR_REG) {		unsigned long base = (unsigned long)hwif->hwif_data;		u32 ext_stat = readl((void __iomem *)(base + 0x10));		u8 watchdog = 0;		if (ext_stat & ((hwif->channel) ? 0x40 : 0x10)) {			u32 sata_error = readl((void __iomem *)SATA_ERROR_REG);			writel(sata_error, (void __iomem *)SATA_ERROR_REG);			watchdog = (sata_error & 0x00680000) ? 1 : 0;			printk(KERN_WARNING "%s: sata_error = 0x%08x, "				"watchdog = %d, %s\n",				drive->name, sata_error, watchdog,				__FUNCTION__);		} else {			watchdog = (ext_stat & 0x8000) ? 1 : 0;		}		ext_stat >>= 16;		if (!(ext_stat & 0x0404) && !watchdog)			return 0;	}	/* return 1 if INTR asserted */	if ((readb((void __iomem *)hwif->dma_status) & 0x04) == 0x04)		return 1;	/* return 1 if Device INTR asserted */	if ((readb((void __iomem *)addr) & 8) == 8)		return 0;	//return 1;	return 0;}/** *	sil_sata_busproc	-	bus isolation IOCTL *	@drive: drive to isolate/restore *	@state: bus state to set * *	Used by the SII3112 to handle bus isolation. As this is a  *	SATA controller the work required is quite limited, we  *	just have to clean up the statistics */static int sil_sata_busproc(ide_drive_t * drive, int state){	ide_hwif_t *hwif	= HWIF(drive);	u32 stat_config		= 0;	unsigned long addr	= siimage_selreg(hwif, 0);	if (hwif->mmio)		stat_config = readl((void __iomem *)addr);	else		pci_read_config_dword(hwif->pci_dev, addr, &stat_config);	switch (state) {		case BUSSTATE_ON:			hwif->drives[0].failures = 0;			hwif->drives[1].failures = 0;			break;		case BUSSTATE_OFF:			hwif->drives[0].failures = hwif->drives[0].max_failures + 1;			hwif->drives[1].failures = hwif->drives[1].max_failures + 1;			break;		case BUSSTATE_TRISTATE:			hwif->drives[0].failures = hwif->drives[0].max_failures + 1;			hwif->drives[1].failures = hwif->drives[1].max_failures + 1;			break;		default:			return -EINVAL;	}	hwif->bus_state = state;	return 0;}/** *	sil_sata_reset_poll	-	wait for SATA reset *	@drive: drive we are resetting * *	Poll the SATA phy and see whether it has come back from the dead *	yet. */static int sil_sata_reset_poll(ide_drive_t *drive){	if (SATA_STATUS_REG) {		ide_hwif_t *hwif	= HWIF(drive);		/* SATA_STATUS_REG is valid only when in MMIO mode */		if ((readl((void __iomem *)SATA_STATUS_REG) & 0x03) != 0x03) {			printk(KERN_WARNING "%s: reset phy dead, status=0x%08x\n",				hwif->name, readl((void __iomem *)SATA_STATUS_REG));			HWGROUP(drive)->polling = 0;			return ide_started;		}	}	return 0;}/** *	sil_sata_pre_reset	-	reset hook *	@drive: IDE device being reset * *	For the SATA devices we need to handle recalibration/geometry *	differently */static void sil_sata_pre_reset(ide_drive_t *drive){	if (drive->media == ide_disk) {

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?