2005-04-17 02:20:36 +04:00
/*
2007-07-20 03:11:56 +04:00
* linux / drivers / ide / pci / serverworks . c Version 0.22 Jun 27 2007
2005-04-17 02:20:36 +04:00
*
* Copyright ( C ) 1998 - 2000 Michel Aubry
* Copyright ( C ) 1998 - 2000 Andrzej Krzysztofowicz
* Copyright ( C ) 1998 - 2000 Andre Hedrick < andre @ linux - ide . org >
2007-05-16 02:51:42 +04:00
* Copyright ( C ) 2007 Bartlomiej Zolnierkiewicz
2005-04-17 02:20:36 +04:00
* Portions copyright ( c ) 2001 Sun Microsystems
*
*
* RCC / ServerWorks IDE driver for Linux
*
* OSB4 : ` Open South Bridge ' IDE Interface ( fn 1 )
* supports UDMA mode 2 ( 33 MB / s )
*
* CSB5 : ` Champion South Bridge ' IDE Interface ( fn 1 )
* all revisions support UDMA mode 4 ( 66 MB / s )
* revision A2 .0 and up support UDMA mode 5 ( 100 MB / s )
*
* * * * The CSB5 does not provide ANY register * * *
* * * * to detect 80 - conductor cable presence . * * *
*
* CSB6 : ` Champion South Bridge ' IDE Interface ( optional : third channel )
*
2005-08-19 00:30:35 +04:00
* HT1000 : AKA BCM5785 - Hypertransport Southbridge for Opteron systems . IDE
* controller same as the CSB6 . Single channel ATA100 only .
*
2005-04-17 02:20:36 +04:00
* Documentation :
* Available under NDA only . Errata info very hard to get .
*
*/
# include <linux/types.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/ioport.h>
# include <linux/pci.h>
# include <linux/hdreg.h>
# include <linux/ide.h>
# include <linux/init.h>
# include <linux/delay.h>
# include <asm/io.h>
# define SVWKS_CSB5_REVISION_NEW 0x92 /* min PCI_REVISION_ID for UDMA5 (A2.0) */
# define SVWKS_CSB6_REVISION 0xa0 /* min PCI_REVISION_ID for UDMA4 (A1.0) */
/* Seagate Barracuda ATA IV Family drives in UDMA mode 5
* can overrun their FIFOs when used with the CSB5 */
static const char * svwks_bad_ata100 [ ] = {
" ST320011A " ,
" ST340016A " ,
" ST360021A " ,
" ST380021A " ,
NULL
} ;
static struct pci_dev * isa_dev ;
static int check_in_drive_lists ( ide_drive_t * drive , const char * * list )
{
while ( * list )
if ( ! strcmp ( * list + + , drive - > id - > model ) )
return 1 ;
return 0 ;
}
2007-05-10 02:01:08 +04:00
static u8 svwks_udma_filter ( ide_drive_t * drive )
2005-04-17 02:20:36 +04:00
{
struct pci_dev * dev = HWIF ( drive ) - > pci_dev ;
2007-05-10 02:01:08 +04:00
u8 mask = 0 ;
2005-04-17 02:20:36 +04:00
2005-08-19 00:30:35 +04:00
if ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_HT1000IDE )
2007-05-10 02:01:08 +04:00
return 0x1f ;
2005-04-17 02:20:36 +04:00
if ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_OSB4IDE ) {
u32 reg = 0 ;
if ( isa_dev )
pci_read_config_dword ( isa_dev , 0x64 , & reg ) ;
/*
* Don ' t enable UDMA on disk devices for the moment
*/
if ( drive - > media = = ide_disk )
return 0 ;
/* Check the OSB4 DMA33 enable bit */
2007-05-10 02:01:08 +04:00
return ( ( reg & 0x00004000 ) = = 0x00004000 ) ? 0x07 : 0 ;
2007-06-09 02:46:36 +04:00
} else if ( dev - > revision < SVWKS_CSB5_REVISION_NEW ) {
2007-05-10 02:01:08 +04:00
return 0x07 ;
2007-06-09 02:46:36 +04:00
} else if ( dev - > revision > = SVWKS_CSB5_REVISION_NEW ) {
2007-05-10 02:01:08 +04:00
u8 btr = 0 , mode ;
2005-04-17 02:20:36 +04:00
pci_read_config_byte ( dev , 0x5A , & btr ) ;
mode = btr & 0x3 ;
2007-05-10 02:01:08 +04:00
2005-04-17 02:20:36 +04:00
/* If someone decides to do UDMA133 on CSB5 the same
issue will bite so be inclusive */
if ( mode > 2 & & check_in_drive_lists ( drive , svwks_bad_ata100 ) )
mode = 2 ;
2007-05-10 02:01:08 +04:00
switch ( mode ) {
case 2 : mask = 0x1f ; break ;
case 1 : mask = 0x07 ; break ;
default : mask = 0x00 ; break ;
}
2005-04-17 02:20:36 +04:00
}
if ( ( ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE ) | |
( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE2 ) ) & &
( ! ( PCI_FUNC ( dev - > devfn ) & 1 ) ) )
2007-05-10 02:01:08 +04:00
mask = 0x1f ;
return mask ;
2005-04-17 02:20:36 +04:00
}
static u8 svwks_csb_check ( struct pci_dev * dev )
{
switch ( dev - > device ) {
case PCI_DEVICE_ID_SERVERWORKS_CSB5IDE :
case PCI_DEVICE_ID_SERVERWORKS_CSB6IDE :
case PCI_DEVICE_ID_SERVERWORKS_CSB6IDE2 :
2005-08-19 00:30:35 +04:00
case PCI_DEVICE_ID_SERVERWORKS_HT1000IDE :
2005-04-17 02:20:36 +04:00
return 1 ;
default :
break ;
}
return 0 ;
}
2007-07-20 03:11:56 +04:00
static void svwks_tune_pio ( ide_drive_t * drive , const u8 pio )
{
static const u8 pio_modes [ ] = { 0x5d , 0x47 , 0x34 , 0x22 , 0x20 } ;
static const u8 drive_pci [ ] = { 0x41 , 0x40 , 0x43 , 0x42 } ;
struct pci_dev * dev = drive - > hwif - > pci_dev ;
pci_write_config_byte ( dev , drive_pci [ drive - > dn ] , pio_modes [ pio ] ) ;
if ( svwks_csb_check ( dev ) ) {
u16 csb_pio = 0 ;
pci_read_config_word ( dev , 0x4a , & csb_pio ) ;
csb_pio & = ~ ( 0x0f < < ( 4 * drive - > dn ) ) ;
csb_pio | = ( pio < < ( 4 * drive - > dn ) ) ;
pci_write_config_word ( dev , 0x4a , csb_pio ) ;
}
}
2007-10-12 01:53:59 +04:00
static int svwks_tune_chipset ( ide_drive_t * drive , const u8 speed )
2005-04-17 02:20:36 +04:00
{
2006-06-28 15:27:02 +04:00
static const u8 udma_modes [ ] = { 0x00 , 0x01 , 0x02 , 0x03 , 0x04 , 0x05 } ;
static const u8 dma_modes [ ] = { 0x77 , 0x21 , 0x20 } ;
static const u8 drive_pci2 [ ] = { 0x45 , 0x44 , 0x47 , 0x46 } ;
2005-04-17 02:20:36 +04:00
ide_hwif_t * hwif = HWIF ( drive ) ;
struct pci_dev * dev = hwif - > pci_dev ;
u8 unit = ( drive - > select . b . unit & 0x01 ) ;
2007-07-20 03:11:56 +04:00
u8 ultra_enable = 0 , ultra_timing = 0 , dma_timing = 0 ;
if ( speed > = XFER_PIO_0 & & speed < = XFER_PIO_4 ) {
svwks_tune_pio ( drive , speed - XFER_PIO_0 ) ;
return ide_config_drive_speed ( drive , speed ) ;
}
2005-04-17 02:20:36 +04:00
/* If we are about to put a disk into UDMA mode we screwed up.
Our code assumes we never _ever_ do this on an OSB4 */
if ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_OSB4 & &
drive - > media = = ide_disk & & speed > = XFER_UDMA_0 )
BUG ( ) ;
2007-07-10 01:17:53 +04:00
2005-04-17 02:20:36 +04:00
pci_read_config_byte ( dev , ( 0x56 | hwif - > channel ) , & ultra_timing ) ;
pci_read_config_byte ( dev , 0x54 , & ultra_enable ) ;
ultra_timing & = ~ ( 0x0F < < ( 4 * unit ) ) ;
ultra_enable & = ~ ( 0x01 < < drive - > dn ) ;
switch ( speed ) {
case XFER_MW_DMA_2 :
case XFER_MW_DMA_1 :
case XFER_MW_DMA_0 :
dma_timing | = dma_modes [ speed - XFER_MW_DMA_0 ] ;
break ;
case XFER_UDMA_5 :
case XFER_UDMA_4 :
case XFER_UDMA_3 :
case XFER_UDMA_2 :
case XFER_UDMA_1 :
case XFER_UDMA_0 :
dma_timing | = dma_modes [ 2 ] ;
ultra_timing | = ( ( udma_modes [ speed - XFER_UDMA_0 ] ) < < ( 4 * unit ) ) ;
ultra_enable | = ( 0x01 < < drive - > dn ) ;
default :
break ;
}
pci_write_config_byte ( dev , drive_pci2 [ drive - > dn ] , dma_timing ) ;
pci_write_config_byte ( dev , ( 0x56 | hwif - > channel ) , ultra_timing ) ;
pci_write_config_byte ( dev , 0x54 , ultra_enable ) ;
return ( ide_config_drive_speed ( drive , speed ) ) ;
}
static void svwks_tune_drive ( ide_drive_t * drive , u8 pio )
{
2007-07-20 03:11:58 +04:00
pio = ide_get_best_pio_mode ( drive , pio , 4 ) ;
2007-07-20 03:11:56 +04:00
svwks_tune_pio ( drive , pio ) ;
( void ) ide_config_drive_speed ( drive , XFER_PIO_0 + pio ) ;
2005-04-17 02:20:36 +04:00
}
static int svwks_config_drive_xfer_rate ( ide_drive_t * drive )
{
drive - > init_speed = 0 ;
2007-05-16 02:51:43 +04:00
if ( ide_tune_dma ( drive ) )
2007-02-17 04:40:26 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
2007-02-17 04:40:25 +03:00
if ( ide_use_fast_pio ( drive ) )
2007-05-16 02:51:42 +04:00
svwks_tune_drive ( drive , 255 ) ;
2007-02-17 04:40:25 +03:00
2007-02-17 04:40:26 +03:00
return - 1 ;
2005-04-17 02:20:36 +04:00
}
static unsigned int __devinit init_chipset_svwks ( struct pci_dev * dev , const char * name )
{
unsigned int reg ;
u8 btr ;
/* force Master Latency Timer value to 64 PCICLKs */
pci_write_config_byte ( dev , PCI_LATENCY_TIMER , 0x40 ) ;
/* OSB4 : South Bridge and IDE */
if ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_OSB4IDE ) {
2006-10-01 10:27:29 +04:00
isa_dev = pci_get_device ( PCI_VENDOR_ID_SERVERWORKS ,
2005-04-17 02:20:36 +04:00
PCI_DEVICE_ID_SERVERWORKS_OSB4 , NULL ) ;
if ( isa_dev ) {
pci_read_config_dword ( isa_dev , 0x64 , & reg ) ;
reg & = ~ 0x00002000 ; /* disable 600ns interrupt mask */
if ( ! ( reg & 0x00004000 ) )
printk ( KERN_DEBUG " %s: UDMA not BIOS enabled. \n " , name ) ;
reg | = 0x00004000 ; /* enable UDMA/33 support */
pci_write_config_dword ( isa_dev , 0x64 , reg ) ;
}
}
/* setup CSB5/CSB6 : South Bridge and IDE option RAID */
else if ( ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB5IDE ) | |
( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE ) | |
( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE2 ) ) {
/* Third Channel Test */
if ( ! ( PCI_FUNC ( dev - > devfn ) & 1 ) ) {
struct pci_dev * findev = NULL ;
u32 reg4c = 0 ;
2006-10-01 10:27:29 +04:00
findev = pci_get_device ( PCI_VENDOR_ID_SERVERWORKS ,
2005-04-17 02:20:36 +04:00
PCI_DEVICE_ID_SERVERWORKS_CSB5 , NULL ) ;
if ( findev ) {
pci_read_config_dword ( findev , 0x4C , & reg4c ) ;
reg4c & = ~ 0x000007FF ;
reg4c | = 0x00000040 ;
reg4c | = 0x00000020 ;
pci_write_config_dword ( findev , 0x4C , reg4c ) ;
2006-10-01 10:27:29 +04:00
pci_dev_put ( findev ) ;
2005-04-17 02:20:36 +04:00
}
outb_p ( 0x06 , 0x0c00 ) ;
dev - > irq = inb_p ( 0x0c01 ) ;
} else {
struct pci_dev * findev = NULL ;
u8 reg41 = 0 ;
2006-10-01 10:27:29 +04:00
findev = pci_get_device ( PCI_VENDOR_ID_SERVERWORKS ,
2005-04-17 02:20:36 +04:00
PCI_DEVICE_ID_SERVERWORKS_CSB6 , NULL ) ;
if ( findev ) {
pci_read_config_byte ( findev , 0x41 , & reg41 ) ;
reg41 & = ~ 0x40 ;
pci_write_config_byte ( findev , 0x41 , reg41 ) ;
2006-10-01 10:27:29 +04:00
pci_dev_put ( findev ) ;
2005-04-17 02:20:36 +04:00
}
/*
* This is a device pin issue on CSB6 .
* Since there will be a future raid mode ,
* early versions of the chipset require the
* interrupt pin to be set , and it is a compatibility
* mode issue .
*/
if ( ( dev - > class > > 8 ) = = PCI_CLASS_STORAGE_IDE )
dev - > irq = 0 ;
}
// pci_read_config_dword(dev, 0x40, &pioreg)
// pci_write_config_dword(dev, 0x40, 0x99999999);
// pci_read_config_dword(dev, 0x44, &dmareg);
// pci_write_config_dword(dev, 0x44, 0xFFFFFFFF);
/* setup the UDMA Control register
*
* 1. clear bit 6 to enable DMA
* 2. enable DMA modes with bits 0 - 1
* 00 : legacy
* 01 : udma2
* 10 : udma2 / udma4
* 11 : udma2 / udma4 / udma5
*/
pci_read_config_byte ( dev , 0x5A , & btr ) ;
btr & = ~ 0x40 ;
if ( ! ( PCI_FUNC ( dev - > devfn ) & 1 ) )
btr | = 0x2 ;
else
2007-06-09 02:46:36 +04:00
btr | = ( dev - > revision > = SVWKS_CSB5_REVISION_NEW ) ? 0x3 : 0x2 ;
2005-04-17 02:20:36 +04:00
pci_write_config_byte ( dev , 0x5A , btr ) ;
}
2005-08-19 00:30:35 +04:00
/* Setup HT1000 SouthBridge Controller - Single Channel Only */
else if ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_HT1000IDE ) {
pci_read_config_byte ( dev , 0x5A , & btr ) ;
btr & = ~ 0x40 ;
btr | = 0x3 ;
pci_write_config_byte ( dev , 0x5A , btr ) ;
}
2005-04-17 02:20:36 +04:00
2006-06-28 15:27:02 +04:00
return dev - > irq ;
2005-04-17 02:20:36 +04:00
}
2007-07-10 01:17:58 +04:00
static u8 __devinit ata66_svwks_svwks ( ide_hwif_t * hwif )
2005-04-17 02:20:36 +04:00
{
2007-07-10 01:17:58 +04:00
return ATA_CBL_PATA80 ;
2005-04-17 02:20:36 +04:00
}
/* On Dell PowerEdge servers with a CSB5/CSB6, the top two bits
* of the subsystem device ID indicate presence of an 80 - pin cable .
* Bit 15 clear = secondary IDE channel does not have 80 - pin cable .
* Bit 15 set = secondary IDE channel has 80 - pin cable .
* Bit 14 clear = primary IDE channel does not have 80 - pin cable .
* Bit 14 set = primary IDE channel has 80 - pin cable .
*/
2007-07-10 01:17:58 +04:00
static u8 __devinit ata66_svwks_dell ( ide_hwif_t * hwif )
2005-04-17 02:20:36 +04:00
{
struct pci_dev * dev = hwif - > pci_dev ;
if ( dev - > subsystem_vendor = = PCI_VENDOR_ID_DELL & &
dev - > vendor = = PCI_VENDOR_ID_SERVERWORKS & &
( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB5IDE | |
dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE ) )
return ( ( 1 < < ( hwif - > channel + 14 ) ) &
2007-07-10 01:17:58 +04:00
dev - > subsystem_device ) ? ATA_CBL_PATA80 : ATA_CBL_PATA40 ;
return ATA_CBL_PATA40 ;
2005-04-17 02:20:36 +04:00
}
/* Sun Cobalt Alpine hardware avoids the 80-pin cable
* detect issue by attaching the drives directly to the board .
* This check follows the Dell precedent ( how scary is that ? ! )
*
* WARNING : this only works on Alpine hardware !
*/
2007-07-10 01:17:58 +04:00
static u8 __devinit ata66_svwks_cobalt ( ide_hwif_t * hwif )
2005-04-17 02:20:36 +04:00
{
struct pci_dev * dev = hwif - > pci_dev ;
if ( dev - > subsystem_vendor = = PCI_VENDOR_ID_SUN & &
dev - > vendor = = PCI_VENDOR_ID_SERVERWORKS & &
dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB5IDE )
return ( ( 1 < < ( hwif - > channel + 14 ) ) &
2007-07-10 01:17:58 +04:00
dev - > subsystem_device ) ? ATA_CBL_PATA80 : ATA_CBL_PATA40 ;
return ATA_CBL_PATA40 ;
2005-04-17 02:20:36 +04:00
}
2007-07-10 01:17:58 +04:00
static u8 __devinit ata66_svwks ( ide_hwif_t * hwif )
2005-04-17 02:20:36 +04:00
{
struct pci_dev * dev = hwif - > pci_dev ;
/* Server Works */
if ( dev - > subsystem_vendor = = PCI_VENDOR_ID_SERVERWORKS )
return ata66_svwks_svwks ( hwif ) ;
/* Dell PowerEdge */
if ( dev - > subsystem_vendor = = PCI_VENDOR_ID_DELL )
return ata66_svwks_dell ( hwif ) ;
/* Cobalt Alpine */
if ( dev - > subsystem_vendor = = PCI_VENDOR_ID_SUN )
return ata66_svwks_cobalt ( hwif ) ;
2006-06-28 15:27:02 +04:00
/* Per Specified Design by OEM, and ASIC Architect */
if ( ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE ) | |
( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE2 ) )
2007-07-10 01:17:58 +04:00
return ATA_CBL_PATA80 ;
2006-06-28 15:27:02 +04:00
2007-07-10 01:17:58 +04:00
return ATA_CBL_PATA40 ;
2005-04-17 02:20:36 +04:00
}
static void __devinit init_hwif_svwks ( ide_hwif_t * hwif )
{
if ( ! hwif - > irq )
hwif - > irq = hwif - > channel ? 15 : 14 ;
hwif - > tuneproc = & svwks_tune_drive ;
hwif - > speedproc = & svwks_tune_chipset ;
2007-05-10 02:01:08 +04:00
hwif - > udma_filter = & svwks_udma_filter ;
2005-04-17 02:20:36 +04:00
hwif - > atapi_dma = 1 ;
if ( hwif - > pci_dev - > device ! = PCI_DEVICE_ID_SERVERWORKS_OSB4IDE )
hwif - > ultra_mask = 0x3f ;
hwif - > mwdma_mask = 0x07 ;
hwif - > autodma = 0 ;
2007-07-20 03:11:56 +04:00
hwif - > drives [ 0 ] . autotune = 1 ;
hwif - > drives [ 1 ] . autotune = 1 ;
if ( ! hwif - > dma_base )
2005-04-17 02:20:36 +04:00
return ;
hwif - > ide_dma_check = & svwks_config_drive_xfer_rate ;
2007-02-17 04:40:23 +03:00
if ( hwif - > pci_dev - > device ! = PCI_DEVICE_ID_SERVERWORKS_OSB4IDE ) {
2007-07-10 01:17:58 +04:00
if ( hwif - > cbl ! = ATA_CBL_PATA40_SHORT )
hwif - > cbl = ata66_svwks ( hwif ) ;
2007-02-17 04:40:23 +03:00
}
2005-04-17 02:20:36 +04:00
if ( ! noautodma )
hwif - > autodma = 1 ;
2007-07-20 03:11:56 +04:00
hwif - > drives [ 0 ] . autodma = hwif - > drives [ 1 ] . autodma = 1 ;
2005-04-17 02:20:36 +04:00
}
static int __devinit init_setup_svwks ( struct pci_dev * dev , ide_pci_device_t * d )
{
return ide_setup_pci_device ( dev , d ) ;
}
2005-06-28 02:24:29 +04:00
static int __devinit init_setup_csb6 ( struct pci_dev * dev , ide_pci_device_t * d )
2005-04-17 02:20:36 +04:00
{
if ( ! ( PCI_FUNC ( dev - > devfn ) & 1 ) ) {
d - > bootable = NEVER_BOARD ;
if ( dev - > resource [ 0 ] . start = = 0x01f1 )
d - > bootable = ON_BOARD ;
}
2007-07-20 03:11:55 +04:00
if ( ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE | |
dev - > device = = PCI_DEVICE_ID_SERVERWORKS_CSB6IDE2 ) & &
( ! ( PCI_FUNC ( dev - > devfn ) & 1 ) ) )
d - > host_flags | = IDE_HFLAG_SINGLE ;
else
d - > host_flags & = ~ IDE_HFLAG_SINGLE ;
2005-04-17 02:20:36 +04:00
return ide_setup_pci_device ( dev , d ) ;
}
static ide_pci_device_t serverworks_chipsets [ ] __devinitdata = {
{ /* 0 */
. name = " SvrWks OSB4 " ,
. init_setup = init_setup_svwks ,
. init_chipset = init_chipset_svwks ,
. init_hwif = init_hwif_svwks ,
. autodma = AUTODMA ,
. bootable = ON_BOARD ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2005-04-17 02:20:36 +04:00
} , { /* 1 */
. name = " SvrWks CSB5 " ,
. init_setup = init_setup_svwks ,
. init_chipset = init_chipset_svwks ,
. init_hwif = init_hwif_svwks ,
. autodma = AUTODMA ,
. bootable = ON_BOARD ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2005-04-17 02:20:36 +04:00
} , { /* 2 */
. name = " SvrWks CSB6 " ,
. init_setup = init_setup_csb6 ,
. init_chipset = init_chipset_svwks ,
. init_hwif = init_hwif_svwks ,
. autodma = AUTODMA ,
. bootable = ON_BOARD ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2005-04-17 02:20:36 +04:00
} , { /* 3 */
. name = " SvrWks CSB6 " ,
. init_setup = init_setup_csb6 ,
. init_chipset = init_chipset_svwks ,
. init_hwif = init_hwif_svwks ,
. autodma = AUTODMA ,
. bootable = ON_BOARD ,
2007-07-20 03:11:55 +04:00
. host_flags = IDE_HFLAG_SINGLE ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2005-08-19 00:30:35 +04:00
} , { /* 4 */
. name = " SvrWks HT1000 " ,
. init_setup = init_setup_svwks ,
. init_chipset = init_chipset_svwks ,
. init_hwif = init_hwif_svwks ,
. autodma = AUTODMA ,
. bootable = ON_BOARD ,
2007-07-20 03:11:55 +04:00
. host_flags = IDE_HFLAG_SINGLE ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2005-04-17 02:20:36 +04:00
}
} ;
/**
* svwks_init_one - called when a OSB / CSB is found
* @ dev : the svwks device
* @ id : the matching pci id
*
* Called when the PCI registration layer ( or the IDE initialization )
* finds a device matching our IDE device tables .
*/
static int __devinit svwks_init_one ( struct pci_dev * dev , const struct pci_device_id * id )
{
ide_pci_device_t * d = & serverworks_chipsets [ id - > driver_data ] ;
return d - > init_setup ( dev , d ) ;
}
static struct pci_device_id svwks_pci_tbl [ ] = {
2006-09-11 17:45:07 +04:00
{ PCI_VENDOR_ID_SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_OSB4IDE , PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , 0 } ,
{ PCI_VENDOR_ID_SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_CSB5IDE , PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , 1 } ,
{ PCI_VENDOR_ID_SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_CSB6IDE , PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , 2 } ,
{ PCI_VENDOR_ID_SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_CSB6IDE2 , PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , 3 } ,
{ PCI_VENDOR_ID_SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_HT1000IDE , PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , 4 } ,
2005-04-17 02:20:36 +04:00
{ 0 , } ,
} ;
MODULE_DEVICE_TABLE ( pci , svwks_pci_tbl ) ;
static struct pci_driver driver = {
. name = " Serverworks_IDE " ,
. id_table = svwks_pci_tbl ,
. probe = svwks_init_one ,
} ;
2007-01-27 15:46:56 +03:00
static int __init svwks_ide_init ( void )
2005-04-17 02:20:36 +04:00
{
return ide_pci_register_driver ( & driver ) ;
}
module_init ( svwks_ide_init ) ;
MODULE_AUTHOR ( " Michael Aubry. Andrzej Krzysztofowicz, Andre Hedrick " ) ;
MODULE_DESCRIPTION ( " PCI driver module for Serverworks OSB4/CSB5/CSB6 IDE " ) ;
MODULE_LICENSE ( " GPL " ) ;