⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 pata_hpt37x.c

📁 linux 内核源代码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* * Libata driver for the highpoint 37x and 30x UDMA66 ATA controllers. * * This driver is heavily based upon: * * linux/drivers/ide/pci/hpt366.c		Version 0.36	April 25, 2003 * * Copyright (C) 1999-2003		Andre Hedrick <andre@linux-ide.org> * Portions Copyright (C) 2001	        Sun Microsystems, Inc. * Portions Copyright (C) 2003		Red Hat Inc * Portions Copyright (C) 2005-2007	MontaVista Software, Inc. * * TODO *	Look into engine reset on timeout errors. Should not be	required. */#include <linux/kernel.h>#include <linux/module.h>#include <linux/pci.h>#include <linux/init.h>#include <linux/blkdev.h>#include <linux/delay.h>#include <scsi/scsi_host.h>#include <linux/libata.h>#define DRV_NAME	"pata_hpt37x"#define DRV_VERSION	"0.6.9"struct hpt_clock {	u8	xfer_speed;	u32	timing;};struct hpt_chip {	const char *name;	unsigned int base;	struct hpt_clock const *clocks[4];};/* key for bus clock timings * bit * 0:3    data_high_time. inactive time of DIOW_/DIOR_ for PIO and MW *        DMA. cycles = value + 1 * 4:8    data_low_time. active time of DIOW_/DIOR_ for PIO and MW *        DMA. cycles = value + 1 * 9:12   cmd_high_time. inactive time of DIOW_/DIOR_ during task file *        register access. * 13:17  cmd_low_time. active time of DIOW_/DIOR_ during task file *        register access. * 18:21  udma_cycle_time. clock freq and clock cycles for UDMA xfer. *        during task file register access. * 22:24  pre_high_time. time to initialize 1st cycle for PIO and MW DMA *        xfer. * 25:27  cmd_pre_high_time. time to initialize 1st PIO cycle for task *        register access. * 28     UDMA enable * 29     DMA enable * 30     PIO_MST enable. if set, the chip is in bus master mode during *        PIO. * 31     FIFO enable. */static struct hpt_clock hpt37x_timings_33[] = {	{ XFER_UDMA_6,		0x12446231 },	/* 0x12646231 ?? */	{ XFER_UDMA_5,		0x12446231 },	{ XFER_UDMA_4,		0x12446231 },	{ XFER_UDMA_3,		0x126c6231 },	{ XFER_UDMA_2,		0x12486231 },	{ XFER_UDMA_1,		0x124c6233 },	{ XFER_UDMA_0,		0x12506297 },	{ XFER_MW_DMA_2,	0x22406c31 },	{ XFER_MW_DMA_1,	0x22406c33 },	{ XFER_MW_DMA_0,	0x22406c97 },	{ XFER_PIO_4,		0x06414e31 },	{ XFER_PIO_3,		0x06414e42 },	{ XFER_PIO_2,		0x06414e53 },	{ XFER_PIO_1,		0x06814e93 },	{ XFER_PIO_0,		0x06814ea7 }};static struct hpt_clock hpt37x_timings_50[] = {	{ XFER_UDMA_6,		0x12848242 },	{ XFER_UDMA_5,		0x12848242 },	{ XFER_UDMA_4,		0x12ac8242 },	{ XFER_UDMA_3,		0x128c8242 },	{ XFER_UDMA_2,		0x120c8242 },	{ XFER_UDMA_1,		0x12148254 },	{ XFER_UDMA_0,		0x121882ea },	{ XFER_MW_DMA_2,	0x22808242 },	{ XFER_MW_DMA_1,	0x22808254 },	{ XFER_MW_DMA_0,	0x228082ea },	{ XFER_PIO_4,		0x0a81f442 },	{ XFER_PIO_3,		0x0a81f443 },	{ XFER_PIO_2,		0x0a81f454 },	{ XFER_PIO_1,		0x0ac1f465 },	{ XFER_PIO_0,		0x0ac1f48a }};static struct hpt_clock hpt37x_timings_66[] = {	{ XFER_UDMA_6,		0x1c869c62 },	{ XFER_UDMA_5,		0x1cae9c62 },	/* 0x1c8a9c62 */	{ XFER_UDMA_4,		0x1c8a9c62 },	{ XFER_UDMA_3,		0x1c8e9c62 },	{ XFER_UDMA_2,		0x1c929c62 },	{ XFER_UDMA_1,		0x1c9a9c62 },	{ XFER_UDMA_0,		0x1c829c62 },	{ XFER_MW_DMA_2,	0x2c829c62 },	{ XFER_MW_DMA_1,	0x2c829c66 },	{ XFER_MW_DMA_0,	0x2c829d2e },	{ XFER_PIO_4,		0x0c829c62 },	{ XFER_PIO_3,		0x0c829c84 },	{ XFER_PIO_2,		0x0c829ca6 },	{ XFER_PIO_1,		0x0d029d26 },	{ XFER_PIO_0,		0x0d029d5e }};static const struct hpt_chip hpt370 = {	"HPT370",	48,	{		hpt37x_timings_33,		NULL,		NULL,		NULL	}};static const struct hpt_chip hpt370a = {	"HPT370A",	48,	{		hpt37x_timings_33,		NULL,		hpt37x_timings_50,		NULL	}};static const struct hpt_chip hpt372 = {	"HPT372",	55,	{		hpt37x_timings_33,		NULL,		hpt37x_timings_50,		hpt37x_timings_66	}};static const struct hpt_chip hpt302 = {	"HPT302",	66,	{		hpt37x_timings_33,		NULL,		hpt37x_timings_50,		hpt37x_timings_66	}};static const struct hpt_chip hpt371 = {	"HPT371",	66,	{		hpt37x_timings_33,		NULL,		hpt37x_timings_50,		hpt37x_timings_66	}};static const struct hpt_chip hpt372a = {	"HPT372A",	66,	{		hpt37x_timings_33,		NULL,		hpt37x_timings_50,		hpt37x_timings_66	}};static const struct hpt_chip hpt374 = {	"HPT374",	48,	{		hpt37x_timings_33,		NULL,		NULL,		NULL	}};/** *	hpt37x_find_mode	-	reset the hpt37x bus *	@ap: ATA port *	@speed: transfer mode * *	Return the 32bit register programming information for this channel *	that matches the speed provided. */static u32 hpt37x_find_mode(struct ata_port *ap, int speed){	struct hpt_clock *clocks = ap->host->private_data;	while(clocks->xfer_speed) {		if (clocks->xfer_speed == speed)			return clocks->timing;		clocks++;	}	BUG();	return 0xffffffffU;	/* silence compiler warning */}static int hpt_dma_blacklisted(const struct ata_device *dev, char *modestr, const char *list[]){	unsigned char model_num[ATA_ID_PROD_LEN + 1];	int i = 0;	ata_id_c_string(dev->id, model_num, ATA_ID_PROD, sizeof(model_num));	while (list[i] != NULL) {		if (!strcmp(list[i], model_num)) {			printk(KERN_WARNING DRV_NAME ": %s is not supported for %s.\n",				modestr, list[i]);			return 1;		}		i++;	}	return 0;}static const char *bad_ata33[] = {	"Maxtor 92720U8", "Maxtor 92040U6", "Maxtor 91360U4", "Maxtor 91020U3", "Maxtor 90845U3", "Maxtor 90650U2",	"Maxtor 91360D8", "Maxtor 91190D7", "Maxtor 91020D6", "Maxtor 90845D5", "Maxtor 90680D4", "Maxtor 90510D3", "Maxtor 90340D2",	"Maxtor 91152D8", "Maxtor 91008D7", "Maxtor 90845D6", "Maxtor 90840D6", "Maxtor 90720D5", "Maxtor 90648D5", "Maxtor 90576D4",	"Maxtor 90510D4",	"Maxtor 90432D3", "Maxtor 90288D2", "Maxtor 90256D2",	"Maxtor 91000D8", "Maxtor 90910D8", "Maxtor 90875D7", "Maxtor 90840D7", "Maxtor 90750D6", "Maxtor 90625D5", "Maxtor 90500D4",	"Maxtor 91728D8", "Maxtor 91512D7", "Maxtor 91303D6", "Maxtor 91080D5", "Maxtor 90845D4", "Maxtor 90680D4", "Maxtor 90648D3", "Maxtor 90432D2",	NULL};static const char *bad_ata100_5[] = {	"IBM-DTLA-307075",	"IBM-DTLA-307060",	"IBM-DTLA-307045",	"IBM-DTLA-307030",	"IBM-DTLA-307020",	"IBM-DTLA-307015",	"IBM-DTLA-305040",	"IBM-DTLA-305030",	"IBM-DTLA-305020",	"IC35L010AVER07-0",	"IC35L020AVER07-0",	"IC35L030AVER07-0",	"IC35L040AVER07-0",	"IC35L060AVER07-0",	"WDC AC310200R",	NULL};/** *	hpt370_filter	-	mode selection filter *	@adev: ATA device * *	Block UDMA on devices that cause trouble with this controller. */static unsigned long hpt370_filter(struct ata_device *adev, unsigned long mask){	if (adev->class == ATA_DEV_ATA) {		if (hpt_dma_blacklisted(adev, "UDMA", bad_ata33))			mask &= ~ATA_MASK_UDMA;		if (hpt_dma_blacklisted(adev, "UDMA100", bad_ata100_5))			mask &= ~(0x1F << ATA_SHIFT_UDMA);	}	return ata_pci_default_filter(adev, mask);}/** *	hpt370a_filter	-	mode selection filter *	@adev: ATA device * *	Block UDMA on devices that cause trouble with this controller. */static unsigned long hpt370a_filter(struct ata_device *adev, unsigned long mask){	if (adev->class == ATA_DEV_ATA) {		if (hpt_dma_blacklisted(adev, "UDMA100", bad_ata100_5))			mask &= ~ (0x1F << ATA_SHIFT_UDMA);	}	return ata_pci_default_filter(adev, mask);}/** *	hpt37x_pre_reset	-	reset the hpt37x bus *	@link: ATA link to reset *	@deadline: deadline jiffies for the operation * *	Perform the initial reset handling for the 370/372 and 374 func 0 */static int hpt37x_pre_reset(struct ata_link *link, unsigned long deadline){	u8 scr2, ata66;	struct ata_port *ap = link->ap;	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	static const struct pci_bits hpt37x_enable_bits[] = {		{ 0x50, 1, 0x04, 0x04 },		{ 0x54, 1, 0x04, 0x04 }	};	if (!pci_test_config_bits(pdev, &hpt37x_enable_bits[ap->port_no]))		return -ENOENT;	pci_read_config_byte(pdev, 0x5B, &scr2);	pci_write_config_byte(pdev, 0x5B, scr2 & ~0x01);	/* Cable register now active */	pci_read_config_byte(pdev, 0x5A, &ata66);	/* Restore state */	pci_write_config_byte(pdev, 0x5B, scr2);	if (ata66 & (2 >> ap->port_no))		ap->cbl = ATA_CBL_PATA40;	else		ap->cbl = ATA_CBL_PATA80;	/* Reset the state machine */	pci_write_config_byte(pdev, 0x50 + 4 * ap->port_no, 0x37);	udelay(100);	return ata_std_prereset(link, deadline);}/** *	hpt37x_error_handler	-	reset the hpt374 *	@ap: ATA port to reset * *	Perform probe for HPT37x, except for HPT374 channel 2 */static void hpt37x_error_handler(struct ata_port *ap){	ata_bmdma_drive_eh(ap, hpt37x_pre_reset, ata_std_softreset, NULL, ata_std_postreset);}static int hpt374_pre_reset(struct ata_link *link, unsigned long deadline){	static const struct pci_bits hpt37x_enable_bits[] = {		{ 0x50, 1, 0x04, 0x04 },		{ 0x54, 1, 0x04, 0x04 }	};	u16 mcr3;	u8 ata66;	struct ata_port *ap = link->ap;	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	unsigned int mcrbase = 0x50 + 4 * ap->port_no;	if (!pci_test_config_bits(pdev, &hpt37x_enable_bits[ap->port_no]))		return -ENOENT;	/* Do the extra channel work */	pci_read_config_word(pdev, mcrbase + 2, &mcr3);	/* Set bit 15 of 0x52 to enable TCBLID as input	 */	pci_write_config_word(pdev, mcrbase + 2, mcr3 | 0x8000);	pci_read_config_byte(pdev, 0x5A, &ata66);	/* Reset TCBLID/FCBLID to output */	pci_write_config_word(pdev, mcrbase + 2, mcr3);	if (ata66 & (2 >> ap->port_no))		ap->cbl = ATA_CBL_PATA40;	else		ap->cbl = ATA_CBL_PATA80;	/* Reset the state machine */	pci_write_config_byte(pdev, 0x50 + 4 * ap->port_no, 0x37);	udelay(100);	return ata_std_prereset(link, deadline);}/** *	hpt374_error_handler	-	reset the hpt374 *	@classes: * *	The 374 cable detect is a little different due to the extra *	channels. The function 0 channels work like usual but function 1 *	is special */static void hpt374_error_handler(struct ata_port *ap){	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	if (!(PCI_FUNC(pdev->devfn) & 1))		hpt37x_error_handler(ap);	else		ata_bmdma_drive_eh(ap, hpt374_pre_reset, ata_std_softreset, NULL, ata_std_postreset);}/** *	hpt370_set_piomode		-	PIO setup *	@ap: ATA interface *	@adev: device on the interface * *	Perform PIO mode setup. */static void hpt370_set_piomode(struct ata_port *ap, struct ata_device *adev){	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	u32 addr1, addr2;	u32 reg;	u32 mode;	u8 fast;	addr1 = 0x40 + 4 * (adev->devno + 2 * ap->port_no);	addr2 = 0x51 + 4 * ap->port_no;	/* Fast interrupt prediction disable, hold off interrupt disable */	pci_read_config_byte(pdev, addr2, &fast);	fast &= ~0x02;	fast |= 0x01;	pci_write_config_byte(pdev, addr2, fast);	pci_read_config_dword(pdev, addr1, &reg);	mode = hpt37x_find_mode(ap, adev->pio_mode);	mode &= ~0x8000000;	/* No FIFO in PIO */	mode &= ~0x30070000;	/* Leave config bits alone */	reg &= 0x30070000;	/* Strip timing bits */	pci_write_config_dword(pdev, addr1, reg | mode);}/** *	hpt370_set_dmamode		-	DMA timing setup *	@ap: ATA interface *	@adev: Device being configured * *	Set up the channel for MWDMA or UDMA modes. Much the same as with *	PIO, load the mode number and then set MWDMA or UDMA flag. */static void hpt370_set_dmamode(struct ata_port *ap, struct ata_device *adev){	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	u32 addr1, addr2;	u32 reg;	u32 mode;	u8 fast;	addr1 = 0x40 + 4 * (adev->devno + 2 * ap->port_no);	addr2 = 0x51 + 4 * ap->port_no;	/* Fast interrupt prediction disable, hold off interrupt disable */	pci_read_config_byte(pdev, addr2, &fast);	fast &= ~0x02;	fast |= 0x01;	pci_write_config_byte(pdev, addr2, fast);	pci_read_config_dword(pdev, addr1, &reg);	mode = hpt37x_find_mode(ap, adev->dma_mode);	mode |= 0x8000000;	/* FIFO in MWDMA or UDMA */	mode &= ~0xC0000000;	/* Leave config bits alone */	reg &= 0xC0000000;	/* Strip timing bits */	pci_write_config_dword(pdev, addr1, reg | mode);}/** *	hpt370_bmdma_start		-	DMA engine begin *	@qc: ATA command * *	The 370 and 370A want us to reset the DMA engine each time we *	use it. The 372 and later are fine. */static void hpt370_bmdma_start(struct ata_queued_cmd *qc){	struct ata_port *ap = qc->ap;	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	pci_write_config_byte(pdev, 0x50 + 4 * ap->port_no, 0x37);	udelay(10);	ata_bmdma_start(qc);}/** *	hpt370_bmdma_end		-	DMA engine stop *	@qc: ATA command * *	Work around the HPT370 DMA engine. */static void hpt370_bmdma_stop(struct ata_queued_cmd *qc){	struct ata_port *ap = qc->ap;	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	u8 dma_stat = ioread8(ap->ioaddr.bmdma_addr + 2);	u8 dma_cmd;	void __iomem *bmdma = ap->ioaddr.bmdma_addr;	if (dma_stat & 0x01) {		udelay(20);		dma_stat = ioread8(bmdma + 2);	}	if (dma_stat & 0x01) {		/* Clear the engine */		pci_write_config_byte(pdev, 0x50 + 4 * ap->port_no, 0x37);		udelay(10);		/* Stop DMA */		dma_cmd = ioread8(bmdma );		iowrite8(dma_cmd & 0xFE, bmdma);		/* Clear Error */		dma_stat = ioread8(bmdma + 2);		iowrite8(dma_stat | 0x06 , bmdma + 2);		/* Clear the engine */		pci_write_config_byte(pdev, 0x50 + 4 * ap->port_no, 0x37);		udelay(10);	}	ata_bmdma_stop(qc);}/** *	hpt372_set_piomode		-	PIO setup *	@ap: ATA interface *	@adev: device on the interface * *	Perform PIO mode setup. */static void hpt372_set_piomode(struct ata_port *ap, struct ata_device *adev){	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	u32 addr1, addr2;	u32 reg;	u32 mode;	u8 fast;	addr1 = 0x40 + 4 * (adev->devno + 2 * ap->port_no);	addr2 = 0x51 + 4 * ap->port_no;	/* Fast interrupt prediction disable, hold off interrupt disable */	pci_read_config_byte(pdev, addr2, &fast);	fast &= ~0x07;	pci_write_config_byte(pdev, addr2, fast);	pci_read_config_dword(pdev, addr1, &reg);	mode = hpt37x_find_mode(ap, adev->pio_mode);	printk("Find mode for %d reports %X\n", adev->pio_mode, mode);	mode &= ~0x80000000;	/* No FIFO in PIO */	mode &= ~0x30070000;	/* Leave config bits alone */	reg &= 0x30070000;	/* Strip timing bits */	pci_write_config_dword(pdev, addr1, reg | mode);}/** *	hpt372_set_dmamode		-	DMA timing setup *	@ap: ATA interface *	@adev: Device being configured * *	Set up the channel for MWDMA or UDMA modes. Much the same as with *	PIO, load the mode number and then set MWDMA or UDMA flag. */static void hpt372_set_dmamode(struct ata_port *ap, struct ata_device *adev){	struct pci_dev *pdev = to_pci_dev(ap->host->dev);	u32 addr1, addr2;	u32 reg;	u32 mode;	u8 fast;	addr1 = 0x40 + 4 * (adev->devno + 2 * ap->port_no);	addr2 = 0x51 + 4 * ap->port_no;	/* Fast interrupt prediction disable, hold off interrupt disable */	pci_read_config_byte(pdev, addr2, &fast);	fast &= ~0x07;	pci_write_config_byte(pdev, addr2, fast);	pci_read_config_dword(pdev, addr1, &reg);	mode = hpt37x_find_mode(ap, adev->dma_mode);	printk("Find mode for DMA %d reports %X\n", adev->dma_mode, mode);	mode &= ~0xC0000000;	/* Leave config bits alone */	mode |= 0x80000000;	/* FIFO in MWDMA or UDMA */	reg &= 0xC0000000;	/* Strip timing bits */	pci_write_config_dword(pdev, addr1, reg | mode);}/** *	hpt37x_bmdma_end		-	DMA engine stop

⌨️ 快捷键说明

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