2007-09-11 19:13:17 -07:00
/*
* linux / arch / arm / mach - pxa / pxa3xx . c
*
* code specific to pxa3xx aka Monahans
*
* Copyright ( C ) 2006 Marvell International Ltd .
*
2007-10-30 08:01:38 +01:00
* 2007 - 09 - 02 : eric miao < eric . miao @ marvell . com >
2007-09-11 19:13:17 -07: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>
2013-04-09 18:12:04 +08:00
# include <linux/gpio-pxa.h>
2007-09-11 19:13:17 -07:00
# include <linux/pm.h>
# include <linux/platform_device.h>
# include <linux/irq.h>
2008-01-07 22:18:30 +00:00
# include <linux/io.h>
2012-07-25 17:52:52 +02:00
# include <linux/of.h>
2011-04-22 22:03:11 +02:00
# include <linux/syscore_ops.h>
2011-02-23 12:38:16 +01:00
# include <linux/i2c/pxa-i2c.h>
2007-09-11 19:13:17 -07:00
2010-10-11 02:20:19 +02:00
# include <asm/mach/map.h>
2011-06-22 17:41:48 +01:00
# include <asm/suspend.h>
2008-08-05 16:14:15 +01:00
# include <mach/hardware.h>
# include <mach/pxa3xx-regs.h>
2008-08-07 11:05:25 +01:00
# include <mach/reset.h>
2012-08-24 15:16:48 +02:00
# include <linux/platform_data/usb-ohci-pxa27x.h>
2015-01-30 10:45:33 +01:00
# include "pm.h"
2008-08-05 16:14:15 +01:00
# include <mach/dma.h>
2010-11-03 16:29:35 +01:00
# include <mach/smemc.h>
2012-01-03 16:53:48 -06:00
# include <mach/irqs.h>
2007-09-11 19:13:17 -07:00
# include "generic.h"
# include "devices.h"
2009-11-11 11:36:59 +02:00
# define PECR_IE(n) ((1 << ((n) * 2)) << 28)
# define PECR_IS(n) ((1 << ((n) * 2)) << 29)
2012-07-22 19:50:22 +02:00
extern void __init pxa_dt_irq_init ( int ( * fn ) ( struct irq_data * , unsigned int ) ) ;
2008-01-07 22:18:30 +00:00
2015-08-23 21:13:57 +02:00
/*
* NAND NFC : DFI bus arbitration subset
*/
# define NDCR (*(volatile u32 __iomem*)(NAND_VIRT + 0))
# define NDCR_ND_ARB_EN (1 << 12)
# define NDCR_ND_ARB_CNTL (1 << 19)
2015-10-12 15:44:49 +02:00
# ifdef CONFIG_PM
# define ISRAM_START 0x5c000000
# define ISRAM_SIZE SZ_256K
2008-01-07 22:18:30 +00:00
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 )
{
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-28 23:00:02 +00: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 16:29:30 +01:00
# ifndef CONFIG_IWMMXT
u64 acc0 ;
2008-01-28 23:00:02 +00:00
ARM: pxa: fix building without IWMMXT
When CONFIG_IWMMXT, the pxa3xx and pxa27x suspend/resume code
emits some xscale specific instructions, which are rejected
by the assembler, because gcc is built with -march=armv5
-mtune=xscale and passes that option to the assembler:
/tmp/cciHumzr.s:553: Error: selected processor does not support ARM mode `mra r2,r3,acc0'
/tmp/cciHumzr.s:605: Error: selected processor does not support ARM mode `mar acc0,r2,r3'
make[3]: *** [arch/arm/mach-pxa/pxa3xx.o] Error 1
/tmp/cci5MUNu.s:326: Error: selected processor does not support ARM mode `mra r2,r3,acc0'
/tmp/cci5MUNu.s:367: Error: selected processor does not support ARM mode `mar acc0,r2,r3'
make[3]: *** [arch/arm/mach-pxa/pxa27x.o] Error 1
Overriding with -Wa,-march=xscale no longer works, so instead
I'm adding an explict ".arch_extension" directive in all four inline
assembly statements, which should work even if they end up in a different
order in the assembly output.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Robert Jarzmik <robert.jarzmik@free.fr>
2016-01-29 15:06:31 +01:00
asm volatile ( " .arch_extension xscale \n \t "
" mra %Q0, %R0, acc0 " : " =r " ( acc0 ) ) ;
2011-06-21 16:29:30 +01:00
# endif
2008-01-28 23:00:02 +00: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 17:41:26 +00:00
* p = virt_to_phys ( cpu_resume ) ;
2008-01-28 23:00:02 +00:00
2011-06-22 17:41:48 +01:00
cpu_suspend ( 0 , pxa3xx_finish_suspend ) ;
2008-01-28 23:00:02 +00:00
* p = saved_data ;
AD3ER = 0 ;
2011-06-21 16:29:30 +01:00
# ifndef CONFIG_IWMMXT
ARM: pxa: fix building without IWMMXT
When CONFIG_IWMMXT, the pxa3xx and pxa27x suspend/resume code
emits some xscale specific instructions, which are rejected
by the assembler, because gcc is built with -march=armv5
-mtune=xscale and passes that option to the assembler:
/tmp/cciHumzr.s:553: Error: selected processor does not support ARM mode `mra r2,r3,acc0'
/tmp/cciHumzr.s:605: Error: selected processor does not support ARM mode `mar acc0,r2,r3'
make[3]: *** [arch/arm/mach-pxa/pxa3xx.o] Error 1
/tmp/cci5MUNu.s:326: Error: selected processor does not support ARM mode `mra r2,r3,acc0'
/tmp/cci5MUNu.s:367: Error: selected processor does not support ARM mode `mar acc0,r2,r3'
make[3]: *** [arch/arm/mach-pxa/pxa27x.o] Error 1
Overriding with -Wa,-march=xscale no longer works, so instead
I'm adding an explict ".arch_extension" directive in all four inline
assembly statements, which should work even if they end up in a different
order in the assembly output.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Robert Jarzmik <robert.jarzmik@free.fr>
2016-01-29 15:06:31 +01:00
asm volatile ( " .arch_extension xscale \n \t "
" mar acc0, %Q0, %R0 " : " =r " ( acc0 ) ) ;
2011-06-21 16:29:30 +01:00
# endif
2008-01-28 23:00:02 +00:00
}
2008-01-07 22:18:30 +00:00
static void pxa3xx_cpu_pm_enter ( suspend_state_t state )
{
/*
* Don ' t sleep if no wakeup sources are defined
*/
2008-04-09 11:32:21 +01:00
if ( wakeup_src = = 0 ) {
printk ( KERN_ERR " Not suspending: no wakeup sources \n " ) ;
2008-01-07 22:18:30 +00:00
return ;
2008-04-09 11:32:21 +01:00
}
2008-01-07 22:18:30 +00:00
switch ( state ) {
case PM_SUSPEND_STANDBY :
pxa3xx_cpu_standby ( PXA3xx_PM_S0D2C2 ) ;
break ;
case PM_SUSPEND_MEM :
2008-01-28 23:00:02 +00:00
pxa3xx_cpu_pm_suspend ( ) ;
2008-01-07 22:18:30 +00: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-11 19:13:17 -07:00
} ;
2008-01-07 22:18:30 +00: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 11:18:26 +01:00
static int pxa3xx_set_wake ( struct irq_data * d , unsigned int on )
2008-01-07 22:18:30 +00:00
{
unsigned long flags , mask = 0 ;
2010-11-29 11:18:26 +01:00
switch ( d - > irq ) {
2008-01-07 22:18:30 +00: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 10:28:18 +01:00
default :
return - EINVAL ;
2008-01-07 22:18:30 +00: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 14:19:58 +08:00
# define pxa3xx_set_wake NULL
2008-01-07 22:18:30 +00:00
# endif
2010-11-29 11:18:26 +01:00
static void pxa_ack_ext_wakeup ( struct irq_data * d )
2009-11-11 11:36:59 +02:00
{
2010-11-29 11:18:26 +01:00
PECR | = PECR_IS ( d - > irq - IRQ_WAKEUP0 ) ;
2009-11-11 11:36:59 +02:00
}
2010-11-29 11:18:26 +01:00
static void pxa_mask_ext_wakeup ( struct irq_data * d )
2009-11-11 11:36:59 +02:00
{
2011-04-27 22:48:04 +08:00
pxa_mask_irq ( d ) ;
2010-11-29 11:18:26 +01:00
PECR & = ~ PECR_IE ( d - > irq - IRQ_WAKEUP0 ) ;
2009-11-11 11:36:59 +02:00
}
2010-11-29 11:18:26 +01:00
static void pxa_unmask_ext_wakeup ( struct irq_data * d )
2009-11-11 11:36:59 +02:00
{
2011-04-27 22:48:04 +08:00
pxa_unmask_irq ( d ) ;
2010-11-29 11:18:26 +01:00
PECR | = PECR_IE ( d - > irq - IRQ_WAKEUP0 ) ;
2009-11-11 11:36:59 +02:00
}
2010-11-29 11:18:26 +01:00
static int pxa_set_ext_wakeup_type ( struct irq_data * d , unsigned int flow_type )
2010-06-13 11:31:48 +03:00
{
if ( flow_type & IRQ_TYPE_EDGE_RISING )
2010-11-29 11:18:26 +01:00
PWER | = 1 < < ( d - > irq - IRQ_WAKEUP0 ) ;
2010-06-13 11:31:48 +03:00
if ( flow_type & IRQ_TYPE_EDGE_FALLING )
2010-11-29 11:18:26 +01:00
PWER | = 1 < < ( d - > irq - IRQ_WAKEUP0 + 2 ) ;
2010-06-13 11:31:48 +03:00
return 0 ;
}
2009-11-11 11:36:59 +02:00
static struct irq_chip pxa_ext_wakeup_chip = {
. name = " WAKEUP " ,
2010-11-29 11:18:26 +01: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 11:36:59 +02:00
} ;
2011-10-17 20:37:52 +08:00
static void __init pxa_init_ext_wakeup_irq ( int ( * fn ) ( struct irq_data * ,
unsigned int ) )
2009-11-11 11:36:59 +02:00
{
int irq ;
for ( irq = IRQ_WAKEUP0 ; irq < = IRQ_WAKEUP1 ; irq + + ) {
2011-03-24 13:35:09 +01:00
irq_set_chip_and_handler ( irq , & pxa_ext_wakeup_chip ,
handle_edge_irq ) ;
2015-07-27 15:55:13 -05:00
irq_clear_status_flags ( irq , IRQ_NOREQUEST ) ;
2009-11-11 11:36:59 +02:00
}
2010-11-29 11:18:26 +01:00
pxa_ext_wakeup_chip . irq_set_wake = fn ;
2009-11-11 11:36:59 +02:00
}
2012-07-22 19:50:22 +02:00
static void __init __pxa3xx_init_irq ( void )
2007-09-11 19:13:17 -07:00
{
/* 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 ) ) ;
2009-11-11 11:36:59 +02:00
pxa_init_ext_wakeup_irq ( pxa3xx_set_wake ) ;
2007-09-11 19:13:17 -07:00
}
2012-07-22 19:50:22 +02:00
void __init pxa3xx_init_irq ( void )
{
__pxa3xx_init_irq ( ) ;
pxa_init_irq ( 56 , pxa3xx_set_wake ) ;
}
2012-08-20 13:46:51 +08:00
# ifdef CONFIG_OF
2012-07-22 19:50:22 +02:00
void __init pxa3xx_dt_init_irq ( void )
{
__pxa3xx_init_irq ( ) ;
pxa_dt_irq_init ( pxa3xx_set_wake ) ;
}
2012-08-20 13:46:51 +08:00
# endif /* CONFIG_OF */
2012-07-22 19:50:22 +02:00
2010-10-11 02:20:19 +02:00
static struct map_desc pxa3xx_io_desc [ ] __initdata = {
{ /* Mem Ctl */
2011-10-01 22:03:45 +02:00
. virtual = ( unsigned long ) SMEMC_VIRT ,
2010-11-03 16:29:35 +01:00
. pfn = __phys_to_pfn ( PXA3XX_SMEMC_BASE ) ,
2014-07-11 13:00:36 +02:00
. length = SMEMC_SIZE ,
2010-10-11 02:20:19 +02:00
. type = MT_DEVICE
2015-08-23 21:13:57 +02:00
} , {
. virtual = ( unsigned long ) NAND_VIRT ,
. pfn = __phys_to_pfn ( NAND_PHYS ) ,
. length = NAND_SIZE ,
. type = MT_DEVICE
} ,
2010-10-11 02:20:19 +02:00
} ;
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-11 19:13:17 -07:00
/*
* device registration specific to PXA3xx .
*/
2008-08-17 06:23:05 +01:00
void __init pxa3xx_set_i2c_power_info ( struct i2c_pxa_platform_data * info )
{
2008-11-28 15:24:12 +08:00
pxa_register_device ( & pxa3xx_device_i2c_power , info ) ;
2008-08-17 06:23:05 +01:00
}
2013-04-09 18:12:04 +08:00
static struct pxa_gpio_platform_data pxa3xx_gpio_pdata = {
. irq_base = PXA_GPIO_TO_IRQ ( 0 ) ,
} ;
2007-09-11 19:13:17 -07:00
static struct platform_device * devices [ ] __initdata = {
2009-04-21 19:19:36 +02:00
& pxa27x_device_udc ,
2010-06-14 00:43:00 +08:00
& pxa_device_pmu ,
2007-09-11 19:13:17 -07:00
& pxa_device_i2s ,
2010-03-17 20:15:21 +00:00
& pxa_device_asoc_ssp1 ,
& pxa_device_asoc_ssp2 ,
& pxa_device_asoc_ssp3 ,
& pxa_device_asoc_ssp4 ,
& pxa_device_asoc_platform ,
2007-09-11 19:13:17 -07:00
& pxa_device_rtc ,
2014-08-13 21:59:19 +02:00
& pxa3xx_device_ssp1 ,
& pxa3xx_device_ssp2 ,
& pxa3xx_device_ssp3 ,
2007-12-10 17:54:36 +08:00
& pxa3xx_device_ssp4 ,
2008-04-13 21:44:04 +01:00
& pxa27x_device_pwm0 ,
& pxa27x_device_pwm1 ,
2007-09-11 19:13:17 -07:00
} ;
static int __init pxa3xx_init ( void )
{
2011-04-22 22:03:11 +02:00
int ret = 0 ;
2007-09-11 19:13:17 -07:00
if ( cpu_is_pxa3xx ( ) ) {
2008-07-29 14:26:00 +08:00
reset_status = ARSR ;
2008-02-08 15:02:03 +01: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 ) ;
2015-08-23 21:13:57 +02:00
/*
* Disable DFI bus arbitration , to prevent a system bus lock if
* somebody disables the NAND clock ( unused clock ) while this
* bit remains set .
*/
NDCR = ( NDCR & ~ NDCR_ND_ARB_EN ) | NDCR_ND_ARB_CNTL ;
2009-01-02 16:26:33 +08:00
if ( ( ret = pxa_init_dma ( IRQ_DMA , 32 ) ) )
2007-09-11 19:13:17 -07:00
return ret ;
2008-01-07 22:18:30 +00:00
pxa3xx_init_pm ( ) ;
2011-04-22 22:03:11 +02:00
register_syscore_ops ( & pxa_irq_syscore_ops ) ;
register_syscore_ops ( & pxa3xx_mfp_syscore_ops ) ;
2008-01-28 23:00:02 +00:00
2013-04-07 16:44:33 +08:00
if ( of_have_populated_dt ( ) )
return 0 ;
2016-02-15 21:57:47 +01:00
pxa2xx_set_dmac_info ( 32 , 100 ) ;
2013-04-07 16:44:33 +08:00
ret = platform_add_devices ( devices , ARRAY_SIZE ( devices ) ) ;
if ( ret )
return ret ;
2013-04-09 18:12:04 +08:00
if ( cpu_is_pxa300 ( ) | | cpu_is_pxa310 ( ) | | cpu_is_pxa320 ( ) ) {
platform_device_add_data ( & pxa3xx_device_gpio ,
& pxa3xx_gpio_pdata ,
sizeof ( pxa3xx_gpio_pdata ) ) ;
2013-04-07 16:44:33 +08:00
ret = platform_device_register ( & pxa3xx_device_gpio ) ;
2013-04-09 18:12:04 +08:00
}
2007-09-11 19:13:17 -07:00
}
2008-01-28 23:00:02 +00:00
return ret ;
2007-09-11 19:13:17 -07:00
}
2008-04-19 10:59:24 +01:00
postcore_initcall ( pxa3xx_init ) ;