2007-09-23 16:00:20 +01:00
/*
* linux / drivers / pcmcia / pxa / pxa_cm_x270 . c
*
* 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 .
*
2008-06-17 09:48:03 +01:00
* Compulab Ltd . , 2003 , 2007 , 2008
2007-09-23 16:00:20 +01:00
* Mike Rapoport < mike @ compulab . co . il >
*
*/
# include <linux/platform_device.h>
# include <linux/irq.h>
# include <linux/delay.h>
2008-06-17 09:48:03 +01:00
# include <linux/gpio.h>
2007-09-23 16:00:20 +01:00
2008-04-24 15:23:25 +01:00
# include <asm/mach-types.h>
2007-09-23 16:00:20 +01:00
# include "soc_common.h"
2008-06-17 09:48:03 +01:00
# define GPIO_PCMCIA_S0_CD_VALID (84)
# define GPIO_PCMCIA_S0_RDYINT (82)
# define GPIO_PCMCIA_RESET (53)
# define PCMCIA_S0_CD_VALID IRQ_GPIO(GPIO_PCMCIA_S0_CD_VALID)
# define PCMCIA_S0_RDYINT IRQ_GPIO(GPIO_PCMCIA_S0_RDYINT)
2007-09-23 16:00:20 +01:00
static struct pcmcia_irqs irqs [ ] = {
{ 0 , PCMCIA_S0_CD_VALID , " PCMCIA0 CD " } ,
} ;
static int cmx270_pcmcia_hw_init ( struct soc_pcmcia_socket * skt )
{
2008-06-17 09:48:03 +01:00
int ret = gpio_request ( GPIO_PCMCIA_RESET , " PCCard reset " ) ;
if ( ret )
return ret ;
gpio_direction_output ( GPIO_PCMCIA_RESET , 0 ) ;
skt - > irq = PCMCIA_S0_RDYINT ;
ret = soc_pcmcia_request_irqs ( skt , irqs , ARRAY_SIZE ( irqs ) ) ;
if ( ! ret )
gpio_free ( GPIO_PCMCIA_RESET ) ;
return ret ;
2007-09-23 16:00:20 +01:00
}
static void cmx270_pcmcia_shutdown ( struct soc_pcmcia_socket * skt )
{
soc_pcmcia_free_irqs ( skt , irqs , ARRAY_SIZE ( irqs ) ) ;
2008-06-17 09:48:03 +01:00
gpio_free ( GPIO_PCMCIA_RESET ) ;
2007-09-23 16:00:20 +01:00
}
static void cmx270_pcmcia_socket_state ( struct soc_pcmcia_socket * skt ,
struct pcmcia_state * state )
{
2008-06-17 09:48:03 +01:00
state - > detect = ( gpio_get_value ( GPIO_PCMCIA_S0_CD_VALID ) = = 0 ) ? 1 : 0 ;
state - > ready = ( gpio_get_value ( GPIO_PCMCIA_S0_RDYINT ) = = 0 ) ? 0 : 1 ;
2007-09-23 16:00:20 +01:00
state - > bvd1 = 1 ;
state - > bvd2 = 1 ;
state - > vs_3v = 0 ;
state - > vs_Xv = 0 ;
state - > wrprot = 0 ; /* not available */
}
static int cmx270_pcmcia_configure_socket ( struct soc_pcmcia_socket * skt ,
const socket_state_t * state )
{
switch ( skt - > nr ) {
case 0 :
if ( state - > flags & SS_RESET ) {
2008-06-17 09:48:03 +01:00
gpio_set_value ( GPIO_PCMCIA_RESET , 1 ) ;
2007-09-23 16:00:20 +01:00
udelay ( 10 ) ;
2008-06-17 09:48:03 +01:00
gpio_set_value ( GPIO_PCMCIA_RESET , 0 ) ;
2007-09-23 16:00:20 +01:00
}
break ;
}
return 0 ;
}
static void cmx270_pcmcia_socket_init ( struct soc_pcmcia_socket * skt )
{
}
static void cmx270_pcmcia_socket_suspend ( struct soc_pcmcia_socket * skt )
{
}
2008-04-24 15:28:11 +01:00
static struct pcmcia_low_level cmx270_pcmcia_ops __initdata = {
2007-09-23 16:00:20 +01:00
. owner = THIS_MODULE ,
. hw_init = cmx270_pcmcia_hw_init ,
. hw_shutdown = cmx270_pcmcia_shutdown ,
. socket_state = cmx270_pcmcia_socket_state ,
. configure_socket = cmx270_pcmcia_configure_socket ,
. socket_init = cmx270_pcmcia_socket_init ,
. socket_suspend = cmx270_pcmcia_socket_suspend ,
2008-06-17 09:48:03 +01:00
. nr = 1 ,
2007-09-23 16:00:20 +01:00
} ;
static struct platform_device * cmx270_pcmcia_device ;
2008-10-05 10:27:22 +01:00
int __init cmx270_pcmcia_init ( void )
2007-09-23 16:00:20 +01:00
{
int ret ;
cmx270_pcmcia_device = platform_device_alloc ( " pxa2xx-pcmcia " , - 1 ) ;
if ( ! cmx270_pcmcia_device )
return - ENOMEM ;
2008-04-24 15:28:11 +01:00
ret = platform_device_add_data ( cmx270_pcmcia_device , & cmx270_pcmcia_ops ,
sizeof ( cmx270_pcmcia_ops ) ) ;
2007-09-23 16:00:20 +01:00
2008-04-24 15:28:11 +01:00
if ( ret = = 0 ) {
printk ( KERN_INFO " Registering cm-x270 PCMCIA interface. \n " ) ;
ret = platform_device_add ( cmx270_pcmcia_device ) ;
}
2007-09-23 16:00:20 +01:00
if ( ret )
platform_device_put ( cmx270_pcmcia_device ) ;
return ret ;
}
2008-10-05 10:27:22 +01:00
void __exit cmx270_pcmcia_exit ( void )
2007-09-23 16:00:20 +01:00
{
platform_device_unregister ( cmx270_pcmcia_device ) ;
}