summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
Diffstat (limited to 'sys-kernel/gentoo-sources/files/gentoo-sources-2.4.20-hpt372.patch')
-rw-r--r--sys-kernel/gentoo-sources/files/gentoo-sources-2.4.20-hpt372.patch1026
1 files changed, 1026 insertions, 0 deletions
diff --git a/sys-kernel/gentoo-sources/files/gentoo-sources-2.4.20-hpt372.patch b/sys-kernel/gentoo-sources/files/gentoo-sources-2.4.20-hpt372.patch
new file mode 100644
index 000000000000..9eb8440bfa1e
--- /dev/null
+++ b/sys-kernel/gentoo-sources/files/gentoo-sources-2.4.20-hpt372.patch
@@ -0,0 +1,1026 @@
+--- linux-2.4.20-gentoo-r5/drivers/ide/ide-pci.c.original 2003-05-13 18:29:31.000000000 -0400
++++ linux-2.4.20-gentoo-r5/drivers/ide/ide-pci.c 2003-05-14 00:14:36.000000000 -0400
+@@ -79,6 +79,8 @@
+ #define DEVID_UM8886BF ((ide_pci_devid_t){PCI_VENDOR_ID_UMC, PCI_DEVICE_ID_UMC_UM8886BF})
+ #define DEVID_HPT34X ((ide_pci_devid_t){PCI_VENDOR_ID_TTI, PCI_DEVICE_ID_TTI_HPT343})
+ #define DEVID_HPT366 ((ide_pci_devid_t){PCI_VENDOR_ID_TTI, PCI_DEVICE_ID_TTI_HPT366})
++#define DEVID_HPT372 ((ide_pci_devid_t){PCI_VENDOR_ID_TTI, PCI_DEVICE_ID_TTI_HPT372})
++#define DEVID_HPT374 ((ide_pci_devid_t){PCI_VENDOR_ID_TTI, PCI_DEVICE_ID_TTI_HPT374})
+ #define DEVID_ALI15X3 ((ide_pci_devid_t){PCI_VENDOR_ID_AL, PCI_DEVICE_ID_AL_M5229})
+ #define DEVID_CY82C693 ((ide_pci_devid_t){PCI_VENDOR_ID_CONTAQ, PCI_DEVICE_ID_CONTAQ_82C693})
+ #define DEVID_HINT ((ide_pci_devid_t){0x3388, 0x8013})
+@@ -452,6 +454,8 @@
+ {DEVID_UM8886BF,"UM8886BF", NULL, NULL, NULL, NULL, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, ON_BOARD, 0 },
+ {DEVID_HPT34X, "HPT34X", PCI_HPT34X, NULL, INIT_HPT34X, NULL, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, NEVER_BOARD, 16 },
+ {DEVID_HPT366, "HPT366", PCI_HPT366, ATA66_HPT366, INIT_HPT366, DMA_HPT366, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, OFF_BOARD, 240 },
++ {DEVID_HPT372, "HPT372", PCI_HPT366, ATA66_HPT366, INIT_HPT366, DMA_HPT366, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, OFF_BOARD, 240 },
++ {DEVID_HPT374, "HPT374", PCI_HPT366, ATA66_HPT366, INIT_HPT366, DMA_HPT366, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, OFF_BOARD, 240 },
+ {DEVID_ALI15X3, "ALI15X3", PCI_ALI15X3, ATA66_ALI15X3, INIT_ALI15X3, DMA_ALI15X3, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, ON_BOARD, 0 },
+ {DEVID_CY82C693,"CY82C693", PCI_CY82C693, NULL, INIT_CY82C693, NULL, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, ON_BOARD, 0 },
+ {DEVID_HINT, "HINT_IDE", NULL, NULL, NULL, NULL, {{0x00,0x00,0x00}, {0x00,0x00,0x00}}, ON_BOARD, 0 },
+@@ -478,6 +482,7 @@
+ {
+ switch(dev->device) {
+ case PCI_DEVICE_ID_TTI_HPT366:
++ case PCI_DEVICE_ID_TTI_HPT374:
+ case PCI_DEVICE_ID_PROMISE_20246:
+ case PCI_DEVICE_ID_PROMISE_20262:
+ case PCI_DEVICE_ID_PROMISE_20265:
+@@ -828,6 +833,7 @@
+ IDE_PCI_DEVID_EQ(d->devid, DEVID_AEC6260R) ||
+ IDE_PCI_DEVID_EQ(d->devid, DEVID_HPT34X) ||
+ IDE_PCI_DEVID_EQ(d->devid, DEVID_HPT366) ||
++ IDE_PCI_DEVID_EQ(d->devid, DEVID_HPT374) ||
+ IDE_PCI_DEVID_EQ(d->devid, DEVID_CS5530) ||
+ IDE_PCI_DEVID_EQ(d->devid, DEVID_CY82C693) ||
+ IDE_PCI_DEVID_EQ(d->devid, DEVID_CMD646) ||
+@@ -910,6 +916,44 @@
+ ide_setup_pci_device(dev2, d2);
+ }
+
++static void __init hpt374_device_order_fixup (struct pci_dev *dev, ide_pci_device_t *d)
++{
++ struct pci_dev *dev2 = NULL, *findev;
++ ide_pci_device_t *d2;
++
++ if (PCI_FUNC(dev->devfn) & 1)
++ return;
++
++ pci_for_each_dev(findev) {
++ if ((findev->vendor == dev->vendor) &&
++ (findev->device == dev->device) &&
++ ((findev->devfn - dev->devfn) == 1) &&
++ (PCI_FUNC(findev->devfn) & 1)) {
++ dev2 = findev;
++ break;
++ }
++ }
++
++ printk("%s: IDE controller on PCI bus %02x dev %02x\n", d->name, dev->bus->number, dev->devfn);
++ ide_setup_pci_device(dev, d);
++ if (!dev2) {
++ return;
++ } else {
++ byte irq = 0, irq2 = 0;
++ pci_read_config_byte(dev, PCI_INTERRUPT_LINE, &irq);
++ pci_read_config_byte(dev2, PCI_INTERRUPT_LINE, &irq2);
++ if (irq != irq2) {
++ pci_write_config_byte(dev2, PCI_INTERRUPT_LINE, irq);
++ dev2->irq = dev->irq;
++ printk("%s: pci-config space interrupt fixed.\n", d->name);
++ }
++ }
++ d2 = d;
++ printk("%s: IDE controller on PCI bus %02x dev %02x\n", d2->name, dev2->bus->number, dev2->devfn);
++ ide_setup_pci_device(dev2, d2);
++
++}
++
+ static void __init hpt366_device_order_fixup (struct pci_dev *dev, ide_pci_device_t *d)
+ {
+ struct pci_dev *dev2 = NULL, *findev;
+@@ -992,6 +1036,8 @@
+ return; /* UM8886A/BF pair */
+ else if (IDE_PCI_DEVID_EQ(d->devid, DEVID_HPT366))
+ hpt366_device_order_fixup(dev, d);
++ else if (IDE_PCI_DEVID_EQ(d->devid, DEVID_HPT374))
++ hpt374_device_order_fixup(dev, d);
+ else if (IDE_PCI_DEVID_EQ(d->devid, DEVID_PDC20270))
+ pdc20270_device_order_fixup(dev, d);
+ else if (!IDE_PCI_DEVID_EQ(d->devid, IDE_PCI_DEVID_NULL) || (dev->class >> 8) == PCI_CLASS_STORAGE_IDE) {
+--- linux-2.4.20-gentoo-r5/drivers/ide/hpt366.c.original 2003-05-13 18:29:31.000000000 -0400
++++ linux-2.4.20-gentoo-r5/drivers/ide/hpt366.c 2003-03-22 14:30:06.000000000 -0500
+@@ -67,8 +67,8 @@
+
+ /* various tuning parameters */
+ #define HPT_RESET_STATE_ENGINE
+-/*#define HPT_DELAY_INTERRUPT*/
+-/*#define HPT_SERIALIZE_IO*/
++#undef HPT_DELAY_INTERRUPT
++#undef HPT_SERIALIZE_IO
+
+ #if defined(DISPLAY_HPT366_TIMINGS) && defined(CONFIG_PROC_FS)
+ #include <linux/stat.h>
+@@ -166,8 +166,7 @@
+ * PIO.
+ * 31 FIFO enable.
+ */
+-struct chipset_bus_clock_list_entry forty_base [] = {
+-
++struct chipset_bus_clock_list_entry forty_base_hpt366[] = {
+ { XFER_UDMA_4, 0x900fd943 },
+ { XFER_UDMA_3, 0x900ad943 },
+ { XFER_UDMA_2, 0x900bd943 },
+@@ -186,7 +185,7 @@
+ { 0, 0x0120d9d9 }
+ };
+
+-struct chipset_bus_clock_list_entry thirty_three_base [] = {
++struct chipset_bus_clock_list_entry thirty_three_base_hpt366[] = {
+
+ { XFER_UDMA_4, 0x90c9a731 },
+ { XFER_UDMA_3, 0x90cfa731 },
+@@ -206,8 +205,7 @@
+ { 0, 0x0120a7a7 }
+ };
+
+-struct chipset_bus_clock_list_entry twenty_five_base [] = {
+-
++struct chipset_bus_clock_list_entry twenty_five_base_hpt366[] = {
+ { XFER_UDMA_4, 0x90c98521 },
+ { XFER_UDMA_3, 0x90cf8521 },
+ { XFER_UDMA_2, 0x90cf8521 },
+@@ -226,7 +224,6 @@
+ { 0, 0x01208585 }
+ };
+
+-#if 1
+ /* these are the current (4 sep 2001) timings from highpoint */
+ struct chipset_bus_clock_list_entry thirty_three_base_hpt370[] = {
+ { XFER_UDMA_5, 0x12446231 },
+@@ -268,9 +265,9 @@
+ { XFER_PIO_0, 0x0d02bf5f },
+ { 0, 0x0d02bf5f }
+ };
+-#else
++
+ /* from highpoint documentation. these are old values */
+-struct chipset_bus_clock_list_entry thirty_three_base_hpt370[] = {
++struct chipset_bus_clock_list_entry thirty_three_base_old_hpt370[] = {
+ { XFER_UDMA_5, 0x16454e31 },
+ { XFER_UDMA_4, 0x16454e31 },
+ { XFER_UDMA_3, 0x166d4e31 },
+@@ -290,7 +287,7 @@
+ { 0, 0x06514e57 }
+ };
+
+-struct chipset_bus_clock_list_entry sixty_six_base_hpt370[] = {
++struct chipset_bus_clock_list_entry sixty_six_base_old_hpt370[] = {
+ { XFER_UDMA_5, 0x14846231 },
+ { XFER_UDMA_4, 0x14886231 },
+ { XFER_UDMA_3, 0x148c6231 },
+@@ -309,7 +306,6 @@
+ { XFER_PIO_0, 0x06914e57 },
+ { 0, 0x06514e57 }
+ };
+-#endif
+
+ struct chipset_bus_clock_list_entry fifty_base_hpt370[] = {
+ { XFER_UDMA_5, 0x12848242 },
+@@ -331,6 +327,144 @@
+ { 0, 0x0ac1f48a }
+ };
+
++struct chipset_bus_clock_list_entry thirty_three_base_hpt372[] = {
++ { XFER_UDMA_6, 0x1c81dc62 },
++ { XFER_UDMA_5, 0x1c6ddc62 },
++ { XFER_UDMA_4, 0x1c8ddc62 },
++ { XFER_UDMA_3, 0x1c8edc62 }, /* checkme */
++ { XFER_UDMA_2, 0x1c91dc62 },
++ { XFER_UDMA_1, 0x1c9adc62 }, /* checkme */
++ { XFER_UDMA_0, 0x1c82dc62 }, /* checkme */
++
++ { XFER_MW_DMA_2, 0x2c829262 },
++ { XFER_MW_DMA_1, 0x2c829266 }, /* checkme */
++ { XFER_MW_DMA_0, 0x2c82922e }, /* checkme */
++
++ { XFER_PIO_4, 0x0c829c62 },
++ { XFER_PIO_3, 0x0c829c84 },
++ { XFER_PIO_2, 0x0c829ca6 },
++ { XFER_PIO_1, 0x0d029d26 },
++ { XFER_PIO_0, 0x0d029d5e },
++ { 0, 0x0d029d5e }
++};
++
++struct chipset_bus_clock_list_entry fifty_base_hpt372[] = {
++ { 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 },
++ { 0, 0x0a81f443 }
++};
++
++struct chipset_bus_clock_list_entry sixty_six_base_hpt372[] = {
++ { XFER_UDMA_6, 0x1c869c62 },
++ { XFER_UDMA_5, 0x1cae9c62 },
++ { 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 },
++ { 0, 0x0d029d26 }
++};
++
++struct chipset_bus_clock_list_entry thirty_three_base_hpt374[] = {
++ { XFER_UDMA_6, 0x12808242 },
++ { 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 },
++ { 0, 0x06814e93 }
++};
++
++#if 0
++struct chipset_bus_clock_list_entry fifty_base_hpt374[] = {
++ { XFER_UDMA_6, },
++ { XFER_UDMA_5, },
++ { XFER_UDMA_4, },
++ { XFER_UDMA_3, },
++ { XFER_UDMA_2, },
++ { XFER_UDMA_1, },
++ { XFER_UDMA_0, },
++ { XFER_MW_DMA_2, },
++ { XFER_MW_DMA_1, },
++ { XFER_MW_DMA_0, },
++ { XFER_PIO_4, },
++ { XFER_PIO_3, },
++ { XFER_PIO_2, },
++ { XFER_PIO_1, },
++ { XFER_PIO_0, },
++ { 0, }
++};
++#endif
++#if 0
++struct chipset_bus_clock_list_entry sixty_six_base_hpt374[] = {
++ { XFER_UDMA_6, 0x12406231 }, /* checkme */
++ { XFER_UDMA_5, 0x12446231 },
++ 0x14846231
++ { XFER_UDMA_4, 0x16814ea7 },
++ 0x14886231
++ { XFER_UDMA_3, 0x16814ea7 },
++ 0x148c6231
++ { XFER_UDMA_2, 0x16814ea7 },
++ 0x148c6231
++ { XFER_UDMA_1, 0x16814ea7 },
++ 0x14906231
++ { XFER_UDMA_0, 0x16814ea7 },
++ 0x14986231
++ { XFER_MW_DMA_2, 0x16814ea7 },
++ 0x26514e21
++ { XFER_MW_DMA_1, 0x16814ea7 },
++ 0x26514e97
++ { XFER_MW_DMA_0, 0x16814ea7 },
++ 0x26514e97
++ { XFER_PIO_4, 0x06814ea7 },
++ 0x06514e21
++ { XFER_PIO_3, 0x06814ea7 },
++ 0x06514e22
++ { XFER_PIO_2, 0x06814ea7 },
++ 0x06514e33
++ { XFER_PIO_1, 0x06814ea7 },
++ 0x06914e43
++ { XFER_PIO_0, 0x06814ea7 },
++ 0x06914e57
++ { 0, 0x06814ea7 }
++};
++#endif
++
+ #define HPT366_DEBUG_DRIVE_INFO 0
+ #define HPT370_ALLOW_ATA100_5 1
+ #define HPT366_ALLOW_ATA66_4 1
+@@ -345,8 +479,11 @@
+ static struct pci_dev *hpt_devs[HPT366_MAX_DEVS];
+ static int n_hpt_devs;
+
+-static unsigned int pci_rev_check_hpt3xx(struct pci_dev *dev);
+ static unsigned int pci_rev2_check_hpt3xx(struct pci_dev *dev);
++static unsigned int pci_rev3_check_hpt3xx(struct pci_dev *dev);
++static unsigned int pci_rev5_check_hpt3xx(struct pci_dev *dev);
++static unsigned int pci_rev7_check_hpt3xx(struct pci_dev *dev);
++
+ byte hpt366_proc = 0;
+ byte hpt363_shared_irq;
+ byte hpt363_shared_pin;
+@@ -360,11 +497,13 @@
+ static int hpt366_get_info (char *buffer, char **addr, off_t offset, int count)
+ {
+ char *p = buffer;
+- char *chipset_nums[] = {"366", "366", "368", "370", "370A"};
++ char *chipset_nums[] = {"366", "366", "368",
++ "370", "370A", "372",
++ "??", "374" };
+ int i;
+
+ p += sprintf(p, "\n "
+- "HighPoint HPT366/368/370\n");
++ "HighPoint HPT366/368/370/372/374\n");
+ for (i = 0; i < n_hpt_devs; i++) {
+ struct pci_dev *dev = hpt_devs[i];
+ unsigned long iobase = dev->resource[4].start;
+@@ -373,9 +512,14 @@
+
+ pci_read_config_dword(dev, PCI_CLASS_REVISION, &class_rev);
+ class_rev &= 0xff;
++
++ if(class_rev >= sizeof(chipset_nums)/sizeof(char *))
+
+ p += sprintf(p, "\nController: %d\n", i);
+- p += sprintf(p, "Chipset: HPT%s\n", chipset_nums[class_rev]);
++ if(class_rev >= sizeof(chipset_nums)/sizeof(char *))
++ p += sprintf(p, "Chipset: UNKNOWN\n");
++ else
++ p += sprintf(p, "Chipset: HPT%s\n", chipset_nums[class_rev]);
+ p += sprintf(p, "--------------- Primary Channel "
+ "--------------- Secondary Channel "
+ "--------------\n");
+@@ -388,7 +532,7 @@
+ (c0 & 0x80) ? "no" : "yes",
+ (c1 & 0x80) ? "no" : "yes");
+
+- if (pci_rev_check_hpt3xx(dev)) {
++ if (pci_rev3_check_hpt3xx(dev)) {
+ u8 cbl;
+ cbl = inb_p(iobase + 0x7b);
+ outb_p(cbl | 1, iobase + 0x7b);
+@@ -437,7 +581,19 @@
+ }
+ #endif /* defined(DISPLAY_HPT366_TIMINGS) && defined(CONFIG_PROC_FS) */
+
+-static unsigned int pci_rev_check_hpt3xx (struct pci_dev *dev)
++/*
++ * fixme: it really needs to be a switch.
++ */
++
++static unsigned int pci_rev2_check_hpt3xx (struct pci_dev *dev)
++{
++ unsigned int class_rev;
++ pci_read_config_dword(dev, PCI_CLASS_REVISION, &class_rev);
++ class_rev &= 0xff;
++ return ((int) (class_rev > 0x01) ? 1 : 0);
++}
++
++static unsigned int pci_rev3_check_hpt3xx (struct pci_dev *dev)
+ {
+ unsigned int class_rev;
+ pci_read_config_dword(dev, PCI_CLASS_REVISION, &class_rev);
+@@ -445,12 +601,20 @@
+ return ((int) (class_rev > 0x02) ? 1 : 0);
+ }
+
+-static unsigned int pci_rev2_check_hpt3xx (struct pci_dev *dev)
++static unsigned int pci_rev5_check_hpt3xx (struct pci_dev *dev)
+ {
+ unsigned int class_rev;
+ pci_read_config_dword(dev, PCI_CLASS_REVISION, &class_rev);
+ class_rev &= 0xff;
+- return ((int) (class_rev > 0x01) ? 1 : 0);
++ return ((int) (class_rev > 0x04) ? 1 : 0);
++}
++
++static unsigned int pci_rev7_check_hpt3xx (struct pci_dev *dev)
++{
++ unsigned int class_rev;
++ pci_read_config_dword(dev, PCI_CLASS_REVISION, &class_rev);
++ class_rev &= 0xff;
++ return ((int) (class_rev > 0x06) ? 1 : 0);
+ }
+
+ static int check_in_drive_lists (ide_drive_t *drive, const char **list)
+@@ -484,54 +648,31 @@
+
+ static void hpt366_tune_chipset (ide_drive_t *drive, byte speed)
+ {
+- byte regtime = (drive->select.b.unit & 0x01) ? 0x44 : 0x40;
++ struct pci_dev *dev = HWIF(drive)->pci_dev;
++ byte drive_pci = (drive->select.b.unit & 0x01) ? 0x44 : 0x40;
+ byte regfast = (HWIF(drive)->channel) ? 0x55 : 0x51;
+- /*
+- * since the channel is always 0 it does not matter.
+- */
+-
+- unsigned int reg1 = 0;
+- unsigned int reg2 = 0;
+ byte drive_fast = 0;
+-
+- /*
+- * Disable the "fast interrupt" prediction.
+- */
+- pci_read_config_byte(HWIF(drive)->pci_dev, regfast, &drive_fast);
++ unsigned int list_conf = 0;
++ unsigned int drive_conf = 0;
++ unsigned int conf_mask = (speed >= XFER_MW_DMA_0) ? 0xc0000000 : 0x30070000;
++
++ /* Disable the "fast interrupt" prediction. */
++ pci_read_config_byte(dev, regfast, &drive_fast);
+ if (drive_fast & 0x02)
+- pci_write_config_byte(HWIF(drive)->pci_dev, regfast, drive_fast & ~0x20);
+-
+- pci_read_config_dword(HWIF(drive)->pci_dev, regtime, &reg1);
+- /* detect bus speed by looking at control reg timing: */
+- switch((reg1 >> 8) & 7) {
+- case 5:
+- reg2 = pci_bus_clock_list(speed, forty_base);
+- break;
+- case 9:
+- reg2 = pci_bus_clock_list(speed, twenty_five_base);
+- break;
+- default:
+- case 7:
+- reg2 = pci_bus_clock_list(speed, thirty_three_base);
+- break;
+- }
+-#if 0
+- /* this is a nice idea ... */
++ pci_write_config_byte(dev, regfast, drive_fast & ~0x20);
+ list_conf = pci_bus_clock_list(speed,
+ (struct chipset_bus_clock_list_entry *)
+ dev->sysdata);
+-#endif
+- /*
+- * Disable on-chip PIO FIFO/buffer (to avoid problems handling I/O errors later)
+- */
+- if (speed >= XFER_MW_DMA_0) {
+- reg2 = (reg2 & ~0xc0000000) | (reg1 & 0xc0000000);
+- } else {
+- reg2 = (reg2 & ~0x30070000) | (reg1 & 0x30070000);
+- }
+- reg2 &= ~0x80000000;
++ pci_read_config_dword(dev, drive_pci, &drive_conf);
++ list_conf = (list_conf & ~conf_mask) | (drive_conf & conf_mask);
++ if (speed < XFER_MW_DMA_0)
++ list_conf &= ~0x80000000; /* Disable on-chip PIO FIFO/buffer */
++ pci_write_config_dword(dev, drive_pci, list_conf);
++}
+
+- pci_write_config_dword(HWIF(drive)->pci_dev, regtime, reg2);
++static void hpt368_tune_chipset (ide_drive_t *drive, byte speed)
++{
++ hpt366_tune_chipset(drive, speed);
+ }
+
+ static void hpt370_tune_chipset (ide_drive_t *drive, byte speed)
+@@ -577,6 +718,39 @@
+ pci_write_config_dword(dev, drive_pci, list_conf);
+ }
+
++static void hpt372_tune_chipset (ide_drive_t *drive, byte speed)
++{
++ byte regfast = (HWIF(drive)->channel) ? 0x55 : 0x51;
++ unsigned int list_conf = 0;
++ unsigned int drive_conf = 0;
++ unsigned int conf_mask = (speed >= XFER_MW_DMA_0) ? 0xc0000000 : 0x30070000;
++ byte drive_pci = 0x40 + (drive->dn * 4);
++ byte drive_fast = 0;
++ struct pci_dev *dev = HWIF(drive)->pci_dev;
++
++ /*
++ * Disable the "fast interrupt" prediction.
++ * don't holdoff on interrupts. (== 0x01 despite what the docs say)
++ */
++ pci_read_config_byte(dev, regfast, &drive_fast);
++ drive_fast &= ~0x07;
++ pci_write_config_byte(HWIF(drive)->pci_dev, regfast, drive_fast);
++
++ list_conf = pci_bus_clock_list(speed,
++ (struct chipset_bus_clock_list_entry *)
++ dev->sysdata);
++ pci_read_config_dword(dev, drive_pci, &drive_conf);
++ list_conf = (list_conf & ~conf_mask) | (drive_conf & conf_mask);
++ if (speed < XFER_MW_DMA_0)
++ list_conf &= ~0x80000000; /* Disable on-chip PIO FIFO/buffer */
++ pci_write_config_dword(dev, drive_pci, list_conf);
++}
++
++static void hpt374_tune_chipset (ide_drive_t *drive, byte speed)
++{
++ hpt372_tune_chipset(drive, speed);
++}
++
+ static int hpt3xx_tune_chipset (ide_drive_t *drive, byte speed)
+ {
+ if ((drive->media != ide_disk) && (speed < XFER_SW_DMA_0))
+@@ -585,9 +759,15 @@
+ if (!drive->init_speed)
+ drive->init_speed = speed;
+
+- if (pci_rev_check_hpt3xx(HWIF(drive)->pci_dev)) {
++ if (pci_rev7_check_hpt3xx(HWIF(drive)->pci_dev)) {
++ hpt374_tune_chipset(drive, speed);
++ } else if (pci_rev5_check_hpt3xx(HWIF(drive)->pci_dev)) {
++ hpt372_tune_chipset(drive, speed);
++ } else if (pci_rev3_check_hpt3xx(HWIF(drive)->pci_dev)) {
+ hpt370_tune_chipset(drive, speed);
+- } else {
++ } else if (pci_rev2_check_hpt3xx(HWIF(drive)->pci_dev)) {
++ hpt368_tune_chipset(drive, speed);
++ } else {
+ hpt366_tune_chipset(drive, speed);
+ }
+ drive->current_speed = speed;
+@@ -664,13 +844,19 @@
+ byte ultra66 = eighty_ninty_three(drive);
+ int rval;
+
++ config_chipset_for_pio(drive);
++
+ if ((drive->media != ide_disk) && (speed < XFER_SW_DMA_0))
+ return ((int) ide_dma_off_quietly);
+
+- if ((id->dma_ultra & 0x0020) &&
++ if ((id->dma_ultra & 0x0040) &&
++ (pci_rev5_check_hpt3xx(HWIF(drive)->pci_dev)) &&
++ (ultra66)) {
++ speed = XFER_UDMA_6;
++ } else if ((id->dma_ultra & 0x0020) &&
+ (!check_in_drive_lists(drive, bad_ata100_5)) &&
+ (HPT370_ALLOW_ATA100_5) &&
+- (pci_rev_check_hpt3xx(HWIF(drive)->pci_dev)) &&
++ (pci_rev3_check_hpt3xx(HWIF(drive)->pci_dev)) &&
+ (ultra66)) {
+ speed = XFER_UDMA_5;
+ } else if ((id->dma_ultra & 0x0010) &&
+@@ -703,7 +889,8 @@
+
+ (void) hpt3xx_tune_chipset(drive, speed);
+
+- rval = (int)( ((id->dma_ultra >> 11) & 7) ? ide_dma_on :
++ rval = (int)( ((id->dma_ultra >> 14) & 3) ? ide_dma_on :
++ ((id->dma_ultra >> 11) & 7) ? ide_dma_on :
+ ((id->dma_ultra >> 8) & 7) ? ide_dma_on :
+ ((id->dma_mword >> 8) & 7) ? ide_dma_on :
+ ide_dma_off_quietly);
+@@ -717,17 +904,16 @@
+
+ void hpt3xx_intrproc (ide_drive_t *drive)
+ {
+- if (drive->quirk_list) {
+- /* drives in the quirk_list may not like intr setups/cleanups */
+- } else {
+- OUT_BYTE((drive)->ctl|2, HWIF(drive)->io_ports[IDE_CONTROL_OFFSET]);
+- }
++ /* drives in the quirk_list may not like intr setups/cleanups */
++ if (drive->quirk_list)
++ return;
++ OUT_BYTE((drive)->ctl|2, HWIF(drive)->io_ports[IDE_CONTROL_OFFSET]);
+ }
+
+ void hpt3xx_maskproc (ide_drive_t *drive, int mask)
+ {
+ if (drive->quirk_list) {
+- if (pci_rev_check_hpt3xx(HWIF(drive)->pci_dev)) {
++ if (pci_rev3_check_hpt3xx(HWIF(drive)->pci_dev)) {
+ byte reg5a = 0;
+ pci_read_config_byte(HWIF(drive)->pci_dev, 0x5a, &reg5a);
+ if (((reg5a & 0x10) >> 4) != mask)
+@@ -758,7 +944,7 @@
+ }
+ dma_func = ide_dma_off_quietly;
+ if (id->field_valid & 4) {
+- if (id->dma_ultra & 0x002F) {
++ if (id->dma_ultra & 0x007F) {
+ /* Force if Capable UltraDMA */
+ dma_func = config_chipset_for_dma(drive);
+ if ((id->field_valid & 2) &&
+@@ -821,8 +1007,10 @@
+ reg50h, reg52h, reg5ah);
+ if (reg5ah & 0x10)
+ pci_write_config_byte(HWIF(drive)->pci_dev, 0x5a, reg5ah & ~0x10);
++#if 1
++ /* how about we flush and reset, mmmkay? */
++ pci_write_config_byte(HWIF(drive)->pci_dev, 0x51, 0x1F);
+ /* fall through to a reset */
+-#if 0
+ case ide_dma_begin:
+ case ide_dma_end:
+ /* reset the chips state over and over.. */
+@@ -889,6 +1077,44 @@
+ }
+ return ide_dmaproc(func, drive); /* use standard DMA stuff */
+ }
++
++int hpt374_dmaproc (ide_dma_action_t func, ide_drive_t *drive)
++{
++ ide_hwif_t *hwif = HWIF(drive);
++ unsigned long dma_base = hwif->dma_base;
++ byte mscreg = hwif->channel ? 0x54 : 0x50;
++// byte reginfo = hwif->channel ? 0x56 : 0x52;
++ byte dma_stat;
++
++ switch (func) {
++ case ide_dma_check:
++ return config_drive_xfer_rate(drive);
++ case ide_dma_test_irq: /* returns 1 if dma irq issued, 0 otherwise */
++ dma_stat = inb(dma_base+2);
++#if 0 /* do not set unless you know what you are doing */
++ if (dma_stat & 4) {
++ byte stat = GET_STAT();
++ outb(dma_base+2, dma_stat & 0xE4);
++ }
++#endif
++ /* return 1 if INTR asserted */
++ return (dma_stat & 4) == 4;
++ case ide_dma_end:
++ {
++ byte bwsr_mask = hwif->channel ? 0x02 : 0x01;
++ byte bwsr_stat, msc_stat;
++ pci_read_config_byte(hwif->pci_dev, 0x6a, &bwsr_stat);
++ pci_read_config_byte(hwif->pci_dev, mscreg, &msc_stat);
++ if ((bwsr_stat & bwsr_mask) == bwsr_mask)
++ pci_write_config_byte(hwif->pci_dev, mscreg, msc_stat|0x30);
++ }
++ default:
++ break;
++ }
++ return ide_dmaproc(func, drive); /* use standard DMA stuff */
++}
++
++
+ #endif /* CONFIG_BLK_DEV_IDEDMA */
+
+ /*
+@@ -1006,6 +1232,184 @@
+ return 0;
+ }
+
++static void __init init_hpt374(struct pci_dev *dev)
++{
++ int adjust, i;
++ u16 freq;
++ u32 pll;
++ byte reg5bh;
++
++ /*
++ * default to pci clock. make sure MA15/16 are set to output
++ * to prevent drives having problems with 40-pin cables.
++ */
++ pci_write_config_byte(dev, 0x5b, 0x23);
++
++ /*
++ * set up the PLL. we need to adjust it so that it's stable.
++ * freq = Tpll * 192 / Tpci
++ */
++ pci_read_config_word(dev, 0x78, &freq);
++ freq &= 0x1FF;
++ if (freq < 0x9c) {
++ pll = F_LOW_PCI_33;
++ dev->sysdata = (void *) thirty_three_base_hpt374;
++ printk("HPT374: using 33MHz PCI clock\n");
++ } else if (freq < 0xb0) {
++ pll = F_LOW_PCI_40;
++ } else if (freq < 0xc8) {
++ pll = F_LOW_PCI_50;
++// dev->sysdata = (void *) fifty_base_hpt374;
++ printk("HPT374: using 50MHz PCI clock\n");
++ } else {
++ pll = F_LOW_PCI_66;
++// dev->sysdata = (void *) sixty_six_base_hpt374;
++ printk("HPT374: using 66MHz PCI clock\n");
++ }
++
++ /*
++ * only try the pll if we don't have a table for the clock
++ * speed that we're running at. NOTE: the internal PLL will
++ * result in slow reads when using a 33MHz PCI clock. we also
++ * don't like to use the PLL because it will cause glitches
++ * on PRST/SRST when the HPT state engine gets reset.
++ */
++ if (dev->sysdata)
++ goto init_hpt374_done;
++
++ /*
++ * adjust PLL based upon PCI clock, enable it, and wait for
++ * stabilization.
++ */
++ adjust = 0;
++ freq = (pll < F_LOW_PCI_50) ? 2 : 4;
++ while (adjust++ < 6) {
++ pci_write_config_dword(dev, 0x5c, (freq + pll) << 16 |
++ pll | 0x100);
++
++ /* wait for clock stabilization */
++ for (i = 0; i < 0x50000; i++) {
++ pci_read_config_byte(dev, 0x5b, &reg5bh);
++ if (reg5bh & 0x80) {
++ /* spin looking for the clock to destabilize */
++ for (i = 0; i < 0x1000; ++i) {
++ pci_read_config_byte(dev, 0x5b,
++ &reg5bh);
++ if ((reg5bh & 0x80) == 0)
++ goto pll_recal;
++ }
++ pci_read_config_dword(dev, 0x5c, &pll);
++ pci_write_config_dword(dev, 0x5c,
++ pll & ~0x100);
++ pci_write_config_byte(dev, 0x5b, 0x21);
++// dev->sysdata = (void *) fifty_base_hpt374;
++ printk("HPT374: using 50MHz internal PLL\n");
++ goto init_hpt374_done;
++ }
++ }
++pll_recal:
++ if (adjust & 1)
++ pll -= (adjust >> 1);
++ else
++ pll += (adjust >> 1);
++ }
++
++init_hpt374_done:
++ /* reset state engine */
++ pci_write_config_byte(dev, 0x50, 0x36);
++ pci_write_config_byte(dev, 0x54, 0x36);
++ udelay(100);
++}
++
++static void __init init_hpt372(struct pci_dev *dev)
++{
++ int adjust, i;
++ u16 freq;
++ u32 pll;
++ byte reg5bh;
++
++ /*
++ * default to pci clock. make sure MA15/16 are set to output
++ * to prevent drives having problems with 40-pin cables.
++ */
++ pci_write_config_byte(dev, 0x5b, 0x23);
++
++ /*
++ * set up the PLL. we need to adjust it so that it's stable.
++ * freq = Tpll * 192 / Tpci
++ */
++ pci_read_config_word(dev, 0x78, &freq);
++ freq &= 0x1FF;
++ if (freq < 0x9c) {
++ pll = F_LOW_PCI_33;
++ dev->sysdata = (void *) thirty_three_base_hpt372;
++ printk("HPT372: using 33MHz PCI clock\n");
++ } else if (freq < 0xb0) {
++ pll = F_LOW_PCI_40;
++ } else if (freq < 0xc8) {
++ pll = F_LOW_PCI_50;
++ dev->sysdata = (void *) fifty_base_hpt372;
++ printk("HPT372: using 50MHz PCI clock\n");
++ } else {
++ pll = F_LOW_PCI_66;
++ dev->sysdata = (void *) sixty_six_base_hpt372;
++ printk("HPT372: using 66MHz PCI clock\n");
++ }
++
++ /*
++ * only try the pll if we don't have a table for the clock
++ * speed that we're running at. NOTE: the internal PLL will
++ * result in slow reads when using a 33MHz PCI clock. we also
++ * don't like to use the PLL because it will cause glitches
++ * on PRST/SRST when the HPT state engine gets reset.
++ */
++ if (dev->sysdata)
++ goto init_hpt372_done;
++
++ /*
++ * adjust PLL based upon PCI clock, enable it, and wait for
++ * stabilization.
++ */
++ adjust = 0;
++ freq = (pll < F_LOW_PCI_50) ? 2 : 4;
++ while (adjust++ < 6) {
++ pci_write_config_dword(dev, 0x5c, (freq + pll) << 16 |
++ pll | 0x100);
++
++ /* wait for clock stabilization */
++ for (i = 0; i < 0x50000; i++) {
++ pci_read_config_byte(dev, 0x5b, &reg5bh);
++ if (reg5bh & 0x80) {
++ /* spin looking for the clock to destabilize */
++ for (i = 0; i < 0x1000; ++i) {
++ pci_read_config_byte(dev, 0x5b,
++ &reg5bh);
++ if ((reg5bh & 0x80) == 0)
++ goto pll_recal;
++ }
++ pci_read_config_dword(dev, 0x5c, &pll);
++ pci_write_config_dword(dev, 0x5c,
++ pll & ~0x100);
++ pci_write_config_byte(dev, 0x5b, 0x21);
++ dev->sysdata = (void *) fifty_base_hpt372;
++ printk("HPT372: using 50MHz internal PLL\n");
++ goto init_hpt372_done;
++ }
++ }
++pll_recal:
++ if (adjust & 1)
++ pll -= (adjust >> 1);
++ else
++ pll += (adjust >> 1);
++ }
++
++init_hpt372_done:
++ /* reset state engine */
++ pci_write_config_byte(dev, 0x50, 0x37);
++ pci_write_config_byte(dev, 0x54, 0x37);
++ udelay(100);
++}
++
+ static void __init init_hpt370(struct pci_dev *dev)
+ {
+ int adjust, i;
+@@ -1095,6 +1499,34 @@
+ udelay(100);
+ }
+
++static void __init init_hpt366 (struct pci_dev *dev)
++{
++ unsigned int reg1 = 0;
++ byte drive_fast = 0;
++
++ /*
++ * Disable the "fast interrupt" prediction.
++ */
++ pci_read_config_byte(dev, 0x51, &drive_fast);
++ if (drive_fast & 0x02)
++ pci_write_config_byte(dev, 0x51, drive_fast & ~0x20);
++ pci_read_config_dword(dev, 0x40, &reg1);
++
++ /* detect bus speed by looking at control reg timing: */
++ switch((reg1 >> 8) & 7) {
++ case 5:
++ dev->sysdata = (void *) forty_base_hpt366;
++ break;
++ case 9:
++ dev->sysdata = (void *) twenty_five_base_hpt366;
++ break;
++ case 7:
++ default:
++ dev->sysdata = (void *) thirty_three_base_hpt366;
++ break;
++ }
++}
++
+ unsigned int __init pci_init_hpt366 (struct pci_dev *dev, const char *name)
+ {
+ byte test = 0;
+@@ -1118,13 +1550,29 @@
+ if (test != 0x08)
+ pci_write_config_byte(dev, PCI_MAX_LAT, 0x08);
+
+- if (pci_rev_check_hpt3xx(dev)) {
+- init_hpt370(dev);
+- hpt_devs[n_hpt_devs++] = dev;
+- } else {
+- hpt_devs[n_hpt_devs++] = dev;
++ if(n_hpt_devs < HPT366_MAX_DEVS)
++ {
++ if (pci_rev7_check_hpt3xx(dev)) {
++ init_hpt374(dev);
++ hpt_devs[n_hpt_devs++] = dev;
++ } else if (pci_rev5_check_hpt3xx(dev)) {
++ init_hpt372(dev);
++ hpt_devs[n_hpt_devs++] = dev;
++ } else if (pci_rev3_check_hpt3xx(dev)) {
++ init_hpt370(dev);
++ hpt_devs[n_hpt_devs++] = dev;
++ } else {
++ if (dev->device == PCI_DEVICE_ID_TTI_HPT372) {
++ init_hpt372(dev);
++ hpt_devs[n_hpt_devs++] = dev;
++ } else {
++ init_hpt366(dev);
++ hpt_devs[n_hpt_devs++] = dev;
++ }
++ }
+ }
+-
++ else printk(KERN_ERR "hpt366: Too many highpoint controllers.\n");
++
+ #if defined(DISPLAY_HPT366_TIMINGS) && defined(CONFIG_PROC_FS)
+ if (!hpt366_proc) {
+ hpt366_proc = 1;
+@@ -1151,8 +1599,6 @@
+
+ void __init ide_init_hpt366 (ide_hwif_t *hwif)
+ {
+- int hpt_rev;
+-
+ hwif->tuneproc = &hpt3xx_tune_drive;
+ hwif->speedproc = &hpt3xx_tune_chipset;
+ hwif->quirkproc = &hpt3xx_quirkproc;
+@@ -1165,47 +1611,61 @@
+ hwif->serialized = hwif->mate->serialized = 1;
+ #endif
+
+- hpt_rev = pci_rev_check_hpt3xx(hwif->pci_dev);
+- if (hpt_rev) {
+- /* set up ioctl for power status. note: power affects both
+- * drives on each channel */
+- hwif->busproc = &hpt370_busproc;
+- }
+-
+- if (pci_rev2_check_hpt3xx(hwif->pci_dev)) {
+- /* do nothing now but will split device types */
+- hwif->resetproc = &hpt3xx_reset;
+-/*
+- * don't do until we can parse out the cobalt box argh ...
+- * hwif->busproc = &hpt3xx_tristate;
+- */
+- }
+-
+ #ifdef CONFIG_BLK_DEV_IDEDMA
+- if (hwif->dma_base) {
+- if (hpt_rev) {
++ if ((pci_rev3_check_hpt3xx(hwif->pci_dev)) ||
++ (hwif->pci_dev->device == PCI_DEVICE_ID_TTI_HPT372)) {
++ if (hwif->dma_base) {
+ byte reg5ah = 0;
+ pci_read_config_byte(hwif->pci_dev, 0x5a, &reg5ah);
+- if (reg5ah & 0x10) /* interrupt force enable */
+- pci_write_config_byte(hwif->pci_dev, 0x5a, reg5ah & ~0x10);
+- hwif->dmaproc = &hpt370_dmaproc;
+- } else {
+- hwif->dmaproc = &hpt366_dmaproc;
++ reg5ah &= ~0x10;
++ pci_write_config_byte(hwif->pci_dev, 0x5a, reg5ah);
+ }
+- if (!noautodma)
+- hwif->autodma = 1;
+- else
+- hwif->autodma = 0;
++ }
++#endif /* CONFIG_BLK_DEV_IDEDMA */
++
++ if (pci_rev7_check_hpt3xx(hwif->pci_dev)) {
++ hwif->resetproc = &hpt3xx_reset;
++ hwif->busproc = &hpt370_busproc;
++#ifdef CONFIG_BLK_DEV_IDEDMA
++ if (hwif->dma_base)
++ hwif->dmaproc = &hpt374_dmaproc;
++#endif /* CONFIG_BLK_DEV_IDEDMA */
++ } else if (pci_rev5_check_hpt3xx(hwif->pci_dev)) {
++ hwif->resetproc = &hpt3xx_reset;
++ hwif->busproc = &hpt370_busproc;
++#ifdef CONFIG_BLK_DEV_IDEDMA
++ if (hwif->dma_base)
++ hwif->dmaproc = &hpt374_dmaproc;
++#endif /* CONFIG_BLK_DEV_IDEDMA */
++ } else if (pci_rev3_check_hpt3xx(hwif->pci_dev)) {
++ /*
++ * set up ioctl for power status. note: power affects both
++ * drives on each channel
++ */
++ hwif->resetproc = &hpt3xx_reset;
++ hwif->busproc = &hpt370_busproc;
++#ifdef CONFIG_BLK_DEV_IDEDMA
++ if (hwif->dma_base)
++ hwif->dmaproc = &hpt370_dmaproc;
++#endif /* CONFIG_BLK_DEV_IDEDMA */
++ } else if (hwif->pci_dev->device == PCI_DEVICE_ID_TTI_HPT372) {
++ hwif->resetproc = &hpt3xx_reset;
++ hwif->busproc = &hpt370_busproc;
++#ifdef CONFIG_BLK_DEV_IDEDMA
++ if (hwif->dma_base)
++ hwif->dmaproc = &hpt370_dmaproc;
++#endif /* CONFIG_BLK_DEV_IDEDMA */
+ } else {
+- hwif->autodma = 0;
+- hwif->drives[0].autotune = 1;
+- hwif->drives[1].autotune = 1;
+- }
+-#else /* !CONFIG_BLK_DEV_IDEDMA */
+- hwif->drives[0].autotune = 1;
+- hwif->drives[1].autotune = 1;
+- hwif->autodma = 0;
++ /* do nothing now but will split device types */
++ hwif->resetproc = &hpt3xx_reset;
++ hwif->busproc = &hpt3xx_tristate;
++#ifdef CONFIG_BLK_DEV_IDEDMA
++ if (hwif->dma_base)
++ hwif->dmaproc = &hpt366_dmaproc;
+ #endif /* CONFIG_BLK_DEV_IDEDMA */
++ }
++ hwif->drives[0].autotune = (hwif->autodma) ? 0 : 1;
++ hwif->drives[1].autotune = (hwif->autodma) ? 0 : 1;
+ }
+
+ void __init ide_dmacapable_hpt366 (ide_hwif_t *hwif, unsigned long dmabase)