xref: /openbmc/linux/arch/arm/mach-ep93xx/clock.c (revision 78c99ba1)
1 /*
2  * arch/arm/mach-ep93xx/clock.c
3  * Clock control for Cirrus EP93xx chips.
4  *
5  * Copyright (C) 2006 Lennert Buytenhek <buytenh@wantstofly.org>
6  *
7  * This program is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 2 of the License, or (at
10  * your option) any later version.
11  */
12 
13 #include <linux/kernel.h>
14 #include <linux/clk.h>
15 #include <linux/err.h>
16 #include <linux/module.h>
17 #include <linux/string.h>
18 #include <linux/io.h>
19 
20 #include <asm/clkdev.h>
21 #include <asm/div64.h>
22 #include <mach/hardware.h>
23 
24 
25 /*
26  * The EP93xx has two external crystal oscillators.  To generate the
27  * required high-frequency clocks, the processor uses two phase-locked-
28  * loops (PLLs) to multiply the incoming external clock signal to much
29  * higher frequencies that are then divided down by programmable dividers
30  * to produce the needed clocks.  The PLLs operate independently of one
31  * another.
32  */
33 #define EP93XX_EXT_CLK_RATE	14745600
34 #define EP93XX_EXT_RTC_RATE	32768
35 
36 
37 struct clk {
38 	unsigned long	rate;
39 	int		users;
40 	int		sw_locked;
41 	u32		enable_reg;
42 	u32		enable_mask;
43 
44 	unsigned long	(*get_rate)(struct clk *clk);
45 };
46 
47 
48 static unsigned long get_uart_rate(struct clk *clk);
49 
50 
51 static struct clk clk_uart1 = {
52 	.sw_locked	= 1,
53 	.enable_reg	= EP93XX_SYSCON_DEVICE_CONFIG,
54 	.enable_mask	= EP93XX_SYSCON_DEVICE_CONFIG_U1EN,
55 	.get_rate	= get_uart_rate,
56 };
57 static struct clk clk_uart2 = {
58 	.sw_locked	= 1,
59 	.enable_reg	= EP93XX_SYSCON_DEVICE_CONFIG,
60 	.enable_mask	= EP93XX_SYSCON_DEVICE_CONFIG_U2EN,
61 	.get_rate	= get_uart_rate,
62 };
63 static struct clk clk_uart3 = {
64 	.sw_locked	= 1,
65 	.enable_reg	= EP93XX_SYSCON_DEVICE_CONFIG,
66 	.enable_mask	= EP93XX_SYSCON_DEVICE_CONFIG_U3EN,
67 	.get_rate	= get_uart_rate,
68 };
69 static struct clk clk_pll1;
70 static struct clk clk_f;
71 static struct clk clk_h;
72 static struct clk clk_p;
73 static struct clk clk_pll2;
74 static struct clk clk_usb_host = {
75 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
76 	.enable_mask	= EP93XX_SYSCON_CLOCK_USH_EN,
77 };
78 
79 /* DMA Clocks */
80 static struct clk clk_m2p0 = {
81 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
82 	.enable_mask	= 0x00020000,
83 };
84 static struct clk clk_m2p1 = {
85 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
86 	.enable_mask	= 0x00010000,
87 };
88 static struct clk clk_m2p2 = {
89 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
90 	.enable_mask	= 0x00080000,
91 };
92 static struct clk clk_m2p3 = {
93 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
94 	.enable_mask	= 0x00040000,
95 };
96 static struct clk clk_m2p4 = {
97 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
98 	.enable_mask	= 0x00200000,
99 };
100 static struct clk clk_m2p5 = {
101 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
102 	.enable_mask	= 0x00100000,
103 };
104 static struct clk clk_m2p6 = {
105 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
106 	.enable_mask	= 0x00800000,
107 };
108 static struct clk clk_m2p7 = {
109 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
110 	.enable_mask	= 0x00400000,
111 };
112 static struct clk clk_m2p8 = {
113 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
114 	.enable_mask	= 0x02000000,
115 };
116 static struct clk clk_m2p9 = {
117 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
118 	.enable_mask	= 0x01000000,
119 };
120 static struct clk clk_m2m0 = {
121 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
122 	.enable_mask	= 0x04000000,
123 };
124 static struct clk clk_m2m1 = {
125 	.enable_reg	= EP93XX_SYSCON_CLOCK_CONTROL,
126 	.enable_mask	= 0x08000000,
127 };
128 
129 #define INIT_CK(dev,con,ck)					\
130 	{ .dev_id = dev, .con_id = con, .clk = ck }
131 
132 static struct clk_lookup clocks[] = {
133 	INIT_CK("apb:uart1", NULL, &clk_uart1),
134 	INIT_CK("apb:uart2", NULL, &clk_uart2),
135 	INIT_CK("apb:uart3", NULL, &clk_uart3),
136 	INIT_CK(NULL, "pll1", &clk_pll1),
137 	INIT_CK(NULL, "fclk", &clk_f),
138 	INIT_CK(NULL, "hclk", &clk_h),
139 	INIT_CK(NULL, "pclk", &clk_p),
140 	INIT_CK(NULL, "pll2", &clk_pll2),
141 	INIT_CK(NULL, "usb_host", &clk_usb_host),
142 	INIT_CK(NULL, "m2p0", &clk_m2p0),
143 	INIT_CK(NULL, "m2p1", &clk_m2p1),
144 	INIT_CK(NULL, "m2p2", &clk_m2p2),
145 	INIT_CK(NULL, "m2p3", &clk_m2p3),
146 	INIT_CK(NULL, "m2p4", &clk_m2p4),
147 	INIT_CK(NULL, "m2p5", &clk_m2p5),
148 	INIT_CK(NULL, "m2p6", &clk_m2p6),
149 	INIT_CK(NULL, "m2p7", &clk_m2p7),
150 	INIT_CK(NULL, "m2p8", &clk_m2p8),
151 	INIT_CK(NULL, "m2p9", &clk_m2p9),
152 	INIT_CK(NULL, "m2m0", &clk_m2m0),
153 	INIT_CK(NULL, "m2m1", &clk_m2m1),
154 };
155 
156 
157 int clk_enable(struct clk *clk)
158 {
159 	if (!clk->users++ && clk->enable_reg) {
160 		u32 value;
161 
162 		value = __raw_readl(clk->enable_reg);
163 		if (clk->sw_locked)
164 			__raw_writel(0xaa, EP93XX_SYSCON_SWLOCK);
165 		__raw_writel(value | clk->enable_mask, clk->enable_reg);
166 	}
167 
168 	return 0;
169 }
170 EXPORT_SYMBOL(clk_enable);
171 
172 void clk_disable(struct clk *clk)
173 {
174 	if (!--clk->users && clk->enable_reg) {
175 		u32 value;
176 
177 		value = __raw_readl(clk->enable_reg);
178 		if (clk->sw_locked)
179 			__raw_writel(0xaa, EP93XX_SYSCON_SWLOCK);
180 		__raw_writel(value & ~clk->enable_mask, clk->enable_reg);
181 	}
182 }
183 EXPORT_SYMBOL(clk_disable);
184 
185 static unsigned long get_uart_rate(struct clk *clk)
186 {
187 	u32 value;
188 
189 	value = __raw_readl(EP93XX_SYSCON_CLOCK_CONTROL);
190 	if (value & EP93XX_SYSCON_CLOCK_UARTBAUD)
191 		return EP93XX_EXT_CLK_RATE;
192 	else
193 		return EP93XX_EXT_CLK_RATE / 2;
194 }
195 
196 unsigned long clk_get_rate(struct clk *clk)
197 {
198 	if (clk->get_rate)
199 		return clk->get_rate(clk);
200 
201 	return clk->rate;
202 }
203 EXPORT_SYMBOL(clk_get_rate);
204 
205 
206 static char fclk_divisors[] = { 1, 2, 4, 8, 16, 1, 1, 1 };
207 static char hclk_divisors[] = { 1, 2, 4, 5, 6, 8, 16, 32 };
208 static char pclk_divisors[] = { 1, 2, 4, 8 };
209 
210 /*
211  * PLL rate = 14.7456 MHz * (X1FBD + 1) * (X2FBD + 1) / (X2IPD + 1) / 2^PS
212  */
213 static unsigned long calc_pll_rate(u32 config_word)
214 {
215 	unsigned long long rate;
216 	int i;
217 
218 	rate = EP93XX_EXT_CLK_RATE;
219 	rate *= ((config_word >> 11) & 0x1f) + 1;		/* X1FBD */
220 	rate *= ((config_word >> 5) & 0x3f) + 1;		/* X2FBD */
221 	do_div(rate, (config_word & 0x1f) + 1);			/* X2IPD */
222 	for (i = 0; i < ((config_word >> 16) & 3); i++)		/* PS */
223 		rate >>= 1;
224 
225 	return (unsigned long)rate;
226 }
227 
228 static void __init ep93xx_dma_clock_init(void)
229 {
230 	clk_m2p0.rate = clk_h.rate;
231 	clk_m2p1.rate = clk_h.rate;
232 	clk_m2p2.rate = clk_h.rate;
233 	clk_m2p3.rate = clk_h.rate;
234 	clk_m2p4.rate = clk_h.rate;
235 	clk_m2p5.rate = clk_h.rate;
236 	clk_m2p6.rate = clk_h.rate;
237 	clk_m2p7.rate = clk_h.rate;
238 	clk_m2p8.rate = clk_h.rate;
239 	clk_m2p9.rate = clk_h.rate;
240 	clk_m2m0.rate = clk_h.rate;
241 	clk_m2m1.rate = clk_h.rate;
242 }
243 
244 static int __init ep93xx_clock_init(void)
245 {
246 	u32 value;
247 	int i;
248 
249 	value = __raw_readl(EP93XX_SYSCON_CLOCK_SET1);
250 	if (!(value & 0x00800000)) {			/* PLL1 bypassed?  */
251 		clk_pll1.rate = EP93XX_EXT_CLK_RATE;
252 	} else {
253 		clk_pll1.rate = calc_pll_rate(value);
254 	}
255 	clk_f.rate = clk_pll1.rate / fclk_divisors[(value >> 25) & 0x7];
256 	clk_h.rate = clk_pll1.rate / hclk_divisors[(value >> 20) & 0x7];
257 	clk_p.rate = clk_h.rate / pclk_divisors[(value >> 18) & 0x3];
258 	ep93xx_dma_clock_init();
259 
260 	value = __raw_readl(EP93XX_SYSCON_CLOCK_SET2);
261 	if (!(value & 0x00080000)) {			/* PLL2 bypassed?  */
262 		clk_pll2.rate = EP93XX_EXT_CLK_RATE;
263 	} else if (value & 0x00040000) {		/* PLL2 enabled?  */
264 		clk_pll2.rate = calc_pll_rate(value);
265 	} else {
266 		clk_pll2.rate = 0;
267 	}
268 	clk_usb_host.rate = clk_pll2.rate / (((value >> 28) & 0xf) + 1);
269 
270 	printk(KERN_INFO "ep93xx: PLL1 running at %ld MHz, PLL2 at %ld MHz\n",
271 		clk_pll1.rate / 1000000, clk_pll2.rate / 1000000);
272 	printk(KERN_INFO "ep93xx: FCLK %ld MHz, HCLK %ld MHz, PCLK %ld MHz\n",
273 		clk_f.rate / 1000000, clk_h.rate / 1000000,
274 		clk_p.rate / 1000000);
275 
276 	for (i = 0; i < ARRAY_SIZE(clocks); i++)
277 		clkdev_add(&clocks[i]);
278 	return 0;
279 }
280 arch_initcall(ep93xx_clock_init);
281