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 >
2010-01-18 10:19:22 +03:00
* Copyright ( C ) 2007 - 2010 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/pci.h>
# include <linux/ide.h>
# include <linux/init.h>
# include <asm/io.h>
2008-07-25 00:53:32 +04:00
# define DRV_NAME "serverworks"
2005-04-17 02:20:36 +04:00
# 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 int check_in_drive_lists ( ide_drive_t * drive , const char * * list )
{
2008-10-11 00:39:19 +04:00
char * m = ( char * ) & drive - > id [ ATA_ID_PROD ] ;
2005-04-17 02:20:36 +04:00
while ( * list )
2008-10-11 00:39:19 +04:00
if ( ! strcmp ( * list + + , m ) )
2005-04-17 02:20:36 +04:00
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
{
2008-02-02 01:09:31 +03:00
struct pci_dev * dev = to_pci_dev ( drive - > hwif - > dev ) ;
2005-04-17 02:20:36 +04:00
2010-01-18 10:19:22 +03:00
if ( dev - > device = = PCI_DEVICE_ID_SERVERWORKS_HT1000IDE ) {
2007-05-10 02:01:08 +04:00
return 0x1f ;
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 ;
2010-01-18 10:19:22 +03:00
} else {
u8 btr = 0 , mode , mask ;
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 ) {
2007-10-17 00:29:52 +04:00
case 3 : mask = 0x3f ; break ;
2007-05-10 02:01:08 +04:00
case 2 : mask = 0x1f ; break ;
case 1 : mask = 0x07 ; break ;
default : mask = 0x00 ; break ;
}
2010-01-18 10:19:22 +03:00
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
2010-01-19 12:44:41 +03:00
static void svwks_set_pio_mode ( ide_hwif_t * hwif , ide_drive_t * drive )
2007-07-20 03:11:56 +04:00
{
static const u8 pio_modes [ ] = { 0x5d , 0x47 , 0x34 , 0x22 , 0x20 } ;
static const u8 drive_pci [ ] = { 0x41 , 0x40 , 0x43 , 0x42 } ;
2010-01-19 12:44:41 +03:00
struct pci_dev * dev = to_pci_dev ( hwif - > dev ) ;
const u8 pio = drive - > pio_mode - XFER_PIO_0 ;
2007-07-20 03:11:56 +04:00
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 ) ;
}
}
2010-01-19 12:45:29 +03:00
static void svwks_set_dma_mode ( ide_hwif_t * hwif , ide_drive_t * drive )
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
2008-02-02 01:09:31 +03:00
struct pci_dev * dev = to_pci_dev ( hwif - > dev ) ;
2010-01-19 12:45:29 +03:00
const u8 speed = drive - > dma_mode ;
2008-10-13 23:39:40 +04:00
u8 unit = drive - > dn & 1 ;
2007-07-20 03:11:56 +04:00
u8 ultra_enable = 0 , ultra_timing = 0 , dma_timing = 0 ;
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 ) ;
2008-01-26 22:12:59 +03:00
if ( speed > = XFER_UDMA_0 ) {
dma_timing | = dma_modes [ 2 ] ;
ultra_timing | = ( udma_modes [ speed - XFER_UDMA_0 ] < < ( 4 * unit ) ) ;
ultra_enable | = ( 0x01 < < drive - > dn ) ;
} else if ( speed > = XFER_MW_DMA_0 )
dma_timing | = dma_modes [ speed - XFER_MW_DMA_0 ] ;
2005-04-17 02:20:36 +04:00
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 ) ;
}
2009-03-25 01:22:53 +03:00
static int init_chipset_svwks ( struct pci_dev * dev )
2005-04-17 02:20:36 +04:00
{
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 ) {
2010-01-18 10:19:22 +03:00
struct pci_dev * isa_dev =
pci_get_device ( PCI_VENDOR_ID_SERVERWORKS ,
PCI_DEVICE_ID_SERVERWORKS_OSB4 , NULL ) ;
2005-04-17 02:20:36 +04:00
if ( isa_dev ) {
pci_read_config_dword ( isa_dev , 0x64 , & reg ) ;
reg & = ~ 0x00002000 ; /* disable 600ns interrupt mask */
if ( ! ( reg & 0x00004000 ) )
2008-07-25 00:53:33 +04:00
printk ( KERN_DEBUG DRV_NAME " %s: UDMA not BIOS "
" enabled. \n " , pci_name ( dev ) ) ;
2005-04-17 02:20:36 +04:00
reg | = 0x00004000 ; /* enable UDMA/33 support */
pci_write_config_dword ( isa_dev , 0x64 , reg ) ;
2010-01-18 10:19:30 +03:00
pci_dev_put ( isa_dev ) ;
2005-04-17 02:20:36 +04:00
}
}
/* 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
2009-03-25 01:22:53 +03:00
return 0 ;
2005-04-17 02:20:36 +04:00
}
2008-08-05 20:17:04 +04:00
static u8 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 .
*/
2008-08-05 20:17:04 +04:00
static u8 ata66_svwks_dell ( ide_hwif_t * hwif )
2005-04-17 02:20:36 +04:00
{
2008-02-02 01:09:31 +03:00
struct pci_dev * dev = to_pci_dev ( hwif - > dev ) ;
2005-04-17 02:20:36 +04:00
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 !
*/
2008-08-05 20:17:04 +04:00
static u8 ata66_svwks_cobalt ( ide_hwif_t * hwif )
2005-04-17 02:20:36 +04:00
{
2008-02-02 01:09:31 +03:00
struct pci_dev * dev = to_pci_dev ( hwif - > dev ) ;
2005-04-17 02:20:36 +04:00
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
}
2008-08-05 20:17:04 +04:00
static u8 svwks_cable_detect ( ide_hwif_t * hwif )
2005-04-17 02:20:36 +04:00
{
2008-02-02 01:09:31 +03:00
struct pci_dev * dev = to_pci_dev ( hwif - > dev ) ;
2005-04-17 02:20:36 +04:00
/* 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
}
2008-04-27 00:25:14 +04:00
static const struct ide_port_ops osb4_port_ops = {
. set_pio_mode = svwks_set_pio_mode ,
. set_dma_mode = svwks_set_dma_mode ,
} ;
2005-04-17 02:20:36 +04:00
2008-04-27 00:25:14 +04:00
static const struct ide_port_ops svwks_port_ops = {
. set_pio_mode = svwks_set_pio_mode ,
. set_dma_mode = svwks_set_dma_mode ,
. udma_filter = svwks_udma_filter ,
. cable_detect = svwks_cable_detect ,
} ;
2005-04-17 02:20:36 +04:00
2012-12-22 01:21:03 +04:00
static const struct ide_port_info serverworks_chipsets [ ] = {
2008-07-25 00:53:32 +04:00
{ /* 0: OSB4 */
. name = DRV_NAME ,
2005-04-17 02:20:36 +04:00
. init_chipset = init_chipset_svwks ,
2008-04-27 00:25:14 +04:00
. port_ops = & osb4_port_ops ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2007-10-19 02:30:07 +04:00
. mwdma_mask = ATA_MWDMA2 ,
. udma_mask = 0x00 , /* UDMA is problematic on OSB4 */
2008-07-25 00:53:32 +04:00
} ,
{ /* 1: CSB5 */
. name = DRV_NAME ,
2005-04-17 02:20:36 +04:00
. init_chipset = init_chipset_svwks ,
2008-04-27 00:25:14 +04:00
. port_ops = & svwks_port_ops ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2007-10-19 02:30:07 +04:00
. mwdma_mask = ATA_MWDMA2 ,
. udma_mask = ATA_UDMA5 ,
2008-07-25 00:53:32 +04:00
} ,
{ /* 2: CSB6 */
. name = DRV_NAME ,
2005-04-17 02:20:36 +04:00
. init_chipset = init_chipset_svwks ,
2008-04-27 00:25:14 +04:00
. port_ops = & svwks_port_ops ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2007-10-19 02:30:07 +04:00
. mwdma_mask = ATA_MWDMA2 ,
. udma_mask = ATA_UDMA5 ,
2008-07-25 00:53:32 +04:00
} ,
{ /* 3: CSB6-2 */
. name = DRV_NAME ,
2005-04-17 02:20:36 +04:00
. init_chipset = init_chipset_svwks ,
2008-04-27 00:25:14 +04:00
. port_ops = & svwks_port_ops ,
2009-03-25 01:22:52 +03:00
. host_flags = IDE_HFLAG_SINGLE ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2007-10-19 02:30:07 +04:00
. mwdma_mask = ATA_MWDMA2 ,
. udma_mask = ATA_UDMA5 ,
2008-07-25 00:53:32 +04:00
} ,
{ /* 4: HT1000 */
. name = DRV_NAME ,
2005-08-19 00:30:35 +04:00
. init_chipset = init_chipset_svwks ,
2008-04-27 00:25:14 +04:00
. port_ops = & svwks_port_ops ,
2009-03-25 01:22:52 +03:00
. host_flags = IDE_HFLAG_SINGLE ,
2007-07-20 03:11:59 +04:00
. pio_mask = ATA_PIO4 ,
2007-10-19 02:30:07 +04:00
. mwdma_mask = ATA_MWDMA2 ,
. udma_mask = ATA_UDMA5 ,
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 .
*/
2012-12-22 01:21:03 +04:00
static int svwks_init_one ( struct pci_dev * dev , const struct pci_device_id * id )
2005-04-17 02:20:36 +04:00
{
2007-10-20 02:32:34 +04:00
struct ide_port_info d ;
2007-10-19 02:30:09 +04:00
u8 idx = id - > driver_data ;
d = serverworks_chipsets [ idx ] ;
2008-02-02 01:09:30 +03:00
if ( idx = = 1 )
d . host_flags | = IDE_HFLAG_CLEAR_SIMPLEX ;
else if ( idx = = 2 | | idx = = 3 ) {
2007-10-19 02:30:09 +04:00
if ( ( PCI_FUNC ( dev - > devfn ) & 1 ) = = 0 ) {
if ( pci_resource_start ( dev , 0 ) ! = 0x01f1 )
2008-04-26 19:36:35 +04:00
d . host_flags | = IDE_HFLAG_NON_BOOTABLE ;
2007-10-19 02:30:09 +04:00
d . host_flags | = IDE_HFLAG_SINGLE ;
} else
d . host_flags & = ~ IDE_HFLAG_SINGLE ;
}
2005-04-17 02:20:36 +04:00
2008-07-25 00:53:14 +04:00
return ide_pci_init_one ( dev , & d , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-17 00:29:56 +04:00
static const struct pci_device_id svwks_pci_tbl [ ] = {
{ PCI_VDEVICE ( SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_OSB4IDE ) , 0 } ,
{ PCI_VDEVICE ( SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_CSB5IDE ) , 1 } ,
{ PCI_VDEVICE ( SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_CSB6IDE ) , 2 } ,
{ PCI_VDEVICE ( SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_CSB6IDE2 ) , 3 } ,
{ PCI_VDEVICE ( SERVERWORKS , PCI_DEVICE_ID_SERVERWORKS_HT1000IDE ) , 4 } ,
2005-04-17 02:20:36 +04:00
{ 0 , } ,
} ;
MODULE_DEVICE_TABLE ( pci , svwks_pci_tbl ) ;
2008-10-13 23:39:41 +04:00
static struct pci_driver svwks_pci_driver = {
2005-04-17 02:20:36 +04:00
. name = " Serverworks_IDE " ,
. id_table = svwks_pci_tbl ,
. probe = svwks_init_one ,
2008-07-25 00:53:25 +04:00
. remove = ide_pci_remove ,
2008-10-11 00:39:32 +04:00
. suspend = ide_pci_suspend ,
. resume = ide_pci_resume ,
2005-04-17 02:20:36 +04:00
} ;
2007-01-27 15:46:56 +03:00
static int __init svwks_ide_init ( void )
2005-04-17 02:20:36 +04:00
{
2008-10-13 23:39:41 +04:00
return ide_pci_register_driver ( & svwks_pci_driver ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-25 00:53:25 +04:00
static void __exit svwks_ide_exit ( void )
{
2008-10-13 23:39:41 +04:00
pci_unregister_driver ( & svwks_pci_driver ) ;
2008-07-25 00:53:25 +04:00
}
2005-04-17 02:20:36 +04:00
module_init ( svwks_ide_init ) ;
2008-07-25 00:53:25 +04:00
module_exit ( svwks_ide_exit ) ;
2005-04-17 02:20:36 +04:00
2010-01-18 10:19:22 +03:00
MODULE_AUTHOR ( " Michael Aubry. Andrzej Krzysztofowicz, Andre Hedrick, Bartlomiej Zolnierkiewicz " ) ;
2005-04-17 02:20:36 +04:00
MODULE_DESCRIPTION ( " PCI driver module for Serverworks OSB4/CSB5/CSB6 IDE " ) ;
MODULE_LICENSE ( " GPL " ) ;