2005-04-16 15:20:36 -07:00
/*
* linux / arch / arm / mach - pxa / pxa25x . c
*
* Author : Nicolas Pitre
* Created : Jun 15 , 2001
* Copyright : MontaVista Software Inc .
*
* Code specific to PXA21x / 25 x / 26 x variants .
*
* 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 .
*
* Since this file should be linked before any other machine specific file ,
* the __initcall ( ) here will be executed first . This serves as default
* initialization stuff for PXA machines which can be overridden later if
* need be .
*/
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/init.h>
2007-05-15 10:39:49 +01:00
# include <linux/platform_device.h>
2007-10-18 03:04:39 -07:00
# include <linux/suspend.h>
2008-01-28 23:00:02 +00:00
# include <linux/sysdev.h>
2005-04-16 15:20:36 -07:00
2008-08-05 16:14:15 +01:00
# include <mach/hardware.h>
# include <mach/irqs.h>
2009-01-06 17:37:37 +08:00
# include <mach/gpio.h>
2009-01-02 23:17:22 +08:00
# include <mach/pxa25x.h>
2008-08-07 11:05:25 +01:00
# include <mach/reset.h>
2008-08-05 16:14:15 +01:00
# include <mach/pm.h>
# include <mach/dma.h>
2005-04-16 15:20:36 -07:00
# include "generic.h"
2007-05-15 15:39:36 +01:00
# include "devices.h"
2007-08-20 10:18:02 +01:00
# include "clock.h"
2005-04-16 15:20:36 -07:00
/*
* Various clock factors driven by the CCCR register .
*/
/* Crystal Frequency to Memory Frequency Multiplier (L) */
static unsigned char L_clk_mult [ 32 ] = { 0 , 27 , 32 , 36 , 40 , 45 , 0 , } ;
/* Memory Frequency to Run Mode Frequency Multiplier (M) */
static unsigned char M_clk_mult [ 4 ] = { 0 , 1 , 2 , 4 } ;
/* Run Mode Frequency to Turbo Mode Frequency Multiplier (N) */
/* Note: we store the value N * 2 here. */
static unsigned char N2_clk_mult [ 8 ] = { 0 , 0 , 2 , 3 , 4 , 0 , 6 , 0 } ;
/* Crystal clock */
# define BASE_CLK 3686400
/*
* Get the clock frequency as reflected by CCCR and the turbo flag .
* We assume these values have been applied via a fcs .
* If info is not 0 we also display the current settings .
*/
2007-08-20 10:07:44 +01:00
unsigned int pxa25x_get_clk_frequency_khz ( int info )
2005-04-16 15:20:36 -07:00
{
unsigned long cccr , turbo ;
unsigned int l , L , m , M , n2 , N ;
cccr = CCCR ;
asm ( " mrc \t p14, 0, %0, c6, c0, 0 " : " =r " ( turbo ) ) ;
l = L_clk_mult [ ( cccr > > 0 ) & 0x1f ] ;
m = M_clk_mult [ ( cccr > > 5 ) & 0x03 ] ;
n2 = N2_clk_mult [ ( cccr > > 7 ) & 0x07 ] ;
L = l * BASE_CLK ;
M = m * L ;
N = n2 * M / 2 ;
if ( info )
{
L + = 5000 ;
printk ( KERN_INFO " Memory clock: %d.%02dMHz (*%d) \n " ,
L / 1000000 , ( L % 1000000 ) / 10000 , l ) ;
M + = 5000 ;
printk ( KERN_INFO " Run Mode clock: %d.%02dMHz (*%d) \n " ,
M / 1000000 , ( M % 1000000 ) / 10000 , m ) ;
N + = 5000 ;
printk ( KERN_INFO " Turbo Mode clock: %d.%02dMHz (*%d.%d, %sactive) \n " ,
N / 1000000 , ( N % 1000000 ) / 10000 , n2 / 2 , ( n2 % 2 ) * 5 ,
( turbo & 1 ) ? " " : " in " ) ;
}
return ( turbo & 1 ) ? ( N / 1000 ) : ( M / 1000 ) ;
}
/*
* Return the current memory clock frequency in units of 10 kHz
*/
2007-08-20 10:07:44 +01:00
unsigned int pxa25x_get_memclk_frequency_10khz ( void )
2005-04-16 15:20:36 -07:00
{
return L_clk_mult [ ( CCCR > > 0 ) & 0x1f ] * BASE_CLK / 10000 ;
}
2007-08-20 10:18:02 +01:00
static unsigned long clk_pxa25x_lcd_getrate ( struct clk * clk )
{
return pxa25x_get_memclk_frequency_10khz ( ) * 10000 ;
}
static const struct clkops clk_pxa25x_lcd_ops = {
. enable = clk_cken_enable ,
. disable = clk_cken_disable ,
. getrate = clk_pxa25x_lcd_getrate ,
} ;
2008-07-08 10:32:08 +01:00
static unsigned long gpio12_config_32k [ ] = {
GPIO12_32KHz ,
} ;
static unsigned long gpio12_config_gpio [ ] = {
GPIO12_GPIO ,
} ;
static void clk_gpio12_enable ( struct clk * clk )
{
pxa2xx_mfp_config ( gpio12_config_32k , 1 ) ;
}
static void clk_gpio12_disable ( struct clk * clk )
{
pxa2xx_mfp_config ( gpio12_config_gpio , 1 ) ;
}
static const struct clkops clk_pxa25x_gpio12_ops = {
. enable = clk_gpio12_enable ,
. disable = clk_gpio12_disable ,
} ;
2008-07-08 10:32:50 +01:00
static unsigned long gpio11_config_3m6 [ ] = {
GPIO11_3_6MHz ,
} ;
static unsigned long gpio11_config_gpio [ ] = {
GPIO11_GPIO ,
} ;
static void clk_gpio11_enable ( struct clk * clk )
{
pxa2xx_mfp_config ( gpio11_config_3m6 , 1 ) ;
}
static void clk_gpio11_disable ( struct clk * clk )
{
pxa2xx_mfp_config ( gpio11_config_gpio , 1 ) ;
}
static const struct clkops clk_pxa25x_gpio11_ops = {
. enable = clk_gpio11_enable ,
. disable = clk_gpio11_disable ,
} ;
2007-08-20 10:18:02 +01:00
/*
* 3.6864 MHz - > OST , GPIO , SSP , PWM , PLLs ( 95.842 MHz , 147.456 MHz )
* 95.842 MHz - > MMC 19.169 MHz , I2C 31.949 MHz , FICP 47.923 MHz , USB 47.923 MHz
* 147.456 MHz - > UART 14.7456 MHz , AC97 12.288 MHz , I2S 5.672 MHz ( allegedly )
*/
2008-11-08 20:25:21 +00:00
static DEFINE_CKEN ( pxa25x_hwuart , HWUART , 14745600 , 1 ) ;
static struct clk_lookup pxa25x_hwuart_clkreg =
INIT_CLKREG ( & clk_pxa25x_hwuart , " pxa2xx-uart.3 " , NULL ) ;
2008-01-27 23:11:48 +01:00
2008-06-30 19:47:59 +01:00
/*
2008-07-26 00:52:36 +01:00
* PXA 2 xx clock declarations .
2008-06-30 19:47:59 +01:00
*/
2008-11-08 20:25:21 +00:00
static DEFINE_CK ( pxa25x_lcd , LCD , & clk_pxa25x_lcd_ops ) ;
static DEFINE_CKEN ( pxa25x_ffuart , FFUART , 14745600 , 1 ) ;
static DEFINE_CKEN ( pxa25x_btuart , BTUART , 14745600 , 1 ) ;
static DEFINE_CKEN ( pxa25x_stuart , STUART , 14745600 , 1 ) ;
static DEFINE_CKEN ( pxa25x_usb , USB , 47923000 , 5 ) ;
static DEFINE_CLK ( pxa25x_gpio11 , & clk_pxa25x_gpio11_ops , 3686400 , 0 ) ;
static DEFINE_CLK ( pxa25x_gpio12 , & clk_pxa25x_gpio12_ops , 32768 , 0 ) ;
static DEFINE_CKEN ( pxa25x_mmc , MMC , 19169000 , 0 ) ;
static DEFINE_CKEN ( pxa25x_i2c , I2C , 31949000 , 0 ) ;
static DEFINE_CKEN ( pxa25x_ssp , SSP , 3686400 , 0 ) ;
static DEFINE_CKEN ( pxa25x_nssp , NSSP , 3686400 , 0 ) ;
static DEFINE_CKEN ( pxa25x_assp , ASSP , 3686400 , 0 ) ;
static DEFINE_CKEN ( pxa25x_pwm0 , PWM0 , 3686400 , 0 ) ;
static DEFINE_CKEN ( pxa25x_pwm1 , PWM1 , 3686400 , 0 ) ;
static DEFINE_CKEN ( pxa25x_ac97 , AC97 , 24576000 , 0 ) ;
static DEFINE_CKEN ( pxa25x_i2s , I2S , 14745600 , 0 ) ;
static DEFINE_CKEN ( pxa25x_ficp , FICP , 47923000 , 0 ) ;
static struct clk_lookup pxa25x_clkregs [ ] = {
INIT_CLKREG ( & clk_pxa25x_lcd , " pxa2xx-fb " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_ffuart , " pxa2xx-uart.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_btuart , " pxa2xx-uart.1 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_stuart , " pxa2xx-uart.2 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_usb , " pxa25x-udc " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_mmc , " pxa2xx-mci.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_i2c , " pxa2xx-i2c.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_ssp , " pxa25x-ssp.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_nssp , " pxa25x-nssp.1 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_assp , " pxa25x-nssp.2 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_pwm0 , " pxa25x-pwm.0 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_pwm1 , " pxa25x-pwm.1 " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_i2s , " pxa2xx-i2s " , NULL ) ,
INIT_CLKREG ( & clk_pxa25x_stuart , " pxa2xx-ir " , " UARTCLK " ) ,
INIT_CLKREG ( & clk_pxa25x_ficp , " pxa2xx-ir " , " FICPCLK " ) ,
INIT_CLKREG ( & clk_pxa25x_ac97 , NULL , " AC97CLK " ) ,
INIT_CLKREG ( & clk_pxa25x_gpio11 , NULL , " GPIO11_CLK " ) ,
INIT_CLKREG ( & clk_pxa25x_gpio12 , NULL , " GPIO12_CLK " ) ,
2007-08-20 10:18:02 +01:00
} ;
2005-06-13 22:35:41 +01:00
# ifdef CONFIG_PM
2005-06-03 20:52:27 +01:00
2007-07-18 11:38:45 +01:00
# define SAVE(x) sleep_save[SLEEP_SAVE_##x] = x
# define RESTORE(x) x = sleep_save[SLEEP_SAVE_##x]
/*
* List of global PXA peripheral registers to preserve .
* More ones like CP and general purpose register values are preserved
* with the stack pointer in sleep . S .
*/
2008-09-03 18:06:34 +08:00
enum {
2007-07-18 11:38:45 +01:00
SLEEP_SAVE_PSTR ,
SLEEP_SAVE_CKEN ,
2008-05-02 21:17:06 +01:00
SLEEP_SAVE_COUNT
2007-07-18 11:38:45 +01:00
} ;
static void pxa25x_cpu_pm_save ( unsigned long * sleep_save )
{
SAVE ( CKEN ) ;
SAVE ( PSTR ) ;
}
static void pxa25x_cpu_pm_restore ( unsigned long * sleep_save )
{
RESTORE ( CKEN ) ;
RESTORE ( PSTR ) ;
}
static void pxa25x_cpu_pm_enter ( suspend_state_t state )
2005-06-03 20:52:27 +01:00
{
2008-05-08 16:50:39 +01:00
/* Clear reset status */
RCSR = RCSR_HWR | RCSR_WDR | RCSR_SMR | RCSR_GPR ;
2005-06-03 20:52:27 +01:00
switch ( state ) {
case PM_SUSPEND_MEM :
2007-07-18 11:40:13 +01:00
pxa25x_cpu_suspend ( PWRMODE_SLEEP ) ;
2005-06-03 20:52:27 +01:00
break ;
}
}
2005-06-13 22:35:41 +01:00
2008-08-27 12:55:04 +01:00
static int pxa25x_cpu_pm_prepare ( void )
{
/* set resume return address */
PSPR = virt_to_phys ( pxa_cpu_resume ) ;
return 0 ;
}
static void pxa25x_cpu_pm_finish ( void )
{
/* ensure not to come back here if it wasn't intended */
PSPR = 0 ;
}
2007-07-18 11:38:45 +01:00
static struct pxa_cpu_pm_fns pxa25x_cpu_pm_fns = {
2008-05-02 21:17:06 +01:00
. save_count = SLEEP_SAVE_COUNT ,
2007-10-18 03:04:40 -07:00
. valid = suspend_valid_only_mem ,
2007-07-18 11:38:45 +01:00
. save = pxa25x_cpu_pm_save ,
. restore = pxa25x_cpu_pm_restore ,
. enter = pxa25x_cpu_pm_enter ,
2008-08-27 12:55:04 +01:00
. prepare = pxa25x_cpu_pm_prepare ,
. finish = pxa25x_cpu_pm_finish ,
2007-05-15 11:16:10 +01:00
} ;
2007-07-18 11:38:45 +01:00
static void __init pxa25x_init_pm ( void )
{
pxa_cpu_pm_fns = & pxa25x_cpu_pm_fns ;
}
2008-01-02 08:24:49 +08:00
# else
static inline void pxa25x_init_pm ( void ) { }
2005-06-13 22:35:41 +01:00
# endif
2007-05-15 11:16:10 +01:00
2007-08-29 10:22:17 +01:00
/* PXA25x: supports wakeup from GPIO0..GPIO15 and RTC alarm
*/
static int pxa25x_set_wake ( unsigned int irq , unsigned int on )
{
int gpio = IRQ_TO_GPIO ( irq ) ;
2008-03-11 09:46:28 +08:00
uint32_t mask = 0 ;
if ( gpio > = 0 & & gpio < 85 )
return gpio_set_wake ( gpio , on ) ;
2007-08-29 10:22:17 +01:00
if ( irq = = IRQ_RTCAlrm ) {
mask = PWER_RTC ;
goto set_pwer ;
}
return - EINVAL ;
set_pwer :
if ( on )
PWER | = mask ;
else
PWER & = ~ mask ;
return 0 ;
}
2007-06-22 04:14:09 +01:00
void __init pxa25x_init_irq ( void )
{
2008-03-04 14:19:58 +08:00
pxa_init_irq ( 32 , pxa25x_set_wake ) ;
2009-01-06 17:37:37 +08:00
pxa_init_gpio ( IRQ_GPIO_2_x , 2 , 84 , pxa25x_set_wake ) ;
2007-06-22 04:14:09 +01:00
}
2008-11-26 18:12:04 +08:00
# ifdef CONFIG_CPU_PXA26x
void __init pxa26x_init_irq ( void )
{
pxa_init_irq ( 32 , pxa25x_set_wake ) ;
2009-01-06 17:37:37 +08:00
pxa_init_gpio ( IRQ_GPIO_2_x , 2 , 89 , pxa25x_set_wake ) ;
2008-11-26 18:12:04 +08:00
}
# endif
2007-05-15 10:39:49 +01:00
static struct platform_device * pxa25x_devices [ ] __initdata = {
2008-06-22 23:36:39 +01:00
& pxa25x_device_udc ,
2007-07-17 10:45:58 +01:00
& pxa_device_i2s ,
2008-11-13 23:50:56 +01:00
& sa1100_device_rtc ,
2007-12-10 17:54:36 +08:00
& pxa25x_device_ssp ,
& pxa25x_device_nssp ,
& pxa25x_device_assp ,
2008-04-13 21:44:04 +01:00
& pxa25x_device_pwm0 ,
& pxa25x_device_pwm1 ,
2007-05-15 10:39:49 +01:00
} ;
2008-01-28 23:00:02 +00:00
static struct sys_device pxa25x_sysdev [ ] = {
{
. cls = & pxa_irq_sysclass ,
2008-09-03 18:06:34 +08:00
} , {
. cls = & pxa2xx_mfp_sysclass ,
2008-01-28 23:00:02 +00:00
} , {
. cls = & pxa_gpio_sysclass ,
2008-01-28 23:00:02 +00:00
} ,
} ;
2007-05-15 11:16:10 +01:00
static int __init pxa25x_init ( void )
{
2008-01-28 23:00:02 +00:00
int i , ret = 0 ;
2007-06-22 05:40:17 +01:00
2008-09-11 10:27:30 +08:00
if ( cpu_is_pxa25x ( ) ) {
2008-07-29 14:26:00 +08:00
reset_status = RCSR ;
2008-11-08 20:25:21 +00:00
clks_register ( pxa25x_clkregs , ARRAY_SIZE ( pxa25x_clkregs ) ) ;
2007-08-20 10:18:02 +01:00
2009-01-02 16:26:33 +08:00
if ( ( ret = pxa_init_dma ( IRQ_DMA , 16 ) ) )
2007-06-22 05:40:17 +01:00
return ret ;
2008-01-02 08:24:49 +08:00
2007-07-18 11:38:45 +01:00
pxa25x_init_pm ( ) ;
2008-01-02 08:24:49 +08:00
2008-01-28 23:00:02 +00:00
for ( i = 0 ; i < ARRAY_SIZE ( pxa25x_sysdev ) ; i + + ) {
ret = sysdev_register ( & pxa25x_sysdev [ i ] ) ;
if ( ret )
pr_err ( " failed to register sysdev[%d] \n " , i ) ;
}
2007-05-15 10:39:49 +01:00
ret = platform_add_devices ( pxa25x_devices ,
ARRAY_SIZE ( pxa25x_devices ) ) ;
2008-01-28 23:00:02 +00:00
if ( ret )
return ret ;
2007-05-15 11:16:10 +01:00
}
2008-01-28 23:00:02 +00:00
2008-09-11 10:25:59 +08:00
/* Only add HWUART for PXA255/26x; PXA210/250 do not have it. */
2009-11-09 13:34:08 +08:00
if ( cpu_is_pxa255 ( ) )
2008-11-08 20:25:21 +00:00
clks_register ( & pxa25x_hwuart_clkreg , 1 ) ;
2007-05-15 10:39:49 +01:00
return ret ;
2007-05-15 11:16:10 +01:00
}
2008-04-19 10:59:24 +01:00
postcore_initcall ( pxa25x_init ) ;