2008-01-09 22:10:41 +03:00
/*
* OF - platform PATA driver
*
* Copyright ( c ) 2007 MontaVista Software , Inc .
* Anton Vorontsov < avorontsov @ ru . mvista . com >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License ( Version 2 ) as
* published by the Free Software Foundation .
*/
# include <linux/kernel.h>
# include <linux/module.h>
2011-09-07 13:36:26 +01:00
# include <linux/of_address.h>
# include <linux/of_irq.h>
2008-01-09 22:10:41 +03:00
# include <linux/of_platform.h>
2008-02-01 18:02:30 -05:00
# include <linux/ata_platform.h>
2008-01-09 22:10:41 +03:00
2011-02-17 02:43:24 -07:00
static int __devinit pata_of_platform_probe ( struct platform_device * ofdev )
2008-01-09 22:10:41 +03:00
{
int ret ;
2010-04-13 16:12:29 -07:00
struct device_node * dn = ofdev - > dev . of_node ;
2008-01-09 22:10:41 +03:00
struct resource io_res ;
struct resource ctl_res ;
struct resource irq_res ;
unsigned int reg_shift = 0 ;
int pio_mode = 0 ;
int pio_mask ;
const u32 * prop ;
ret = of_address_to_resource ( dn , 0 , & io_res ) ;
if ( ret ) {
dev_err ( & ofdev - > dev , " can't get IO address from "
" device tree \n " ) ;
return - EINVAL ;
}
2007-12-04 14:44:32 -06:00
if ( of_device_is_compatible ( dn , " electra-ide " ) ) {
/* Altstatus is really at offset 0x3f6 from the primary window
* on electra - ide . Adjust ctl_res and io_res accordingly .
*/
ctl_res = io_res ;
ctl_res . start = ctl_res . start + 0x3f6 ;
io_res . end = ctl_res . start - 1 ;
} else {
ret = of_address_to_resource ( dn , 1 , & ctl_res ) ;
if ( ret ) {
dev_err ( & ofdev - > dev , " can't get CTL address from "
" device tree \n " ) ;
return - EINVAL ;
}
2008-01-09 22:10:41 +03:00
}
ret = of_irq_to_resource ( dn , 0 , & irq_res ) ;
2011-11-10 20:28:59 +04:00
if ( ! ret )
2008-10-06 07:26:54 +00:00
irq_res . start = irq_res . end = 0 ;
2008-01-09 22:10:41 +03:00
else
irq_res . flags = 0 ;
prop = of_get_property ( dn , " reg-shift " , NULL ) ;
if ( prop )
2011-09-07 13:36:26 +01:00
reg_shift = be32_to_cpup ( prop ) ;
2008-01-09 22:10:41 +03:00
prop = of_get_property ( dn , " pio-mode " , NULL ) ;
if ( prop ) {
2011-09-07 13:36:26 +01:00
pio_mode = be32_to_cpup ( prop ) ;
2008-01-09 22:10:41 +03:00
if ( pio_mode > 6 ) {
dev_err ( & ofdev - > dev , " invalid pio-mode \n " ) ;
return - EINVAL ;
}
} else {
dev_info ( & ofdev - > dev , " pio-mode unspecified, assuming PIO0 \n " ) ;
}
pio_mask = 1 < < pio_mode ;
pio_mask | = ( 1 < < pio_mode ) - 1 ;
return __pata_platform_probe ( & ofdev - > dev , & io_res , & ctl_res , & irq_res ,
reg_shift , pio_mask ) ;
}
2010-08-06 09:25:50 -06:00
static int __devexit pata_of_platform_remove ( struct platform_device * ofdev )
2008-01-09 22:10:41 +03:00
{
return __pata_platform_remove ( & ofdev - > dev ) ;
}
static struct of_device_id pata_of_platform_match [ ] = {
{ . compatible = " ata-generic " , } ,
2007-12-04 14:44:32 -06:00
{ . compatible = " electra-ide " , } ,
2008-01-09 22:10:41 +03:00
{ } ,
} ;
MODULE_DEVICE_TABLE ( of , pata_of_platform_match ) ;
2011-02-17 02:43:24 -07:00
static struct platform_driver pata_of_platform_driver = {
2010-04-13 16:13:02 -07:00
. driver = {
. name = " pata_of_platform " ,
. owner = THIS_MODULE ,
. of_match_table = pata_of_platform_match ,
} ,
2008-01-09 22:10:41 +03:00
. probe = pata_of_platform_probe ,
. remove = __devexit_p ( pata_of_platform_remove ) ,
} ;
static int __init pata_of_platform_init ( void )
{
2011-02-17 02:43:24 -07:00
return platform_driver_register ( & pata_of_platform_driver ) ;
2008-01-09 22:10:41 +03:00
}
module_init ( pata_of_platform_init ) ;
static void __exit pata_of_platform_exit ( void )
{
2011-02-17 02:43:24 -07:00
platform_driver_unregister ( & pata_of_platform_driver ) ;
2008-01-09 22:10:41 +03:00
}
module_exit ( pata_of_platform_exit ) ;
MODULE_DESCRIPTION ( " OF-platform PATA driver " ) ;
MODULE_AUTHOR ( " Anton Vorontsov <avorontsov@ru.mvista.com> " ) ;
MODULE_LICENSE ( " GPL " ) ;