diff options
-rw-r--r-- | arch/arm/mach-s3c6400/s3c6400.c | 3 | ||||
-rw-r--r-- | arch/arm/mach-s3c6410/cpu.c | 3 | ||||
-rw-r--r-- | arch/arm/plat-s3c64xx/include/plat/s3c6400.h | 2 | ||||
-rw-r--r-- | arch/arm/plat-s3c64xx/s3c6400-clock.c | 76 |
4 files changed, 80 insertions, 4 deletions
diff --git a/arch/arm/mach-s3c6400/s3c6400.c b/arch/arm/mach-s3c6400/s3c6400.c index bd17f3db2c21..1ece887d90bb 100644 --- a/arch/arm/mach-s3c6400/s3c6400.c +++ b/arch/arm/mach-s3c6400/s3c6400.c | |||
@@ -30,6 +30,7 @@ | |||
30 | 30 | ||
31 | #include <plat/cpu-freq.h> | 31 | #include <plat/cpu-freq.h> |
32 | #include <plat/regs-serial.h> | 32 | #include <plat/regs-serial.h> |
33 | #include <plat/regs-clock.h> | ||
33 | 34 | ||
34 | #include <plat/cpu.h> | 35 | #include <plat/cpu.h> |
35 | #include <plat/devs.h> | 36 | #include <plat/devs.h> |
@@ -54,7 +55,7 @@ void __init s3c6400_init_clocks(int xtal) | |||
54 | printk(KERN_DEBUG "%s: initialising clocks\n", __func__); | 55 | printk(KERN_DEBUG "%s: initialising clocks\n", __func__); |
55 | s3c24xx_register_baseclocks(xtal); | 56 | s3c24xx_register_baseclocks(xtal); |
56 | s3c64xx_register_clocks(); | 57 | s3c64xx_register_clocks(); |
57 | s3c6400_register_clocks(); | 58 | s3c6400_register_clocks(S3C6400_CLKDIV0_ARM_MASK); |
58 | s3c6400_setup_clocks(); | 59 | s3c6400_setup_clocks(); |
59 | } | 60 | } |
60 | 61 | ||
diff --git a/arch/arm/mach-s3c6410/cpu.c b/arch/arm/mach-s3c6410/cpu.c index 6a73ca6b7a3a..ade904de8895 100644 --- a/arch/arm/mach-s3c6410/cpu.c +++ b/arch/arm/mach-s3c6410/cpu.c | |||
@@ -31,6 +31,7 @@ | |||
31 | 31 | ||
32 | #include <plat/cpu-freq.h> | 32 | #include <plat/cpu-freq.h> |
33 | #include <plat/regs-serial.h> | 33 | #include <plat/regs-serial.h> |
34 | #include <plat/regs-clock.h> | ||
34 | 35 | ||
35 | #include <plat/cpu.h> | 36 | #include <plat/cpu.h> |
36 | #include <plat/devs.h> | 37 | #include <plat/devs.h> |
@@ -68,7 +69,7 @@ void __init s3c6410_init_clocks(int xtal) | |||
68 | printk(KERN_DEBUG "%s: initialising clocks\n", __func__); | 69 | printk(KERN_DEBUG "%s: initialising clocks\n", __func__); |
69 | s3c24xx_register_baseclocks(xtal); | 70 | s3c24xx_register_baseclocks(xtal); |
70 | s3c64xx_register_clocks(); | 71 | s3c64xx_register_clocks(); |
71 | s3c6400_register_clocks(); | 72 | s3c6400_register_clocks(S3C6410_CLKDIV0_ARM_MASK); |
72 | s3c6400_setup_clocks(); | 73 | s3c6400_setup_clocks(); |
73 | } | 74 | } |
74 | 75 | ||
diff --git a/arch/arm/plat-s3c64xx/include/plat/s3c6400.h b/arch/arm/plat-s3c64xx/include/plat/s3c6400.h index e12296666d46..11f2e1e119b0 100644 --- a/arch/arm/plat-s3c64xx/include/plat/s3c6400.h +++ b/arch/arm/plat-s3c64xx/include/plat/s3c6400.h | |||
@@ -15,7 +15,7 @@ | |||
15 | /* Common init code for S3C6400 related SoCs */ | 15 | /* Common init code for S3C6400 related SoCs */ |
16 | 16 | ||
17 | extern void s3c6400_common_init_uarts(struct s3c2410_uartcfg *cfg, int no); | 17 | extern void s3c6400_common_init_uarts(struct s3c2410_uartcfg *cfg, int no); |
18 | extern void s3c6400_register_clocks(void); | 18 | extern void s3c6400_register_clocks(unsigned armclk_divlimit); |
19 | extern void s3c6400_setup_clocks(void); | 19 | extern void s3c6400_setup_clocks(void); |
20 | 20 | ||
21 | #ifdef CONFIG_CPU_S3C6400 | 21 | #ifdef CONFIG_CPU_S3C6400 |
diff --git a/arch/arm/plat-s3c64xx/s3c6400-clock.c b/arch/arm/plat-s3c64xx/s3c6400-clock.c index 96fa9ea4d5bc..1debc1f9f987 100644 --- a/arch/arm/plat-s3c64xx/s3c6400-clock.c +++ b/arch/arm/plat-s3c64xx/s3c6400-clock.c | |||
@@ -133,6 +133,65 @@ static struct clksrc_clk clk_mout_mpll = { | |||
133 | .sources = &clk_src_mpll, | 133 | .sources = &clk_src_mpll, |
134 | }; | 134 | }; |
135 | 135 | ||
136 | static unsigned int armclk_mask; | ||
137 | |||
138 | static unsigned long s3c64xx_clk_arm_get_rate(struct clk *clk) | ||
139 | { | ||
140 | unsigned long rate = clk_get_rate(clk->parent); | ||
141 | u32 clkdiv; | ||
142 | |||
143 | /* divisor mask starts at bit0, so no need to shift */ | ||
144 | clkdiv = __raw_readl(S3C_CLK_DIV0) & armclk_mask; | ||
145 | |||
146 | return rate / (clkdiv + 1); | ||
147 | } | ||
148 | |||
149 | static unsigned long s3c64xx_clk_arm_round_rate(struct clk *clk, | ||
150 | unsigned long rate) | ||
151 | { | ||
152 | unsigned long parent = clk_get_rate(clk->parent); | ||
153 | u32 div; | ||
154 | |||
155 | if (parent < rate) | ||
156 | return rate; | ||
157 | |||
158 | div = (parent / rate) - 1; | ||
159 | if (div > armclk_mask) | ||
160 | div = armclk_mask; | ||
161 | |||
162 | return parent / (div + 1); | ||
163 | } | ||
164 | |||
165 | static int s3c64xx_clk_arm_set_rate(struct clk *clk, unsigned long rate) | ||
166 | { | ||
167 | unsigned long parent = clk_get_rate(clk->parent); | ||
168 | u32 div; | ||
169 | u32 val; | ||
170 | |||
171 | if (rate < parent / (armclk_mask + 1)) | ||
172 | return -EINVAL; | ||
173 | |||
174 | rate = clk_round_rate(clk, rate); | ||
175 | div = clk_get_rate(clk->parent) / rate; | ||
176 | |||
177 | val = __raw_readl(S3C_CLK_DIV0); | ||
178 | val &= armclk_mask; | ||
179 | val |= (div - 1); | ||
180 | __raw_writel(val, S3C_CLK_DIV0); | ||
181 | |||
182 | return 0; | ||
183 | |||
184 | } | ||
185 | |||
186 | static struct clk clk_arm = { | ||
187 | .name = "armclk", | ||
188 | .id = -1, | ||
189 | .parent = &clk_mout_apll.clk, | ||
190 | .get_rate = s3c64xx_clk_arm_get_rate, | ||
191 | .set_rate = s3c64xx_clk_arm_set_rate, | ||
192 | .round_rate = s3c64xx_clk_arm_round_rate, | ||
193 | }; | ||
194 | |||
136 | static unsigned long s3c64xx_clk_doutmpll_get_rate(struct clk *clk) | 195 | static unsigned long s3c64xx_clk_doutmpll_get_rate(struct clk *clk) |
137 | { | 196 | { |
138 | unsigned long rate = clk_get_rate(clk->parent); | 197 | unsigned long rate = clk_get_rate(clk->parent); |
@@ -665,14 +724,29 @@ static struct clk *clks[] __initdata = { | |||
665 | &clk_audio1.clk, | 724 | &clk_audio1.clk, |
666 | &clk_irda.clk, | 725 | &clk_irda.clk, |
667 | &clk_camif.clk, | 726 | &clk_camif.clk, |
727 | &clk_arm, | ||
668 | }; | 728 | }; |
669 | 729 | ||
670 | void __init s3c6400_register_clocks(void) | 730 | /** |
731 | * s3c6400_register_clocks - register clocks for s3c6400 and above | ||
732 | * @armclk_divlimit: Divisor mask for ARMCLK | ||
733 | * | ||
734 | * Register the clocks for the S3C6400 and above SoC range, such | ||
735 | * as ARMCLK and the clocks which have divider chains attached. | ||
736 | * | ||
737 | * This call does not setup the clocks, which is left to the | ||
738 | * s3c6400_setup_clocks() call which may be needed by the cpufreq | ||
739 | * or resume code to re-set the clocks if the bootloader has changed | ||
740 | * them. | ||
741 | */ | ||
742 | void __init s3c6400_register_clocks(unsigned armclk_divlimit) | ||
671 | { | 743 | { |
672 | struct clk *clkp; | 744 | struct clk *clkp; |
673 | int ret; | 745 | int ret; |
674 | int ptr; | 746 | int ptr; |
675 | 747 | ||
748 | armclk_mask = armclk_divlimit; | ||
749 | |||
676 | for (ptr = 0; ptr < ARRAY_SIZE(clks); ptr++) { | 750 | for (ptr = 0; ptr < ARRAY_SIZE(clks); ptr++) { |
677 | clkp = clks[ptr]; | 751 | clkp = clks[ptr]; |
678 | ret = s3c24xx_register_clock(clkp); | 752 | ret = s3c24xx_register_clock(clkp); |