2007-02-11 18:31:01 +01:00
/* linux/arch/arm/plat-s3c24xx/cpu.c
2005-04-16 15:20:36 -07:00
*
* Copyright ( c ) 2004 - 2005 Simtec Electronics
* http : //www.simtec.co.uk/products/SWLINUX/
* Ben Dooks < ben @ simtec . co . uk >
*
2012-05-12 16:22:17 +09:00
* Common code for S3C24XX machines
2005-04-16 15:20:36 -07:00
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/init.h>
# include <linux/module.h>
# include <linux/interrupt.h>
# include <linux/ioport.h>
2006-12-17 23:22:26 +01:00
# include <linux/serial_core.h>
2013-04-12 21:17:22 +02:00
# include <clocksource/samsung_pwm.h>
2005-10-29 19:07:23 +01:00
# include <linux/platform_device.h>
2008-04-16 00:15:20 +01:00
# include <linux/delay.h>
2008-09-06 12:10:45 +01:00
# include <linux/io.h>
2005-04-16 15:20:36 -07:00
2008-08-05 16:14:15 +01:00
# include <mach/hardware.h>
2011-08-03 11:34:59 -04:00
# include <mach/regs-clock.h>
2005-04-16 15:20:36 -07:00
# include <asm/irq.h>
2008-04-16 00:15:20 +01:00
# include <asm/cacheflush.h>
2012-03-28 18:30:01 +01:00
# include <asm/system_info.h>
2012-03-29 23:22:44 -07:00
# include <asm/system_misc.h>
2005-04-16 15:20:36 -07:00
# include <asm/mach/arch.h>
# include <asm/mach/map.h>
2008-08-05 16:14:15 +01:00
# include <mach/regs-gpio.h>
2008-10-07 22:26:09 +01:00
# include <plat/regs-serial.h>
2005-04-16 15:20:36 -07:00
2008-10-07 22:26:09 +01:00
# include <plat/cpu.h>
# include <plat/devs.h>
2008-10-07 23:09:51 +01:00
# include <plat/clock.h>
2012-05-12 16:22:18 +09:00
# include <plat/cpu-freq.h>
# include <plat/pll.h>
2013-04-12 21:17:22 +02:00
# include <plat/pwm-core.h>
2005-04-16 15:20:36 -07:00
2013-02-08 10:31:28 -08:00
# include "common.h"
2005-04-16 15:20:36 -07:00
/* table of supported CPUs */
static const char name_s3c2410 [ ] = " S3C2410 " ;
2006-06-24 21:21:27 +01:00
static const char name_s3c2412 [ ] = " S3C2412 " ;
2010-04-30 16:32:26 +09:00
static const char name_s3c2416 [ ] = " S3C2416/S3C2450 " ;
2005-04-16 15:20:36 -07:00
static const char name_s3c2440 [ ] = " S3C2440 " ;
2006-06-18 23:06:41 +01:00
static const char name_s3c2442 [ ] = " S3C2442 " ;
2009-09-22 21:40:39 +01:00
static const char name_s3c2442b [ ] = " S3C2442B " ;
2007-02-16 12:12:31 +01:00
static const char name_s3c2443 [ ] = " S3C2443 " ;
2005-04-16 15:20:36 -07:00
static const char name_s3c2410a [ ] = " S3C2410A " ;
static const char name_s3c2440a [ ] = " S3C2440A " ;
static struct cpu_table cpu_ids [ ] __initdata = {
{
. idcode = 0x32410000 ,
. idmask = 0xffffffff ,
. map_io = s3c2410_map_io ,
. init_clocks = s3c2410_init_clocks ,
. init_uarts = s3c2410_init_uarts ,
. init = s3c2410_init ,
. name = name_s3c2410
} ,
{
. idcode = 0x32410002 ,
. idmask = 0xffffffff ,
. map_io = s3c2410_map_io ,
. init_clocks = s3c2410_init_clocks ,
. init_uarts = s3c2410_init_uarts ,
2009-07-30 23:23:38 +01:00
. init = s3c2410a_init ,
2005-04-16 15:20:36 -07:00
. name = name_s3c2410a
} ,
{
. idcode = 0x32440000 ,
. idmask = 0xffffffff ,
2010-12-01 08:29:23 +02:00
. map_io = s3c2440_map_io ,
2006-06-18 23:06:41 +01:00
. init_clocks = s3c244x_init_clocks ,
. init_uarts = s3c244x_init_uarts ,
2005-04-16 15:20:36 -07:00
. init = s3c2440_init ,
. name = name_s3c2440
} ,
{
. idcode = 0x32440001 ,
. idmask = 0xffffffff ,
2010-12-01 08:29:23 +02:00
. map_io = s3c2440_map_io ,
2006-06-18 23:06:41 +01:00
. init_clocks = s3c244x_init_clocks ,
. init_uarts = s3c244x_init_uarts ,
2005-04-16 15:20:36 -07:00
. init = s3c2440_init ,
. name = name_s3c2440a
2006-02-01 21:24:24 +00:00
} ,
2006-06-18 23:06:41 +01:00
{
. idcode = 0x32440aaa ,
. idmask = 0xffffffff ,
2010-12-01 08:29:23 +02:00
. map_io = s3c2442_map_io ,
2006-06-18 23:06:41 +01:00
. init_clocks = s3c244x_init_clocks ,
. init_uarts = s3c244x_init_uarts ,
. init = s3c2442_init ,
. name = name_s3c2442
} ,
2009-09-22 21:40:39 +01:00
{
. idcode = 0x32440aab ,
. idmask = 0xffffffff ,
2010-12-01 08:29:23 +02:00
. map_io = s3c2442_map_io ,
2009-09-22 21:40:39 +01:00
. init_clocks = s3c244x_init_clocks ,
. init_uarts = s3c244x_init_uarts ,
. init = s3c2442_init ,
. name = name_s3c2442b
} ,
2006-06-24 21:21:27 +01:00
{
. idcode = 0x32412001 ,
. idmask = 0xffffffff ,
. map_io = s3c2412_map_io ,
. init_clocks = s3c2412_init_clocks ,
. init_uarts = s3c2412_init_uarts ,
. init = s3c2412_init ,
. name = name_s3c2412 ,
} ,
2006-09-20 20:39:15 +01:00
{ /* a newer version of the s3c2412 */
. idcode = 0x32412003 ,
. idmask = 0xffffffff ,
. map_io = s3c2412_map_io ,
. init_clocks = s3c2412_init_clocks ,
. init_uarts = s3c2412_init_uarts ,
. init = s3c2412_init ,
. name = name_s3c2412 ,
} ,
2010-04-28 18:09:01 +09:00
{ /* a strange version of the s3c2416 */
. idcode = 0x32450003 ,
. idmask = 0xffffffff ,
. map_io = s3c2416_map_io ,
. init_clocks = s3c2416_init_clocks ,
. init_uarts = s3c2416_init_uarts ,
. init = s3c2416_init ,
. name = name_s3c2416 ,
} ,
2007-02-16 12:12:31 +01:00
{
. idcode = 0x32443001 ,
. idmask = 0xffffffff ,
. map_io = s3c2443_map_io ,
. init_clocks = s3c2443_init_clocks ,
. init_uarts = s3c2443_init_uarts ,
. init = s3c2443_init ,
. name = name_s3c2443 ,
} ,
2005-04-16 15:20:36 -07:00
} ;
/* minimal IO mapping */
static struct map_desc s3c_iodesc [ ] __initdata = {
IODESC_ENT ( GPIO ) ,
IODESC_ENT ( IRQ ) ,
IODESC_ENT ( MEMCTRL ) ,
IODESC_ENT ( UART )
} ;
2008-10-21 14:06:31 +01:00
/* read cpu identificaiton code */
2005-04-16 15:20:36 -07:00
2006-06-24 21:21:27 +01:00
static unsigned long s3c24xx_read_idcode_v5 ( void )
{
2010-04-28 18:00:07 +09:00
# if defined(CONFIG_CPU_S3C2416)
/* s3c2416 is v5, with S3C24XX_GSTATUS1 instead of S3C2412_GSTATUS1 */
u32 gs = __raw_readl ( S3C24XX_GSTATUS1 ) ;
/* test for s3c2416 or similar device */
if ( ( gs > > 16 ) = = 0x3245 )
return gs ;
# endif
2006-06-24 21:21:27 +01:00
# if defined(CONFIG_CPU_S3C2412) || defined(CONFIG_CPU_S3C2413)
return __raw_readl ( S3C2412_GSTATUS1 ) ;
# else
return 1UL ; /* don't look like an 2400 */
# endif
}
static unsigned long s3c24xx_read_idcode_v4 ( void )
{
return __raw_readl ( S3C2410_GSTATUS1 ) ;
}
2011-08-03 11:34:59 -04:00
static void s3c24xx_default_idle ( void )
{
2013-01-18 08:58:23 -08:00
unsigned long tmp = 0 ;
2011-08-03 11:34:59 -04:00
int i ;
/* idle the system by using the idle mode which will wait for an
* interrupt to happen before restarting the system .
*/
/* Warning: going into idle state upsets jtag scanning */
__raw_writel ( __raw_readl ( S3C2410_CLKCON ) | S3C2410_CLKCON_IDLE ,
S3C2410_CLKCON ) ;
/* the samsung port seems to do a loop and then unset idle.. */
for ( i = 0 ; i < 50 ; i + + )
tmp + = __raw_readl ( S3C2410_CLKCON ) ; /* ensure loop not optimised out */
/* this bit is not cleared on re-start... */
__raw_writel ( __raw_readl ( S3C2410_CLKCON ) & ~ S3C2410_CLKCON_IDLE ,
S3C2410_CLKCON ) ;
}
2013-04-12 21:17:22 +02:00
static struct samsung_pwm_variant s3c24xx_pwm_variant = {
. bits = 16 ,
. div_base = 1 ,
. has_tint_cstat = false ,
. tclk_mask = ( 1 < < 4 ) ,
} ;
2005-04-16 15:20:36 -07:00
void __init s3c24xx_init_io ( struct map_desc * mach_desc , int size )
{
2011-08-03 11:34:59 -04:00
arm_pm_idle = s3c24xx_default_idle ;
2005-04-16 15:20:36 -07:00
/* initialise the io descriptors we need for initialisation */
2008-10-21 14:06:31 +01:00
iotable_init ( mach_desc , size ) ;
2005-04-16 15:20:36 -07:00
iotable_init ( s3c_iodesc , ARRAY_SIZE ( s3c_iodesc ) ) ;
2006-06-24 21:21:27 +01:00
if ( cpu_architecture ( ) > = CPU_ARCH_ARMv5 ) {
2011-08-20 02:18:18 +09:00
samsung_cpu_id = s3c24xx_read_idcode_v5 ( ) ;
2006-06-24 21:21:27 +01:00
} else {
2011-08-20 02:18:18 +09:00
samsung_cpu_id = s3c24xx_read_idcode_v4 ( ) ;
2006-06-24 21:21:27 +01:00
}
2011-08-20 12:18:07 +09:00
s3c24xx_init_cpu ( ) ;
2006-02-01 21:24:24 +00:00
2011-08-20 02:18:18 +09:00
s3c_init_cpu ( samsung_cpu_id , cpu_ids , ARRAY_SIZE ( cpu_ids ) ) ;
2013-04-12 21:17:22 +02:00
samsung_pwm_set_platdata ( & s3c24xx_pwm_variant ) ;
2006-06-18 23:04:05 +01:00
}
2012-05-12 16:22:17 +09:00
2013-04-28 02:25:01 +02:00
void __init samsung_set_timer_source ( unsigned int event , unsigned int source )
{
s3c24xx_pwm_variant . output_mask = BIT ( SAMSUNG_PWM_NUM ) - 1 ;
s3c24xx_pwm_variant . output_mask & = ~ ( BIT ( event ) | BIT ( source ) ) ;
}
void __init samsung_timer_init ( void )
{
unsigned int timer_irqs [ SAMSUNG_PWM_NUM ] = {
IRQ_TIMER0 , IRQ_TIMER1 , IRQ_TIMER2 , IRQ_TIMER3 , IRQ_TIMER4 ,
} ;
samsung_pwm_clocksource_init ( S3C_VA_TIMER ,
timer_irqs , & s3c24xx_pwm_variant ) ;
}
2012-05-12 16:22:17 +09:00
/* Serial port registrations */
2013-04-11 02:04:48 +02:00
# define S3C2410_PA_UART0 (S3C24XX_PA_UART)
# define S3C2410_PA_UART1 (S3C24XX_PA_UART + 0x4000 )
# define S3C2410_PA_UART2 (S3C24XX_PA_UART + 0x8000 )
# define S3C2443_PA_UART3 (S3C24XX_PA_UART + 0xC000 )
2012-05-12 16:22:17 +09:00
static struct resource s3c2410_uart0_resource [ ] = {
2012-05-12 16:24:59 +09:00
[ 0 ] = DEFINE_RES_MEM ( S3C2410_PA_UART0 , SZ_16K ) ,
[ 1 ] = DEFINE_RES_NAMED ( IRQ_S3CUART_RX0 , \
IRQ_S3CUART_ERR0 - IRQ_S3CUART_RX0 + 1 , \
NULL , IORESOURCE_IRQ )
2012-05-12 16:22:17 +09:00
} ;
static struct resource s3c2410_uart1_resource [ ] = {
2012-05-12 16:24:59 +09:00
[ 0 ] = DEFINE_RES_MEM ( S3C2410_PA_UART1 , SZ_16K ) ,
[ 1 ] = DEFINE_RES_NAMED ( IRQ_S3CUART_RX1 , \
IRQ_S3CUART_ERR1 - IRQ_S3CUART_RX1 + 1 , \
NULL , IORESOURCE_IRQ )
2012-05-12 16:22:17 +09:00
} ;
static struct resource s3c2410_uart2_resource [ ] = {
2012-05-12 16:24:59 +09:00
[ 0 ] = DEFINE_RES_MEM ( S3C2410_PA_UART2 , SZ_16K ) ,
[ 1 ] = DEFINE_RES_NAMED ( IRQ_S3CUART_RX2 , \
IRQ_S3CUART_ERR2 - IRQ_S3CUART_RX2 + 1 , \
NULL , IORESOURCE_IRQ )
2012-05-12 16:22:17 +09:00
} ;
static struct resource s3c2410_uart3_resource [ ] = {
2012-05-12 16:24:59 +09:00
[ 0 ] = DEFINE_RES_MEM ( S3C2443_PA_UART3 , SZ_16K ) ,
[ 1 ] = DEFINE_RES_NAMED ( IRQ_S3CUART_RX3 , \
IRQ_S3CUART_ERR3 - IRQ_S3CUART_RX3 + 1 , \
NULL , IORESOURCE_IRQ )
2012-05-12 16:22:17 +09:00
} ;
struct s3c24xx_uart_resources s3c2410_uart_resources [ ] __initdata = {
[ 0 ] = {
. resources = s3c2410_uart0_resource ,
. nr_resources = ARRAY_SIZE ( s3c2410_uart0_resource ) ,
} ,
[ 1 ] = {
. resources = s3c2410_uart1_resource ,
. nr_resources = ARRAY_SIZE ( s3c2410_uart1_resource ) ,
} ,
[ 2 ] = {
. resources = s3c2410_uart2_resource ,
. nr_resources = ARRAY_SIZE ( s3c2410_uart2_resource ) ,
} ,
[ 3 ] = {
. resources = s3c2410_uart3_resource ,
. nr_resources = ARRAY_SIZE ( s3c2410_uart3_resource ) ,
} ,
} ;
2012-05-12 16:22:18 +09:00
/* initialise all the clocks */
void __init_or_cpufreq s3c24xx_setup_clocks ( unsigned long fclk ,
unsigned long hclk ,
unsigned long pclk )
{
clk_upll . rate = s3c24xx_get_pll ( __raw_readl ( S3C2410_UPLLCON ) ,
clk_xtal . rate ) ;
clk_mpll . rate = fclk ;
clk_h . rate = hclk ;
clk_p . rate = pclk ;
clk_f . rate = fclk ;
}