2005-04-17 02:20:36 +04:00
/*
* sata_nv . c - NVIDIA nForce SATA
*
* Copyright 2004 NVIDIA Corp . All rights reserved .
* Copyright 2004 Andrew Chew
*
2005-08-29 23:12:56 +04:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 , or ( at your option )
* any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; see the file COPYING . If not , write to
* the Free Software Foundation , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
2005-04-17 02:20:36 +04:00
*
2005-08-29 04:18:39 +04:00
*
* libata documentation is available via ' make { ps | pdf } docs ' ,
* as Documentation / DocBook / libata . *
*
* No hardware documentation available outside of NVIDIA .
* This driver programs the NVIDIA SATA controller in a similar
* fashion as with other PCI IDE BMDMA controllers , with a few
* NV - specific details such as register offsets , SATA phy location ,
* hotplug info , etc .
*
2005-04-17 02:20:36 +04:00
*/
# include <linux/config.h>
# 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 <linux/interrupt.h>
2005-10-30 22:39:11 +03:00
# include <linux/device.h>
2005-04-17 02:20:36 +04:00
# include <scsi/scsi_host.h>
# include <linux/libata.h>
# define DRV_NAME "sata_nv"
2006-04-03 04:41:36 +04:00
# define DRV_VERSION "0.9"
2005-04-17 02:20:36 +04:00
2006-03-23 07:50:50 +03:00
enum {
NV_PORTS = 2 ,
NV_PIO_MASK = 0x1f ,
NV_MWDMA_MASK = 0x07 ,
NV_UDMA_MASK = 0x7f ,
NV_PORT0_SCR_REG_OFFSET = 0x00 ,
NV_PORT1_SCR_REG_OFFSET = 0x40 ,
2005-04-17 02:20:36 +04:00
2006-06-17 10:49:55 +04:00
/* INT_STATUS/ENABLE */
2006-03-23 07:50:50 +03:00
NV_INT_STATUS = 0x10 ,
NV_INT_ENABLE = 0x11 ,
2006-06-17 10:49:55 +04:00
NV_INT_STATUS_CK804 = 0x440 ,
2006-03-23 07:50:50 +03:00
NV_INT_ENABLE_CK804 = 0x441 ,
2005-04-17 02:20:36 +04:00
2006-06-17 10:49:55 +04:00
/* INT_STATUS/ENABLE bits */
NV_INT_DEV = 0x01 ,
NV_INT_PM = 0x02 ,
NV_INT_ADDED = 0x04 ,
NV_INT_REMOVED = 0x08 ,
NV_INT_PORT_SHIFT = 4 , /* each port occupies 4 bits */
/* INT_CONFIG */
2006-03-23 07:50:50 +03:00
NV_INT_CONFIG = 0x12 ,
NV_INT_CONFIG_METHD = 0x01 , // 0 = INT, 1 = SMI
2005-04-17 02:20:36 +04:00
2006-03-23 07:50:50 +03:00
// For PCI config register 20
NV_MCP_SATA_CFG_20 = 0x50 ,
NV_MCP_SATA_CFG_20_SATA_SPACE_EN = 0x04 ,
} ;
2005-04-17 02:20:36 +04:00
static int nv_init_one ( struct pci_dev * pdev , const struct pci_device_id * ent ) ;
static irqreturn_t nv_interrupt ( int irq , void * dev_instance ,
struct pt_regs * regs ) ;
static u32 nv_scr_read ( struct ata_port * ap , unsigned int sc_reg ) ;
static void nv_scr_write ( struct ata_port * ap , unsigned int sc_reg , u32 val ) ;
enum nv_host_type
{
GENERIC ,
NFORCE2 ,
2006-06-17 10:49:55 +04:00
NFORCE3 = NFORCE2 , /* NF2 == NF3 as far as sata_nv is concerned */
2005-10-07 19:53:39 +04:00
CK804
2005-04-17 02:20:36 +04:00
} ;
2005-11-10 19:04:11 +03:00
static const struct pci_device_id nv_pci_tbl [ ] = {
2005-04-17 02:20:36 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE2S_SATA ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , NFORCE2 } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE3S_SATA ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , NFORCE3 } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE3S_SATA2 ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , NFORCE3 } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_CK804_SATA ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , CK804 } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_CK804_SATA2 ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , CK804 } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP04_SATA ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , CK804 } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP04_SATA2 ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , CK804 } ,
2005-07-03 16:44:39 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP51_SATA ,
2005-10-07 19:53:39 +04:00
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , GENERIC } ,
2005-07-03 16:44:39 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP51_SATA2 ,
2005-10-07 19:53:39 +04:00
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , GENERIC } ,
2005-07-03 16:44:39 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP55_SATA ,
2005-10-07 19:53:39 +04:00
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , GENERIC } ,
2005-09-19 17:17:52 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP55_SATA2 ,
2005-10-07 19:53:39 +04:00
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , GENERIC } ,
2006-04-21 02:54:26 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP61_SATA ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , GENERIC } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP61_SATA2 ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , GENERIC } ,
{ PCI_VENDOR_ID_NVIDIA , PCI_DEVICE_ID_NVIDIA_NFORCE_MCP61_SATA3 ,
PCI_ANY_ID , PCI_ANY_ID , 0 , 0 , GENERIC } ,
2005-04-17 02:20:36 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_ANY_ID ,
PCI_ANY_ID , PCI_ANY_ID ,
PCI_CLASS_STORAGE_IDE < < 8 , 0xffff00 , GENERIC } ,
2005-07-03 16:44:39 +04:00
{ PCI_VENDOR_ID_NVIDIA , PCI_ANY_ID ,
PCI_ANY_ID , PCI_ANY_ID ,
PCI_CLASS_STORAGE_RAID < < 8 , 0xffff00 , GENERIC } ,
2005-04-17 02:20:36 +04:00
{ 0 , } /* terminate list */
} ;
static struct pci_driver nv_pci_driver = {
. name = DRV_NAME ,
. id_table = nv_pci_tbl ,
. probe = nv_init_one ,
. remove = ata_pci_remove_one ,
} ;
2005-11-07 08:59:37 +03:00
static struct scsi_host_template nv_sht = {
2005-04-17 02:20:36 +04:00
. module = THIS_MODULE ,
. name = DRV_NAME ,
. ioctl = ata_scsi_ioctl ,
. queuecommand = ata_scsi_queuecmd ,
. can_queue = ATA_DEF_QUEUE ,
. this_id = ATA_SHT_THIS_ID ,
. sg_tablesize = LIBATA_MAX_PRD ,
. cmd_per_lun = ATA_SHT_CMD_PER_LUN ,
. emulated = ATA_SHT_EMULATED ,
. use_clustering = ATA_SHT_USE_CLUSTERING ,
. proc_name = DRV_NAME ,
. dma_boundary = ATA_DMA_BOUNDARY ,
. slave_configure = ata_scsi_slave_config ,
2006-05-31 13:28:09 +04:00
. slave_destroy = ata_scsi_slave_destroy ,
2005-04-17 02:20:36 +04:00
. bios_param = ata_std_bios_param ,
} ;
2005-10-22 22:27:05 +04:00
static const struct ata_port_operations nv_ops = {
2005-04-17 02:20:36 +04:00
. port_disable = ata_port_disable ,
. tf_load = ata_tf_load ,
. tf_read = ata_tf_read ,
. exec_command = ata_exec_command ,
. check_status = ata_check_status ,
. dev_select = ata_std_dev_select ,
. phy_reset = sata_phy_reset ,
. bmdma_setup = ata_bmdma_setup ,
. bmdma_start = ata_bmdma_start ,
. bmdma_stop = ata_bmdma_stop ,
. bmdma_status = ata_bmdma_status ,
. qc_prep = ata_qc_prep ,
. qc_issue = ata_qc_issue_prot ,
. eng_timeout = ata_eng_timeout ,
2006-05-22 19:59:59 +04:00
. data_xfer = ata_pio_data_xfer ,
2005-04-17 02:20:36 +04:00
. irq_handler = nv_interrupt ,
. irq_clear = ata_bmdma_irq_clear ,
. scr_read = nv_scr_read ,
. scr_write = nv_scr_write ,
. port_start = ata_port_start ,
. port_stop = ata_port_stop ,
2006-06-17 10:49:55 +04:00
. host_stop = ata_pci_host_stop ,
2005-04-17 02:20:36 +04:00
} ;
/* FIXME: The hardware provides the necessary SATA PHY controls
* to support ATA_FLAG_SATA_RESET . However , it is currently
* necessary to disable that flag , to solve misdetection problems .
* See http : //bugme.osdl.org/show_bug.cgi?id=3352 for more info.
*
* This problem really needs to be investigated further . But in the
* meantime , we avoid ATA_FLAG_SATA_RESET to get people working .
*/
static struct ata_port_info nv_port_info = {
. sht = & nv_sht ,
. host_flags = ATA_FLAG_SATA |
/* ATA_FLAG_SATA_RESET | */
ATA_FLAG_SRST |
ATA_FLAG_NO_LEGACY ,
. pio_mask = NV_PIO_MASK ,
. mwdma_mask = NV_MWDMA_MASK ,
. udma_mask = NV_UDMA_MASK ,
. port_ops = & nv_ops ,
} ;
MODULE_AUTHOR ( " NVIDIA " ) ;
MODULE_DESCRIPTION ( " low-level driver for NVIDIA nForce SATA controller " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_DEVICE_TABLE ( pci , nv_pci_tbl ) ;
MODULE_VERSION ( DRV_VERSION ) ;
static irqreturn_t nv_interrupt ( int irq , void * dev_instance ,
struct pt_regs * regs )
{
struct ata_host_set * host_set = dev_instance ;
unsigned int i ;
unsigned int handled = 0 ;
unsigned long flags ;
spin_lock_irqsave ( & host_set - > lock , flags ) ;
for ( i = 0 ; i < host_set - > n_ports ; i + + ) {
struct ata_port * ap ;
ap = host_set - > ports [ i ] ;
2005-08-22 09:59:24 +04:00
if ( ap & &
2006-04-02 18:30:40 +04:00
! ( ap - > flags & ATA_FLAG_DISABLED ) ) {
2005-04-17 02:20:36 +04:00
struct ata_queued_cmd * qc ;
qc = ata_qc_from_tag ( ap , ap - > active_tag ) ;
2005-09-27 13:39:50 +04:00
if ( qc & & ( ! ( qc - > tf . flags & ATA_TFLAG_POLLING ) ) )
2005-04-17 02:20:36 +04:00
handled + = ata_host_intr ( ap , qc ) ;
2006-01-05 06:13:04 +03:00
else
// No request pending? Clear interrupt status
// anyway, in case there's one pending.
ap - > ops - > check_status ( ap ) ;
2005-04-17 02:20:36 +04:00
}
}
spin_unlock_irqrestore ( & host_set - > lock , flags ) ;
return IRQ_RETVAL ( handled ) ;
}
static u32 nv_scr_read ( struct ata_port * ap , unsigned int sc_reg )
{
if ( sc_reg > SCR_CONTROL )
return 0xffffffffU ;
2006-03-23 07:59:46 +03:00
return ioread32 ( ( void __iomem * ) ap - > ioaddr . scr_addr + ( sc_reg * 4 ) ) ;
2005-04-17 02:20:36 +04:00
}
static void nv_scr_write ( struct ata_port * ap , unsigned int sc_reg , u32 val )
{
if ( sc_reg > SCR_CONTROL )
return ;
2006-03-23 07:59:46 +03:00
iowrite32 ( val , ( void __iomem * ) ap - > ioaddr . scr_addr + ( sc_reg * 4 ) ) ;
2005-04-17 02:20:36 +04:00
}
static int nv_init_one ( struct pci_dev * pdev , const struct pci_device_id * ent )
{
static int printed_version = 0 ;
struct ata_port_info * ppi ;
struct ata_probe_ent * probe_ent ;
int pci_dev_busy = 0 ;
int rc ;
u32 bar ;
2006-03-23 07:59:46 +03:00
unsigned long base ;
2005-04-17 02:20:36 +04:00
// Make sure this is a SATA controller by counting the number of bars
// (NVIDIA SATA controllers will always have six bars). Otherwise,
// it's an IDE controller and we ignore it.
for ( bar = 0 ; bar < 6 ; bar + + )
if ( pci_resource_start ( pdev , bar ) = = 0 )
return - ENODEV ;
if ( ! printed_version + + )
2005-10-30 22:39:11 +03:00
dev_printk ( KERN_DEBUG , & pdev - > dev , " version " DRV_VERSION " \n " ) ;
2005-04-17 02:20:36 +04:00
rc = pci_enable_device ( pdev ) ;
if ( rc )
goto err_out ;
rc = pci_request_regions ( pdev , DRV_NAME ) ;
if ( rc ) {
pci_dev_busy = 1 ;
goto err_out_disable ;
}
rc = pci_set_dma_mask ( pdev , ATA_DMA_MASK ) ;
if ( rc )
goto err_out_regions ;
rc = pci_set_consistent_dma_mask ( pdev , ATA_DMA_MASK ) ;
if ( rc )
goto err_out_regions ;
rc = - ENOMEM ;
ppi = & nv_port_info ;
2005-10-04 16:09:19 +04:00
probe_ent = ata_pci_init_native_mode ( pdev , & ppi , ATA_PORT_PRIMARY | ATA_PORT_SECONDARY ) ;
2005-04-17 02:20:36 +04:00
if ( ! probe_ent )
goto err_out_regions ;
2006-03-23 07:59:46 +03:00
probe_ent - > mmio_base = pci_iomap ( pdev , 5 , 0 ) ;
if ( ! probe_ent - > mmio_base ) {
rc = - EIO ;
2006-06-17 10:49:55 +04:00
goto err_out_free_ent ;
2006-03-23 07:59:46 +03:00
}
2005-04-17 02:20:36 +04:00
2006-03-23 07:59:46 +03:00
base = ( unsigned long ) probe_ent - > mmio_base ;
2005-04-17 02:20:36 +04:00
2006-03-23 07:59:46 +03:00
probe_ent - > port [ 0 ] . scr_addr = base + NV_PORT0_SCR_REG_OFFSET ;
probe_ent - > port [ 1 ] . scr_addr = base + NV_PORT1_SCR_REG_OFFSET ;
2005-04-17 02:20:36 +04:00
pci_set_master ( pdev ) ;
rc = ata_device_add ( probe_ent ) ;
if ( rc ! = NV_PORTS )
goto err_out_iounmap ;
kfree ( probe_ent ) ;
return 0 ;
err_out_iounmap :
2006-03-23 07:59:46 +03:00
pci_iounmap ( pdev , probe_ent - > mmio_base ) ;
2005-04-17 02:20:36 +04:00
err_out_free_ent :
kfree ( probe_ent ) ;
err_out_regions :
pci_release_regions ( pdev ) ;
err_out_disable :
if ( ! pci_dev_busy )
pci_disable_device ( pdev ) ;
err_out :
return rc ;
}
static int __init nv_init ( void )
{
return pci_module_init ( & nv_pci_driver ) ;
}
static void __exit nv_exit ( void )
{
pci_unregister_driver ( & nv_pci_driver ) ;
}
module_init ( nv_init ) ;
module_exit ( nv_exit ) ;