2007-09-12 06:13:17 +04:00
/*
* linux / arch / arm / mach - pxa / pxa3xx . c
*
* code specific to pxa3xx aka Monahans
*
* Copyright ( C ) 2006 Marvell International Ltd .
*
2007-10-30 10:01:38 +03:00
* 2007 - 09 - 02 : eric miao < eric . miao @ marvell . com >
2007-09-12 06:13:17 +04:00
* initial version
*
* 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/module.h>
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/pm.h>
# include <linux/platform_device.h>
# include <linux/irq.h>
2008-01-08 01:18:30 +03:00
# include <linux/io.h>
2011-04-23 00:03:11 +04:00
# include <linux/syscore_ops.h>
2011-02-23 14:38:16 +03:00
# include <linux/i2c/pxa-i2c.h>
2007-09-12 06:13:17 +04:00
2010-10-11 04:20:19 +04:00
# include <asm/mach/map.h>
2011-06-22 20:41:48 +04:00
# include <asm/suspend.h>
2008-08-05 19:14:15 +04:00
# include <mach/hardware.h>
# include <mach/pxa3xx-regs.h>
2008-08-07 14:05:25 +04:00
# include <mach/reset.h>
2008-08-05 19:14:15 +04:00
# include <mach/ohci.h>
# include <mach/pm.h>
# include <mach/dma.h>
2010-11-03 18:29:35 +03:00
# include <mach/smemc.h>
2007-09-12 06:13:17 +04:00
# include "generic.h"
# include "devices.h"
# include "clock.h"
2009-11-11 12:36:59 +03:00
# define PECR_IE(n) ((1 << ((n) * 2)) << 28)
# define PECR_IS(n) ((1 << ((n) * 2)) << 29)
2008-11-08 23:25:21 +03:00
static DEFINE_PXA3_CKEN ( pxa3xx_ffuart , FFUART , 14857000 , 1 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_btuart , BTUART , 14857000 , 1 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_stuart , STUART , 14857000 , 1 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_i2c , I2C , 32842000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_udc , UDC , 48000000 , 5 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_usbh , USBH , 48000000 , 0 ) ;
2009-11-04 15:14:39 +03:00
static DEFINE_PXA3_CKEN ( pxa3xx_u2d , USB2 , 48000000 , 0 ) ;
2008-11-08 23:25:21 +03:00
static DEFINE_PXA3_CKEN ( pxa3xx_keypad , KEYPAD , 32768 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_ssp1 , SSP1 , 13000000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_ssp2 , SSP2 , 13000000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_ssp3 , SSP3 , 13000000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_ssp4 , SSP4 , 13000000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_pwm0 , PWM0 , 13000000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_pwm1 , PWM1 , 13000000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_mmc1 , MMC1 , 19500000 , 0 ) ;
static DEFINE_PXA3_CKEN ( pxa3xx_mmc2 , MMC2 , 19500000 , 0 ) ;
2011-10-17 17:26:55 +04:00
static DEFINE_PXA3_CKEN ( pxa3xx_gpio , GPIO , 13000000 , 0 ) ;
2008-11-08 23:25:21 +03:00
2010-11-22 04:41:39 +03:00
static DEFINE_CK ( pxa3xx_lcd , LCD , & clk_pxa3xx_hsio_ops ) ;
2010-11-29 17:56:00 +03:00
static DEFINE_CK ( pxa3xx_smemc , SMC , & clk_pxa3xx_smemc_ops ) ;
2010-11-22 04:41:39 +03:00
static DEFINE_CK ( pxa3xx_camera , CAMERA , & clk_pxa3xx_hsio_ops ) ;
static DEFINE_CK ( pxa3xx_ac97 , AC97 , & clk_pxa3xx_ac97_ops ) ;
2010-11-22 05:49:55 +03:00
static DEFINE_CLK ( pxa3xx_pout , & clk_pxa3xx_pout_ops , 13000000 , 70 ) ;
2010-11-22 04:41:39 +03:00
2008-11-08 23:25:21 +03:00
static struct clk_lookup pxa3xx_clkregs [ ] = {
INIT_CLKREG ( & clk_pxa3xx_pout , NULL , " CLK_POUT " ) ,
/* Power I2C clock is always on */
2009-06-22 23:01:58 +04:00
INIT_CLKREG ( & clk_dummy , " pxa3xx-pwri2c.1 " , NULL ) ,
2008-11-08 23:25:21 +03:00
INIT_CLKREG ( & clk_pxa3xx_lcd , " pxa2xx-fb " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_camera , NULL , " CAMCLK " ) ,
INIT_CLKREG ( & clk_pxa3xx_ac97 , NULL , " AC97CLK " ) ,
INIT_CLKREG ( & clk_pxa3xx_ffuart , " pxa2xx-uart.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_btuart , " pxa2xx-uart.1 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_stuart , " pxa2xx-uart.2 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_stuart , " pxa2xx-ir " , " UARTCLK " ) ,
INIT_CLKREG ( & clk_pxa3xx_i2c , " pxa2xx-i2c.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_udc , " pxa27x-udc " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_usbh , " pxa27x-ohci " , NULL ) ,
2010-07-27 16:06:58 +04:00
INIT_CLKREG ( & clk_pxa3xx_u2d , " pxa3xx-u2d " , NULL ) ,
2008-11-08 23:25:21 +03:00
INIT_CLKREG ( & clk_pxa3xx_keypad , " pxa27x-keypad " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_ssp1 , " pxa27x-ssp.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_ssp2 , " pxa27x-ssp.1 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_ssp3 , " pxa27x-ssp.2 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_ssp4 , " pxa27x-ssp.3 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_pwm0 , " pxa27x-pwm.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_pwm1 , " pxa27x-pwm.1 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_mmc1 , " pxa2xx-mci.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa3xx_mmc2 , " pxa2xx-mci.1 " , NULL ) ,
2010-11-29 17:56:00 +03:00
INIT_CLKREG ( & clk_pxa3xx_smemc , " pxa2xx-pcmcia " , NULL ) ,
2011-10-17 17:26:55 +04:00
INIT_CLKREG ( & clk_pxa3xx_gpio , " pxa-gpio " , NULL ) ,
2007-09-12 06:13:17 +04:00
} ;
2008-01-08 01:18:30 +03:00
# ifdef CONFIG_PM
# define ISRAM_START 0x5c000000
# define ISRAM_SIZE SZ_256K
static void __iomem * sram ;
static unsigned long wakeup_src ;
/*
* Enter a standby mode ( S0D1C2 or S0D2C2 ) . Upon wakeup , the dynamic
* memory controller has to be reinitialised , so we place some code
* in the SRAM to perform this function .
*
* We disable FIQs across the standby - otherwise , we might receive a
* FIQ while the SDRAM is unavailable .
*/
static void pxa3xx_cpu_standby ( unsigned int pwrmode )
{
extern const char pm_enter_standby_start [ ] , pm_enter_standby_end [ ] ;
void ( * fn ) ( unsigned int ) = ( void __force * ) ( sram + 0x8000 ) ;
memcpy_toio ( sram + 0x8000 , pm_enter_standby_start ,
pm_enter_standby_end - pm_enter_standby_start ) ;
AD2D0SR = ~ 0 ;
AD2D1SR = ~ 0 ;
AD2D0ER = wakeup_src ;
AD2D1ER = 0 ;
ASCR = ASCR ;
ARSR = ARSR ;
local_fiq_disable ( ) ;
fn ( pwrmode ) ;
local_fiq_enable ( ) ;
AD2D0ER = 0 ;
AD2D1ER = 0 ;
}
2008-01-29 02:00:02 +03:00
/*
* NOTE : currently , the OBM ( OEM Boot Module ) binary comes along with
* PXA3xx development kits assumes that the resuming process continues
* with the address stored within the first 4 bytes of SDRAM . The PSPR
* register is used privately by BootROM and OBM , and _must_ be set to
* 0x5c014000 for the moment .
*/
static void pxa3xx_cpu_pm_suspend ( void )
{
volatile unsigned long * p = ( volatile void * ) 0xc0000000 ;
unsigned long saved_data = * p ;
2011-06-21 19:29:30 +04:00
# ifndef CONFIG_IWMMXT
u64 acc0 ;
2008-01-29 02:00:02 +03:00
2011-06-21 19:29:30 +04:00
asm volatile ( " mra %Q0, %R0, acc0 " : " =r " ( acc0 ) ) ;
# endif
2011-07-02 12:54:01 +04:00
extern int pxa3xx_finish_suspend ( unsigned long ) ;
2008-01-29 02:00:02 +03:00
/* resuming from D2 requires the HSIO2/BOOT/TPM clocks enabled */
CKENA | = ( 1 < < CKEN_BOOT ) | ( 1 < < CKEN_TPM ) ;
CKENB | = 1 < < ( CKEN_HSIO2 & 0x1f ) ;
/* clear and setup wakeup source */
AD3SR = ~ 0 ;
AD3ER = wakeup_src ;
ASCR = ASCR ;
ARSR = ARSR ;
PCFR | = ( 1u < < 13 ) ; /* L1_DIS */
PCFR & = ~ ( ( 1u < < 12 ) | ( 1u < < 1 ) ) ; /* L0_EN | SL_ROD */
PSPR = 0x5c014000 ;
/* overwrite with the resume address */
2011-02-06 20:41:26 +03:00
* p = virt_to_phys ( cpu_resume ) ;
2008-01-29 02:00:02 +03:00
2011-06-22 20:41:48 +04:00
cpu_suspend ( 0 , pxa3xx_finish_suspend ) ;
2008-01-29 02:00:02 +03:00
* p = saved_data ;
AD3ER = 0 ;
2011-06-21 19:29:30 +04:00
# ifndef CONFIG_IWMMXT
asm volatile ( " mar acc0, %Q0, %R0 " : " =r " ( acc0 ) ) ;
# endif
2008-01-29 02:00:02 +03:00
}
2008-01-08 01:18:30 +03:00
static void pxa3xx_cpu_pm_enter ( suspend_state_t state )
{
/*
* Don ' t sleep if no wakeup sources are defined
*/
2008-04-09 14:32:21 +04:00
if ( wakeup_src = = 0 ) {
printk ( KERN_ERR " Not suspending: no wakeup sources \n " ) ;
2008-01-08 01:18:30 +03:00
return ;
2008-04-09 14:32:21 +04:00
}
2008-01-08 01:18:30 +03:00
switch ( state ) {
case PM_SUSPEND_STANDBY :
pxa3xx_cpu_standby ( PXA3xx_PM_S0D2C2 ) ;
break ;
case PM_SUSPEND_MEM :
2008-01-29 02:00:02 +03:00
pxa3xx_cpu_pm_suspend ( ) ;
2008-01-08 01:18:30 +03:00
break ;
}
}
static int pxa3xx_cpu_pm_valid ( suspend_state_t state )
{
return state = = PM_SUSPEND_MEM | | state = = PM_SUSPEND_STANDBY ;
}
static struct pxa_cpu_pm_fns pxa3xx_cpu_pm_fns = {
. valid = pxa3xx_cpu_pm_valid ,
. enter = pxa3xx_cpu_pm_enter ,
2007-09-12 06:13:17 +04:00
} ;
2008-01-08 01:18:30 +03:00
static void __init pxa3xx_init_pm ( void )
{
sram = ioremap ( ISRAM_START , ISRAM_SIZE ) ;
if ( ! sram ) {
printk ( KERN_ERR " Unable to map ISRAM: disabling standby/suspend \n " ) ;
return ;
}
/*
* Since we copy wakeup code into the SRAM , we need to ensure
* that it is preserved over the low power modes . Note : bit 8
* is undocumented in the developer manual , but must be set .
*/
AD1R | = ADXR_L2 | ADXR_R0 ;
AD2R | = ADXR_L2 | ADXR_R0 ;
AD3R | = ADXR_L2 | ADXR_R0 ;
/*
* Clear the resume enable registers .
*/
AD1D0ER = 0 ;
AD2D0ER = 0 ;
AD2D1ER = 0 ;
AD3ER = 0 ;
pxa_cpu_pm_fns = & pxa3xx_cpu_pm_fns ;
}
2010-11-29 13:18:26 +03:00
static int pxa3xx_set_wake ( struct irq_data * d , unsigned int on )
2008-01-08 01:18:30 +03:00
{
unsigned long flags , mask = 0 ;
2010-11-29 13:18:26 +03:00
switch ( d - > irq ) {
2008-01-08 01:18:30 +03:00
case IRQ_SSP3 :
mask = ADXER_MFP_WSSP3 ;
break ;
case IRQ_MSL :
mask = ADXER_WMSL0 ;
break ;
case IRQ_USBH2 :
case IRQ_USBH1 :
mask = ADXER_WUSBH ;
break ;
case IRQ_KEYPAD :
mask = ADXER_WKP ;
break ;
case IRQ_AC97 :
mask = ADXER_MFP_WAC97 ;
break ;
case IRQ_USIM :
mask = ADXER_WUSIM0 ;
break ;
case IRQ_SSP2 :
mask = ADXER_MFP_WSSP2 ;
break ;
case IRQ_I2C :
mask = ADXER_MFP_WI2C ;
break ;
case IRQ_STUART :
mask = ADXER_MFP_WUART3 ;
break ;
case IRQ_BTUART :
mask = ADXER_MFP_WUART2 ;
break ;
case IRQ_FFUART :
mask = ADXER_MFP_WUART1 ;
break ;
case IRQ_MMC :
mask = ADXER_MFP_WMMC1 ;
break ;
case IRQ_SSP :
mask = ADXER_MFP_WSSP1 ;
break ;
case IRQ_RTCAlrm :
mask = ADXER_WRTC ;
break ;
case IRQ_SSP4 :
mask = ADXER_MFP_WSSP4 ;
break ;
case IRQ_TSI :
mask = ADXER_WTSI ;
break ;
case IRQ_USIM2 :
mask = ADXER_WUSIM1 ;
break ;
case IRQ_MMC2 :
mask = ADXER_MFP_WMMC2 ;
break ;
case IRQ_NAND :
mask = ADXER_MFP_WFLASH ;
break ;
case IRQ_USB2 :
mask = ADXER_WUSB2 ;
break ;
case IRQ_WAKEUP0 :
mask = ADXER_WEXTWAKE0 ;
break ;
case IRQ_WAKEUP1 :
mask = ADXER_WEXTWAKE1 ;
break ;
case IRQ_MMC3 :
mask = ADXER_MFP_GEN12 ;
break ;
2008-04-23 13:28:18 +04:00
default :
return - EINVAL ;
2008-01-08 01:18:30 +03:00
}
local_irq_save ( flags ) ;
if ( on )
wakeup_src | = mask ;
else
wakeup_src & = ~ mask ;
local_irq_restore ( flags ) ;
return 0 ;
}
# else
static inline void pxa3xx_init_pm ( void ) { }
2008-03-04 09:19:58 +03:00
# define pxa3xx_set_wake NULL
2008-01-08 01:18:30 +03:00
# endif
2010-11-29 13:18:26 +03:00
static void pxa_ack_ext_wakeup ( struct irq_data * d )
2009-11-11 12:36:59 +03:00
{
2010-11-29 13:18:26 +03:00
PECR | = PECR_IS ( d - > irq - IRQ_WAKEUP0 ) ;
2009-11-11 12:36:59 +03:00
}
2010-11-29 13:18:26 +03:00
static void pxa_mask_ext_wakeup ( struct irq_data * d )
2009-11-11 12:36:59 +03:00
{
2011-04-27 18:48:04 +04:00
pxa_mask_irq ( d ) ;
2010-11-29 13:18:26 +03:00
PECR & = ~ PECR_IE ( d - > irq - IRQ_WAKEUP0 ) ;
2009-11-11 12:36:59 +03:00
}
2010-11-29 13:18:26 +03:00
static void pxa_unmask_ext_wakeup ( struct irq_data * d )
2009-11-11 12:36:59 +03:00
{
2011-04-27 18:48:04 +04:00
pxa_unmask_irq ( d ) ;
2010-11-29 13:18:26 +03:00
PECR | = PECR_IE ( d - > irq - IRQ_WAKEUP0 ) ;
2009-11-11 12:36:59 +03:00
}
2010-11-29 13:18:26 +03:00
static int pxa_set_ext_wakeup_type ( struct irq_data * d , unsigned int flow_type )
2010-06-13 12:31:48 +04:00
{
if ( flow_type & IRQ_TYPE_EDGE_RISING )
2010-11-29 13:18:26 +03:00
PWER | = 1 < < ( d - > irq - IRQ_WAKEUP0 ) ;
2010-06-13 12:31:48 +04:00
if ( flow_type & IRQ_TYPE_EDGE_FALLING )
2010-11-29 13:18:26 +03:00
PWER | = 1 < < ( d - > irq - IRQ_WAKEUP0 + 2 ) ;
2010-06-13 12:31:48 +04:00
return 0 ;
}
2009-11-11 12:36:59 +03:00
static struct irq_chip pxa_ext_wakeup_chip = {
. name = " WAKEUP " ,
2010-11-29 13:18:26 +03:00
. irq_ack = pxa_ack_ext_wakeup ,
. irq_mask = pxa_mask_ext_wakeup ,
. irq_unmask = pxa_unmask_ext_wakeup ,
. irq_set_type = pxa_set_ext_wakeup_type ,
2009-11-11 12:36:59 +03:00
} ;
2011-10-17 16:37:52 +04:00
static void __init pxa_init_ext_wakeup_irq ( int ( * fn ) ( struct irq_data * ,
unsigned int ) )
2009-11-11 12:36:59 +03:00
{
int irq ;
for ( irq = IRQ_WAKEUP0 ; irq < = IRQ_WAKEUP1 ; irq + + ) {
2011-03-24 15:35:09 +03:00
irq_set_chip_and_handler ( irq , & pxa_ext_wakeup_chip ,
handle_edge_irq ) ;
2009-11-11 12:36:59 +03:00
set_irq_flags ( irq , IRQF_VALID ) ;
}
2010-11-29 13:18:26 +03:00
pxa_ext_wakeup_chip . irq_set_wake = fn ;
2009-11-11 12:36:59 +03:00
}
2007-09-12 06:13:17 +04:00
void __init pxa3xx_init_irq ( void )
{
/* enable CP6 access */
u32 value ;
__asm__ __volatile__ ( " mrc p15, 0, %0, c15, c1, 0 \n " : " =r " ( value ) ) ;
value | = ( 1 < < 6 ) ;
__asm__ __volatile__ ( " mcr p15, 0, %0, c15, c1, 0 \n " : : " r " ( value ) ) ;
2008-03-04 09:19:58 +03:00
pxa_init_irq ( 56 , pxa3xx_set_wake ) ;
2009-11-11 12:36:59 +03:00
pxa_init_ext_wakeup_irq ( pxa3xx_set_wake ) ;
2007-09-12 06:13:17 +04:00
}
2010-10-11 04:20:19 +04:00
static struct map_desc pxa3xx_io_desc [ ] __initdata = {
{ /* Mem Ctl */
2011-10-02 00:03:45 +04:00
. virtual = ( unsigned long ) SMEMC_VIRT ,
2010-11-03 18:29:35 +03:00
. pfn = __phys_to_pfn ( PXA3XX_SMEMC_BASE ) ,
2010-10-11 04:20:19 +04:00
. length = 0x00200000 ,
. type = MT_DEVICE
}
} ;
void __init pxa3xx_map_io ( void )
{
pxa_map_io ( ) ;
iotable_init ( ARRAY_AND_SIZE ( pxa3xx_io_desc ) ) ;
pxa3xx_get_clk_frequency_khz ( 1 ) ;
}
2007-09-12 06:13:17 +04:00
/*
* device registration specific to PXA3xx .
*/
2008-08-17 09:23:05 +04:00
void __init pxa3xx_set_i2c_power_info ( struct i2c_pxa_platform_data * info )
{
2008-11-28 10:24:12 +03:00
pxa_register_device ( & pxa3xx_device_i2c_power , info ) ;
2008-08-17 09:23:05 +04:00
}
2007-09-12 06:13:17 +04:00
static struct platform_device * devices [ ] __initdata = {
2011-10-17 16:37:52 +04:00
& pxa_device_gpio ,
2009-04-21 21:19:36 +04:00
& pxa27x_device_udc ,
2010-06-13 20:43:00 +04:00
& pxa_device_pmu ,
2007-09-12 06:13:17 +04:00
& pxa_device_i2s ,
2010-03-17 23:15:21 +03:00
& pxa_device_asoc_ssp1 ,
& pxa_device_asoc_ssp2 ,
& pxa_device_asoc_ssp3 ,
& pxa_device_asoc_ssp4 ,
& pxa_device_asoc_platform ,
2008-11-14 01:50:56 +03:00
& sa1100_device_rtc ,
2007-09-12 06:13:17 +04:00
& pxa_device_rtc ,
2007-12-10 12:54:36 +03:00
& pxa27x_device_ssp1 ,
& pxa27x_device_ssp2 ,
& pxa27x_device_ssp3 ,
& pxa3xx_device_ssp4 ,
2008-04-14 00:44:04 +04:00
& pxa27x_device_pwm0 ,
& pxa27x_device_pwm1 ,
2007-09-12 06:13:17 +04:00
} ;
static int __init pxa3xx_init ( void )
{
2011-04-23 00:03:11 +04:00
int ret = 0 ;
2007-09-12 06:13:17 +04:00
if ( cpu_is_pxa3xx ( ) ) {
2008-07-29 10:26:00 +04:00
reset_status = ARSR ;
2008-02-08 17:02:03 +03:00
/*
* clear RDH bit every time after reset
*
* Note : the last 3 bits DxS are write - 1 - to - clear so carefully
* preserve them here in case they will be referenced later
*/
ASCR & = ~ ( ASCR_RDH | ASCR_D1S | ASCR_D2S | ASCR_D3S ) ;
2010-01-12 15:28:00 +03:00
clkdev_add_table ( pxa3xx_clkregs , ARRAY_SIZE ( pxa3xx_clkregs ) ) ;
2007-09-12 06:13:17 +04:00
2009-01-02 11:26:33 +03:00
if ( ( ret = pxa_init_dma ( IRQ_DMA , 32 ) ) )
2007-09-12 06:13:17 +04:00
return ret ;
2008-01-08 01:18:30 +03:00
pxa3xx_init_pm ( ) ;
2011-04-23 00:03:11 +04:00
register_syscore_ops ( & pxa_irq_syscore_ops ) ;
register_syscore_ops ( & pxa3xx_mfp_syscore_ops ) ;
register_syscore_ops ( & pxa3xx_clock_syscore_ops ) ;
2008-01-29 02:00:02 +03:00
ret = platform_add_devices ( devices , ARRAY_SIZE ( devices ) ) ;
2007-09-12 06:13:17 +04:00
}
2008-01-29 02:00:02 +03:00
return ret ;
2007-09-12 06:13:17 +04:00
}
2008-04-19 13:59:24 +04:00
postcore_initcall ( pxa3xx_init ) ;