2006-10-27 19:08:48 -07:00
/*
* Generic platform device PATA driver
*
2007-11-08 11:15:21 +09:00
* Copyright ( C ) 2006 - 2007 Paul Mundt
2006-10-27 19:08:48 -07:00
*
* Based on pata_pcmcia :
*
* Copyright 2005 - 2006 Red Hat Inc < alan @ redhat . com > , all rights reserved .
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/init.h>
# include <linux/blkdev.h>
# include <scsi/scsi_host.h>
# include <linux/ata.h>
# include <linux/libata.h>
# include <linux/platform_device.h>
# include <linux/pata_platform.h>
# define DRV_NAME "pata_platform"
2007-11-08 11:15:21 +09:00
# define DRV_VERSION "1.2"
2006-10-27 19:08:48 -07:00
static int pio_mask = 1 ;
/*
* Provide our own set_mode ( ) as we don ' t want to change anything that has
* already been configured . .
*/
2007-08-06 18:36:23 +09:00
static int pata_platform_set_mode ( struct ata_link * link , struct ata_device * * unused )
2006-10-27 19:08:48 -07:00
{
2007-08-06 18:36:23 +09:00
struct ata_device * dev ;
2006-10-27 19:08:48 -07:00
2007-08-06 18:36:23 +09:00
ata_link_for_each_dev ( dev , link ) {
2006-10-27 19:08:48 -07:00
if ( ata_dev_enabled ( dev ) ) {
/* We don't really care */
dev - > pio_mode = dev - > xfer_mode = XFER_PIO_0 ;
dev - > xfer_shift = ATA_SHIFT_PIO ;
dev - > flags | = ATA_DFLAG_PIO ;
2007-02-20 17:35:43 +00:00
ata_dev_printk ( dev , KERN_INFO , " configured for PIO \n " ) ;
2006-10-27 19:08:48 -07:00
}
}
2007-01-30 13:23:45 +00:00
return 0 ;
2006-10-27 19:08:48 -07:00
}
2007-05-09 23:00:20 -05:00
static int ata_dummy_ret0 ( struct ata_port * ap ) { return 0 ; }
2006-10-27 19:08:48 -07:00
static struct scsi_host_template pata_platform_sht = {
. 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-11-29 12:10:46 +09:00
. slave_destroy = ata_scsi_slave_destroy ,
2006-10-27 19:08:48 -07:00
. bios_param = ata_std_bios_param ,
} ;
static struct ata_port_operations pata_platform_port_ops = {
. set_mode = pata_platform_set_mode ,
. tf_load = ata_tf_load ,
. tf_read = ata_tf_read ,
. check_status = ata_check_status ,
. exec_command = ata_exec_command ,
. dev_select = ata_std_dev_select ,
. freeze = ata_bmdma_freeze ,
. thaw = ata_bmdma_thaw ,
. error_handler = ata_bmdma_error_handler ,
. post_internal_cmd = ata_bmdma_post_internal_cmd ,
2007-03-08 22:54:04 +00:00
. cable_detect = ata_cable_unknown ,
2006-10-27 19:08:48 -07:00
. qc_prep = ata_qc_prep ,
. qc_issue = ata_qc_issue_prot ,
2007-02-01 15:06:36 +09:00
. data_xfer = ata_data_xfer_noirq ,
2006-10-27 19:08:48 -07:00
. irq_clear = ata_bmdma_irq_clear ,
2007-01-26 16:27:58 +09:00
. irq_on = ata_irq_on ,
2006-10-27 19:08:48 -07:00
2007-05-09 23:00:20 -05:00
. port_start = ata_dummy_ret0 ,
2006-10-27 19:08:48 -07:00
} ;
static void pata_platform_setup_port ( struct ata_ioports * ioaddr ,
2008-01-09 22:10:22 +03:00
unsigned int shift )
2006-10-27 19:08:48 -07:00
{
/* Fixup the port shift for platforms that need it */
ioaddr - > data_addr = ioaddr - > cmd_addr + ( ATA_REG_DATA < < shift ) ;
ioaddr - > error_addr = ioaddr - > cmd_addr + ( ATA_REG_ERR < < shift ) ;
ioaddr - > feature_addr = ioaddr - > cmd_addr + ( ATA_REG_FEATURE < < shift ) ;
ioaddr - > nsect_addr = ioaddr - > cmd_addr + ( ATA_REG_NSECT < < shift ) ;
ioaddr - > lbal_addr = ioaddr - > cmd_addr + ( ATA_REG_LBAL < < shift ) ;
ioaddr - > lbam_addr = ioaddr - > cmd_addr + ( ATA_REG_LBAM < < shift ) ;
ioaddr - > lbah_addr = ioaddr - > cmd_addr + ( ATA_REG_LBAH < < shift ) ;
ioaddr - > device_addr = ioaddr - > cmd_addr + ( ATA_REG_DEVICE < < shift ) ;
ioaddr - > status_addr = ioaddr - > cmd_addr + ( ATA_REG_STATUS < < shift ) ;
ioaddr - > command_addr = ioaddr - > cmd_addr + ( ATA_REG_CMD < < shift ) ;
}
/**
2008-01-09 22:10:22 +03:00
* __pata_platform_probe - attach a platform interface
* @ dev : device
* @ io_res : Resource representing I / O base
* @ ctl_res : Resource representing CTL base
* @ irq_res : Resource representing IRQ and its flags
* @ ioport_shift : I / O port shift
* @ __pio_mask : PIO mask
2006-10-27 19:08:48 -07:00
*
* Register a platform bus IDE interface . Such interfaces are PIO and we
* assume do not support IRQ sharing .
*
2007-11-08 11:15:21 +09:00
* Platform devices are expected to contain at least 2 resources per port :
2006-10-27 19:08:48 -07:00
*
* - I / O Base ( IORESOURCE_IO or IORESOURCE_MEM )
* - CTL Base ( IORESOURCE_IO or IORESOURCE_MEM )
2007-11-08 11:15:21 +09:00
*
* and optionally :
*
2006-10-27 19:08:48 -07:00
* - IRQ ( IORESOURCE_IRQ )
*
* If the base resources are both mem types , the ioremap ( ) is handled
* here . For IORESOURCE_IO , it ' s assumed that there ' s no remapping
* necessary .
2007-11-08 11:15:21 +09:00
*
* If no IRQ resource is present , PIO polling mode is used instead .
2006-10-27 19:08:48 -07:00
*/
2008-01-09 22:10:22 +03:00
int __devinit __pata_platform_probe ( struct device * dev ,
struct resource * io_res ,
struct resource * ctl_res ,
struct resource * irq_res ,
unsigned int ioport_shift ,
int __pio_mask )
2006-10-27 19:08:48 -07:00
{
libata: convert the remaining PATA drivers to new init model
Convert pdc_adma, pata_cs5520, pata_isapnp, pata_ixp4xx_cf,
pata_legacy, pata_mpc52xx, pata_mpiix, pata_pcmcia, pata_pdc2027x,
pata_platform, pata_qdi, pata_scc and pata_winbond to new init model.
* init_one()'s now follow more consistent init order
* cs5520 now registers one host with two ports, not two hosts. If any
of the two ports are disabled, it's made dummy as other drivers do.
Tested pdc_adma and pata_legacy. Both are as broken as before. The
rest are compile tested only.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-04-17 23:44:08 +09:00
struct ata_host * host ;
struct ata_port * ap ;
2006-10-27 19:08:48 -07:00
unsigned int mmio ;
2008-01-09 22:10:22 +03:00
int irq = 0 ;
int irq_flags = 0 ;
2006-10-27 19:08:48 -07:00
/*
* Check for MMIO
*/
mmio = ( ( io_res - > flags = = IORESOURCE_MEM ) & &
( ctl_res - > flags = = IORESOURCE_MEM ) ) ;
2007-11-08 11:15:21 +09:00
/*
* And the IRQ
*/
2008-01-09 22:10:22 +03:00
if ( irq_res & & irq_res - > start > 0 ) {
irq = irq_res - > start ;
irq_flags = irq_res - > flags ;
}
2007-11-08 11:15:21 +09:00
2006-10-27 19:08:48 -07:00
/*
* Now that that ' s out of the way , wire up the port . .
*/
2008-01-09 22:10:22 +03:00
host = ata_host_alloc ( dev , 1 ) ;
libata: convert the remaining PATA drivers to new init model
Convert pdc_adma, pata_cs5520, pata_isapnp, pata_ixp4xx_cf,
pata_legacy, pata_mpc52xx, pata_mpiix, pata_pcmcia, pata_pdc2027x,
pata_platform, pata_qdi, pata_scc and pata_winbond to new init model.
* init_one()'s now follow more consistent init order
* cs5520 now registers one host with two ports, not two hosts. If any
of the two ports are disabled, it's made dummy as other drivers do.
Tested pdc_adma and pata_legacy. Both are as broken as before. The
rest are compile tested only.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-04-17 23:44:08 +09:00
if ( ! host )
return - ENOMEM ;
ap = host - > ports [ 0 ] ;
ap - > ops = & pata_platform_port_ops ;
2008-01-09 22:10:22 +03:00
ap - > pio_mask = __pio_mask ;
libata: convert the remaining PATA drivers to new init model
Convert pdc_adma, pata_cs5520, pata_isapnp, pata_ixp4xx_cf,
pata_legacy, pata_mpc52xx, pata_mpiix, pata_pcmcia, pata_pdc2027x,
pata_platform, pata_qdi, pata_scc and pata_winbond to new init model.
* init_one()'s now follow more consistent init order
* cs5520 now registers one host with two ports, not two hosts. If any
of the two ports are disabled, it's made dummy as other drivers do.
Tested pdc_adma and pata_legacy. Both are as broken as before. The
rest are compile tested only.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-04-17 23:44:08 +09:00
ap - > flags | = ATA_FLAG_SLAVE_POSS ;
2006-10-27 19:08:48 -07:00
2007-11-08 11:15:21 +09:00
/*
* Use polling mode if there ' s no IRQ
*/
if ( ! irq ) {
ap - > flags | = ATA_FLAG_PIO_POLLING ;
ata_port_desc ( ap , " no IRQ, using PIO polling " ) ;
}
2006-10-27 19:08:48 -07:00
/*
* Handle the MMIO case
*/
if ( mmio ) {
2008-01-09 22:10:22 +03:00
ap - > ioaddr . cmd_addr = devm_ioremap ( dev , io_res - > start ,
2007-02-01 15:06:36 +09:00
io_res - > end - io_res - > start + 1 ) ;
2008-01-09 22:10:22 +03:00
ap - > ioaddr . ctl_addr = devm_ioremap ( dev , ctl_res - > start ,
2007-02-01 15:06:36 +09:00
ctl_res - > end - ctl_res - > start + 1 ) ;
2006-10-27 19:08:48 -07:00
} else {
2008-01-09 22:10:22 +03:00
ap - > ioaddr . cmd_addr = devm_ioport_map ( dev , io_res - > start ,
2007-02-01 15:06:36 +09:00
io_res - > end - io_res - > start + 1 ) ;
2008-01-09 22:10:22 +03:00
ap - > ioaddr . ctl_addr = devm_ioport_map ( dev , ctl_res - > start ,
2007-02-01 15:06:36 +09:00
ctl_res - > end - ctl_res - > start + 1 ) ;
}
libata: convert the remaining PATA drivers to new init model
Convert pdc_adma, pata_cs5520, pata_isapnp, pata_ixp4xx_cf,
pata_legacy, pata_mpc52xx, pata_mpiix, pata_pcmcia, pata_pdc2027x,
pata_platform, pata_qdi, pata_scc and pata_winbond to new init model.
* init_one()'s now follow more consistent init order
* cs5520 now registers one host with two ports, not two hosts. If any
of the two ports are disabled, it's made dummy as other drivers do.
Tested pdc_adma and pata_legacy. Both are as broken as before. The
rest are compile tested only.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-04-17 23:44:08 +09:00
if ( ! ap - > ioaddr . cmd_addr | | ! ap - > ioaddr . ctl_addr ) {
2008-01-09 22:10:22 +03:00
dev_err ( dev , " failed to map IO/CTL base \n " ) ;
2007-02-01 15:06:36 +09:00
return - ENOMEM ;
2006-10-27 19:08:48 -07:00
}
libata: convert the remaining PATA drivers to new init model
Convert pdc_adma, pata_cs5520, pata_isapnp, pata_ixp4xx_cf,
pata_legacy, pata_mpc52xx, pata_mpiix, pata_pcmcia, pata_pdc2027x,
pata_platform, pata_qdi, pata_scc and pata_winbond to new init model.
* init_one()'s now follow more consistent init order
* cs5520 now registers one host with two ports, not two hosts. If any
of the two ports are disabled, it's made dummy as other drivers do.
Tested pdc_adma and pata_legacy. Both are as broken as before. The
rest are compile tested only.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-04-17 23:44:08 +09:00
ap - > ioaddr . altstatus_addr = ap - > ioaddr . ctl_addr ;
2006-10-27 19:08:48 -07:00
2008-01-09 22:10:22 +03:00
pata_platform_setup_port ( & ap - > ioaddr , ioport_shift ) ;
2006-10-27 19:08:48 -07:00
2007-08-18 13:14:55 +09:00
ata_port_desc ( ap , " %s cmd 0x%llx ctl 0x%llx " , mmio ? " mmio " : " ioport " ,
( unsigned long long ) io_res - > start ,
( unsigned long long ) ctl_res - > start ) ;
libata: convert the remaining PATA drivers to new init model
Convert pdc_adma, pata_cs5520, pata_isapnp, pata_ixp4xx_cf,
pata_legacy, pata_mpc52xx, pata_mpiix, pata_pcmcia, pata_pdc2027x,
pata_platform, pata_qdi, pata_scc and pata_winbond to new init model.
* init_one()'s now follow more consistent init order
* cs5520 now registers one host with two ports, not two hosts. If any
of the two ports are disabled, it's made dummy as other drivers do.
Tested pdc_adma and pata_legacy. Both are as broken as before. The
rest are compile tested only.
Signed-off-by: Tejun Heo <htejun@gmail.com>
Signed-off-by: Jeff Garzik <jeff@garzik.org>
2007-04-17 23:44:08 +09:00
/* activate */
2007-11-08 11:15:21 +09:00
return ata_host_activate ( host , irq , irq ? ata_interrupt : NULL ,
2008-01-09 22:10:22 +03:00
irq_flags , & pata_platform_sht ) ;
2006-10-27 19:08:48 -07:00
}
2008-01-09 22:10:22 +03:00
EXPORT_SYMBOL_GPL ( __pata_platform_probe ) ;
2006-10-27 19:08:48 -07:00
/**
2008-01-09 22:10:22 +03:00
* __pata_platform_remove - unplug a platform interface
* @ dev : device
2006-10-27 19:08:48 -07:00
*
* A platform bus ATA device has been unplugged . Perform the needed
* cleanup . Also called on module unload for any active devices .
*/
2008-01-09 22:10:22 +03:00
int __devexit __pata_platform_remove ( struct device * dev )
2006-10-27 19:08:48 -07:00
{
struct ata_host * host = dev_get_drvdata ( dev ) ;
2007-02-01 15:05:22 +09:00
ata_host_detach ( host ) ;
2006-10-27 19:08:48 -07:00
return 0 ;
}
2008-01-09 22:10:22 +03:00
EXPORT_SYMBOL_GPL ( __pata_platform_remove ) ;
static int __devinit pata_platform_probe ( struct platform_device * pdev )
{
struct resource * io_res ;
struct resource * ctl_res ;
struct resource * irq_res ;
struct pata_platform_info * pp_info = pdev - > dev . platform_data ;
/*
* Simple resource validation . .
*/
if ( ( pdev - > num_resources ! = 3 ) & & ( pdev - > num_resources ! = 2 ) ) {
dev_err ( & pdev - > dev , " invalid number of resources \n " ) ;
return - EINVAL ;
}
/*
* Get the I / O base first
*/
io_res = platform_get_resource ( pdev , IORESOURCE_IO , 0 ) ;
if ( io_res = = NULL ) {
io_res = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( unlikely ( io_res = = NULL ) )
return - EINVAL ;
}
/*
* Then the CTL base
*/
ctl_res = platform_get_resource ( pdev , IORESOURCE_IO , 1 ) ;
if ( ctl_res = = NULL ) {
ctl_res = platform_get_resource ( pdev , IORESOURCE_MEM , 1 ) ;
if ( unlikely ( ctl_res = = NULL ) )
return - EINVAL ;
}
/*
* And the IRQ
*/
irq_res = platform_get_resource ( pdev , IORESOURCE_IRQ , 0 ) ;
if ( irq_res )
irq_res - > flags = pp_info ? pp_info - > irq_flags : 0 ;
return __pata_platform_probe ( & pdev - > dev , io_res , ctl_res , irq_res ,
pp_info ? pp_info - > ioport_shift : 0 ,
pio_mask ) ;
}
static int __devexit pata_platform_remove ( struct platform_device * pdev )
{
return __pata_platform_remove ( & pdev - > dev ) ;
}
2006-10-27 19:08:48 -07:00
static struct platform_driver pata_platform_driver = {
. probe = pata_platform_probe ,
. remove = __devexit_p ( pata_platform_remove ) ,
. driver = {
. name = DRV_NAME ,
. owner = THIS_MODULE ,
} ,
} ;
static int __init pata_platform_init ( void )
{
return platform_driver_register ( & pata_platform_driver ) ;
}
static void __exit pata_platform_exit ( void )
{
platform_driver_unregister ( & pata_platform_driver ) ;
}
module_init ( pata_platform_init ) ;
module_exit ( pata_platform_exit ) ;
module_param ( pio_mask , int , 0 ) ;
MODULE_AUTHOR ( " Paul Mundt " ) ;
MODULE_DESCRIPTION ( " low-level driver for platform device ATA " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_VERSION ( DRV_VERSION ) ;