2005-04-16 15:20:36 -07:00
/*
* Copyright ( C ) 1996 - 1998 Linus Torvalds & authors ( see below )
*/
/*
* Authors :
* Jaromir Koutek < miri @ punknet . cz > ,
* Jan Harkes < jaharkes @ cwi . nl > ,
* Mark Lord < mlord @ pobox . com >
* Some parts of code are from ali14xx . c and from rz1000 . c .
*
* OPTi is trademark of OPTi , Octek is trademark of Octek .
*
* I used docs from OPTi databook , from ftp . opti . com , file 9123 - 0002. ps
* and disassembled / traced setupvic . exe ( DOS program ) .
* It increases kernel code about 2 kB .
* I don ' t have this card no more , but I hope I can get some in case
* of needed development .
* My card is Octek PIDE 1.01 ( on card ) or OPTiViC ( program ) .
* It has a place for a secondary connector in circuit , but nothing
* is there . Also BIOS says no address for
* secondary controller ( see bellow in ide_init_opti621 ) .
* I ' ve only tested this on my system , which only has one disk .
* It ' s Western Digital WDAC2850 , with PIO mode 3. The PCI bus
* is at 20 MHz ( I have DX2 / 80 , I tried PCI at 40 , but I got random
* lockups ) . I tried the OCTEK double speed CD - ROM and
* it does not work ! But I can ' t boot DOS also , so it ' s probably
* hardware fault . I have connected Conner 80 MB , the Seagate 850 MB ( no
* problems ) and Seagate 1 GB ( as slave , WD as master ) . My experiences
* with the third , 1 GB drive : I got 3 MB / s ( hdparm ) , but sometimes
* it slows to about 100 kB / s ! I don ' t know why and I have
* not this drive now , so I can ' t try it again .
* I write this driver because I lost the paper ( " manual " ) with
* settings of jumpers on the card and I have to boot Linux with
* Loadlin except LILO , cause I have to run the setupvic . exe program
* already or I get disk errors ( my test : rpm - Vf
* / usr / X11R6 / bin / XF86_SVGA - or any big file ) .
* Some numbers from hdparm - t / dev / hda :
* Timing buffer - cache reads : 32 MB in 3.02 seconds = 10.60 MB / sec
* Timing buffered disk reads : 16 MB in 5.52 seconds = 2.90 MB / sec
* I have 4 Megs / s before , but I don ' t know why ( maybe changes
* in hdparm test ) .
* After release of 0.1 , I got some successful reports , so it might work .
*
* The main problem with OPTi is that some timings for master
* and slave must be the same . For example , if you have master
* PIO 3 and slave PIO 0 , driver have to set some timings of
2007-10-11 23:54:00 +02:00
* master for PIO 0. Second problem is that opti621_set_pio_mode
2005-04-16 15:20:36 -07:00
* got only one drive to set , but have to set both drives .
* This is solved in compute_pios . If you don ' t set
* the second drive , compute_pios use ide_get_best_pio_mode
* for autoselect mode ( you can change it to PIO 0 , if you want ) .
* If you then set the second drive to another PIO , the old value
* ( automatically selected ) will be overrided by yours .
* There is a 25 / 33 MHz switch in configuration
2008-04-27 15:38:29 +02:00
* register , but driver is written for use at any frequency .
2005-04-16 15:20:36 -07:00
*
* Version 0.1 , Nov 8 , 1996
2008-04-26 17:36:41 +02:00
* by Jaromir Koutek , for 2.1 .8 .
2005-04-16 15:20:36 -07:00
* Initial version of driver .
2008-04-26 17:36:41 +02:00
*
2005-04-16 15:20:36 -07:00
* Version 0.2
* Number 0.2 skipped .
*
* Version 0.3 , Nov 29 , 1997
* by Mark Lord ( probably ) , for 2.1 .68
* Updates for use with new IDE block driver .
*
* Version 0.4 , Dec 14 , 1997
* by Jan Harkes
* Fixed some errors and cleaned the code .
*
* Version 0.5 , Jan 2 , 1998
* by Jaromir Koutek
* Updates for use with ( again ) new IDE block driver .
* Update of documentation .
2008-04-26 17:36:41 +02:00
*
2005-04-16 15:20:36 -07:00
* Version 0.6 , Jan 2 , 1999
* by Jaromir Koutek
* Reversed to version 0.3 of the driver , because
* 0.5 doesn ' t work .
*/
# include <linux/types.h>
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/pci.h>
# include <linux/ide.h>
# include <asm/io.h>
2008-07-24 22:53:32 +02:00
# define DRV_NAME "opti621"
2005-04-16 15:20:36 -07:00
# define READ_REG 0 /* index of Read cycle timing register */
# define WRITE_REG 1 /* index of Write cycle timing register */
# define CNTRL_REG 3 /* index of Control register */
# define STRAP_REG 5 /* index of Strap register */
# define MISC_REG 6 /* index of Miscellaneous register */
static int reg_base ;
2007-10-20 00:32:35 +02:00
static DEFINE_SPINLOCK ( opti621_lock ) ;
2005-04-16 15:20:36 -07:00
/* Write value to register reg, base of register
* is at reg_base ( 0x1f0 primary , 0x170 secondary ,
* if not changed by PCI configuration ) .
* This is from setupvic . exe program .
*/
2007-02-17 02:40:25 +01:00
static void write_reg ( u8 value , int reg )
2005-04-16 15:20:36 -07:00
{
2007-02-17 02:40:25 +01:00
inw ( reg_base + 1 ) ;
inw ( reg_base + 1 ) ;
outb ( 3 , reg_base + 2 ) ;
outb ( value , reg_base + reg ) ;
outb ( 0x83 , reg_base + 2 ) ;
2005-04-16 15:20:36 -07:00
}
/* Read value from register reg, base of register
* is at reg_base ( 0x1f0 primary , 0x170 secondary ,
* if not changed by PCI configuration ) .
* This is from setupvic . exe program .
*/
2007-02-17 02:40:25 +01:00
static u8 read_reg ( int reg )
2005-04-16 15:20:36 -07:00
{
u8 ret = 0 ;
2007-02-17 02:40:25 +01:00
inw ( reg_base + 1 ) ;
inw ( reg_base + 1 ) ;
outb ( 3 , reg_base + 2 ) ;
ret = inb ( reg_base + reg ) ;
outb ( 0x83 , reg_base + 2 ) ;
2005-04-16 15:20:36 -07:00
return ret ;
}
2007-10-11 23:54:00 +02:00
static void opti621_set_pio_mode ( ide_drive_t * drive , const u8 pio )
2005-04-16 15:20:36 -07:00
{
2008-06-15 21:00:22 +02:00
ide_hwif_t * hwif = drive - > hwif ;
2008-10-10 22:39:26 +02:00
ide_drive_t * pair = ide_get_pair_dev ( drive ) ;
2005-04-16 15:20:36 -07:00
unsigned long flags ;
2009-06-15 22:13:44 +02:00
unsigned long mode = XFER_PIO_0 + pio , pair_mode ;
2008-06-15 21:00:22 +02:00
u8 tim , misc , addr_pio = pio , clk ;
/* DRDY is default 2 (by OPTi Databook) */
2008-06-15 21:00:22 +02:00
static const u8 addr_timings [ 2 ] [ 5 ] = {
{ 0x20 , 0x10 , 0x00 , 0x00 , 0x00 } , /* 33 MHz */
{ 0x10 , 0x10 , 0x00 , 0x00 , 0x00 } , /* 25 MHz */
2008-06-15 21:00:22 +02:00
} ;
2008-06-15 21:00:22 +02:00
static const u8 data_rec_timings [ 2 ] [ 5 ] = {
{ 0x5b , 0x45 , 0x32 , 0x21 , 0x20 } , /* 33 MHz */
{ 0x48 , 0x34 , 0x21 , 0x10 , 0x10 } /* 25 MHz */
2008-06-15 21:00:22 +02:00
} ;
2005-04-16 15:20:36 -07:00
2009-06-15 22:13:44 +02:00
ide_set_drivedata ( drive , ( void * ) mode ) ;
2008-06-15 21:00:22 +02:00
2008-10-10 22:39:26 +02:00
if ( pair ) {
2009-06-15 22:13:44 +02:00
pair_mode = ( unsigned long ) ide_get_drivedata ( pair ) ;
if ( pair_mode & & pair_mode < mode )
addr_pio = pair_mode - XFER_PIO_0 ;
2008-06-15 21:00:22 +02:00
}
2005-04-16 15:20:36 -07:00
2008-06-15 21:00:22 +02:00
spin_lock_irqsave ( & opti621_lock , flags ) ;
reg_base = hwif - > io_ports . data_addr ;
/* allow Register-B */
outb ( 0xc0 , reg_base + CNTRL_REG ) ;
/* hmm, setupvic.exe does this ;-) */
outb ( 0xff , reg_base + 5 ) ;
/* if reads 0xff, adapter not exist? */
( void ) inb ( reg_base + CNTRL_REG ) ;
/* if reads 0xc0, no interface exist? */
read_reg ( CNTRL_REG ) ;
/* check CLK speed */
clk = read_reg ( STRAP_REG ) & 1 ;
printk ( KERN_INFO " %s: CLK = %d MHz \n " , hwif - > name , clk ? 25 : 33 ) ;
2008-06-15 21:00:22 +02:00
tim = data_rec_timings [ clk ] [ pio ] ;
misc = addr_timings [ clk ] [ addr_pio ] ;
2005-04-16 15:20:36 -07:00
2008-06-15 21:00:22 +02:00
/* select Index-0/1 for Register-A/B */
2008-10-13 21:39:40 +02:00
write_reg ( drive - > dn & 1 , MISC_REG ) ;
2007-02-17 02:40:25 +01:00
/* set read cycle timings */
2008-06-15 21:00:22 +02:00
write_reg ( tim , READ_REG ) ;
2007-02-17 02:40:25 +01:00
/* set write cycle timings */
2008-06-15 21:00:22 +02:00
write_reg ( tim , WRITE_REG ) ;
2005-04-16 15:20:36 -07:00
/* use Register-A for drive 0 */
/* use Register-B for drive 1 */
2007-02-17 02:40:25 +01:00
write_reg ( 0x85 , CNTRL_REG ) ;
2005-04-16 15:20:36 -07:00
/* set address setup, DRDY timings, */
/* and read prefetch for both drives */
2007-02-17 02:40:25 +01:00
write_reg ( misc , MISC_REG ) ;
2005-04-16 15:20:36 -07:00
2007-10-20 00:32:35 +02:00
spin_unlock_irqrestore ( & opti621_lock , flags ) ;
2005-04-16 15:20:36 -07:00
}
2008-04-26 22:25:14 +02:00
static const struct ide_port_ops opti621_port_ops = {
. set_pio_mode = opti621_set_pio_mode ,
} ;
2005-04-16 15:20:36 -07:00
2008-06-15 21:00:22 +02:00
static const struct ide_port_info opti621_chipset __devinitdata = {
2008-07-24 22:53:32 +02:00
. name = DRV_NAME ,
2008-06-15 21:00:22 +02:00
. enablebits = { { 0x45 , 0x80 , 0x00 } , { 0x40 , 0x08 , 0x00 } } ,
. port_ops = & opti621_port_ops ,
. host_flags = IDE_HFLAG_NO_DMA ,
. pio_mask = ATA_PIO4 ,
2005-04-16 15:20:36 -07:00
} ;
static int __devinit opti621_init_one ( struct pci_dev * dev , const struct pci_device_id * id )
{
2008-07-24 22:53:14 +02:00
return ide_pci_init_one ( dev , & opti621_chipset , NULL ) ;
2005-04-16 15:20:36 -07:00
}
2007-10-16 22:29:56 +02:00
static const struct pci_device_id opti621_pci_tbl [ ] = {
{ PCI_VDEVICE ( OPTI , PCI_DEVICE_ID_OPTI_82C621 ) , 0 } ,
2008-06-15 21:00:22 +02:00
{ PCI_VDEVICE ( OPTI , PCI_DEVICE_ID_OPTI_82C825 ) , 0 } ,
2005-04-16 15:20:36 -07:00
{ 0 , } ,
} ;
MODULE_DEVICE_TABLE ( pci , opti621_pci_tbl ) ;
2008-10-13 21:39:41 +02:00
static struct pci_driver opti621_pci_driver = {
2005-04-16 15:20:36 -07:00
. name = " Opti621_IDE " ,
. id_table = opti621_pci_tbl ,
. probe = opti621_init_one ,
2008-07-24 22:53:23 +02:00
. remove = ide_pci_remove ,
2008-10-10 22:39:32 +02:00
. suspend = ide_pci_suspend ,
. resume = ide_pci_resume ,
2005-04-16 15:20:36 -07:00
} ;
2007-01-27 13:46:56 +01:00
static int __init opti621_ide_init ( void )
2005-04-16 15:20:36 -07:00
{
2008-10-13 21:39:41 +02:00
return ide_pci_register_driver ( & opti621_pci_driver ) ;
2005-04-16 15:20:36 -07:00
}
2008-07-24 22:53:23 +02:00
static void __exit opti621_ide_exit ( void )
{
2008-10-13 21:39:41 +02:00
pci_unregister_driver ( & opti621_pci_driver ) ;
2008-07-24 22:53:23 +02:00
}
2005-04-16 15:20:36 -07:00
module_init ( opti621_ide_init ) ;
2008-07-24 22:53:23 +02:00
module_exit ( opti621_ide_exit ) ;
2005-04-16 15:20:36 -07:00
MODULE_AUTHOR ( " Jaromir Koutek, Jan Harkes, Mark Lord " ) ;
MODULE_DESCRIPTION ( " PCI driver module for Opti621 IDE " ) ;
MODULE_LICENSE ( " GPL " ) ;