blob: 1322851d1acb031bd441e2f4a433de6d68dad8db [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/* linux/arch/arm/mach-s3c2410/clock.c
2 *
Ben Dooksa21765a2007-02-11 18:31:01 +01003 * Copyright (c) 2006 Simtec Electronics
Linus Torvalds1da177e2005-04-16 15:20:36 -07004 * Ben Dooks <ben@simtec.co.uk>
5 *
Ben Dooksa21765a2007-02-11 18:31:01 +01006 * S3C2410,S3C2440,S3C2442 Clock control support
Linus Torvalds1da177e2005-04-16 15:20:36 -07007 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
21*/
22
23#include <linux/init.h>
24#include <linux/module.h>
25#include <linux/kernel.h>
26#include <linux/list.h>
27#include <linux/errno.h>
28#include <linux/err.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070029#include <linux/sysdev.h>
Russell Kingf8ce2542006-01-07 16:15:52 +000030#include <linux/clk.h>
Arjan van de Ven00431702006-01-12 18:42:23 +000031#include <linux/mutex.h>
Ben Dooks8e40a2f2006-03-20 17:10:04 +000032#include <linux/delay.h>
Ben Dooksa21765a2007-02-11 18:31:01 +010033#include <linux/serial_core.h>
34
35#include <asm/mach/map.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070036
Russell Kinga09e64f2008-08-05 16:14:15 +010037#include <mach/hardware.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070038#include <asm/io.h>
39
Ben Dooks531b6172007-07-22 16:05:25 +010040#include <asm/plat-s3c/regs-serial.h>
Russell Kinga09e64f2008-08-05 16:14:15 +010041#include <mach/regs-clock.h>
42#include <mach/regs-gpio.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070043
Ben Dooksa21765a2007-02-11 18:31:01 +010044#include <asm/plat-s3c24xx/s3c2410.h>
45#include <asm/plat-s3c24xx/clock.h>
46#include <asm/plat-s3c24xx/cpu.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070047
Ben Dooksa21765a2007-02-11 18:31:01 +010048int s3c2410_clkcon_enable(struct clk *clk, int enable)
Linus Torvalds1da177e2005-04-16 15:20:36 -070049{
Ben Dooksa21765a2007-02-11 18:31:01 +010050 unsigned int clocks = clk->ctrlbit;
51 unsigned long clkcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -070052
Ben Dooksa21765a2007-02-11 18:31:01 +010053 clkcon = __raw_readl(S3C2410_CLKCON);
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000054
55 if (enable)
Ben Dooksa21765a2007-02-11 18:31:01 +010056 clkcon |= clocks;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000057 else
Ben Dooksa21765a2007-02-11 18:31:01 +010058 clkcon &= ~clocks;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000059
Ben Dooksa21765a2007-02-11 18:31:01 +010060 /* ensure none of the special function bits set */
61 clkcon &= ~(S3C2410_CLKCON_IDLE|S3C2410_CLKCON_POWER);
62
63 __raw_writel(clkcon, S3C2410_CLKCON);
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000064
65 return 0;
66}
67
Ben Dooksa21765a2007-02-11 18:31:01 +010068static int s3c2410_upll_enable(struct clk *clk, int enable)
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000069{
Ben Dooksa21765a2007-02-11 18:31:01 +010070 unsigned long clkslow = __raw_readl(S3C2410_CLKSLOW);
71 unsigned long orig = clkslow;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000072
Ben Dooksa21765a2007-02-11 18:31:01 +010073 if (enable)
74 clkslow &= ~S3C2410_CLKSLOW_UCLK_OFF;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000075 else
Ben Dooksa21765a2007-02-11 18:31:01 +010076 clkslow |= S3C2410_CLKSLOW_UCLK_OFF;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000077
Ben Dooksa21765a2007-02-11 18:31:01 +010078 __raw_writel(clkslow, S3C2410_CLKSLOW);
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000079
Ben Dooksa21765a2007-02-11 18:31:01 +010080 /* if we started the UPLL, then allow to settle */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000081
Ben Dooksa21765a2007-02-11 18:31:01 +010082 if (enable && (orig & S3C2410_CLKSLOW_UCLK_OFF))
83 udelay(200);
84
85 return 0;
86}
87
88/* standard clock definitions */
89
90static struct clk init_clocks_disable[] = {
91 {
92 .name = "nand",
93 .id = -1,
94 .parent = &clk_h,
95 .enable = s3c2410_clkcon_enable,
96 .ctrlbit = S3C2410_CLKCON_NAND,
97 }, {
98 .name = "sdi",
99 .id = -1,
100 .parent = &clk_p,
101 .enable = s3c2410_clkcon_enable,
102 .ctrlbit = S3C2410_CLKCON_SDI,
103 }, {
104 .name = "adc",
105 .id = -1,
106 .parent = &clk_p,
107 .enable = s3c2410_clkcon_enable,
108 .ctrlbit = S3C2410_CLKCON_ADC,
109 }, {
110 .name = "i2c",
111 .id = -1,
112 .parent = &clk_p,
113 .enable = s3c2410_clkcon_enable,
114 .ctrlbit = S3C2410_CLKCON_IIC,
115 }, {
116 .name = "iis",
117 .id = -1,
118 .parent = &clk_p,
119 .enable = s3c2410_clkcon_enable,
120 .ctrlbit = S3C2410_CLKCON_IIS,
121 }, {
122 .name = "spi",
123 .id = -1,
124 .parent = &clk_p,
125 .enable = s3c2410_clkcon_enable,
126 .ctrlbit = S3C2410_CLKCON_SPI,
127 }
128};
129
130static struct clk init_clocks[] = {
131 {
132 .name = "lcd",
133 .id = -1,
134 .parent = &clk_h,
135 .enable = s3c2410_clkcon_enable,
136 .ctrlbit = S3C2410_CLKCON_LCDC,
137 }, {
138 .name = "gpio",
139 .id = -1,
140 .parent = &clk_p,
141 .enable = s3c2410_clkcon_enable,
142 .ctrlbit = S3C2410_CLKCON_GPIO,
143 }, {
144 .name = "usb-host",
145 .id = -1,
146 .parent = &clk_h,
147 .enable = s3c2410_clkcon_enable,
148 .ctrlbit = S3C2410_CLKCON_USBH,
149 }, {
150 .name = "usb-device",
151 .id = -1,
152 .parent = &clk_h,
153 .enable = s3c2410_clkcon_enable,
154 .ctrlbit = S3C2410_CLKCON_USBD,
155 }, {
156 .name = "timers",
157 .id = -1,
158 .parent = &clk_p,
159 .enable = s3c2410_clkcon_enable,
160 .ctrlbit = S3C2410_CLKCON_PWMT,
161 }, {
162 .name = "uart",
163 .id = 0,
164 .parent = &clk_p,
165 .enable = s3c2410_clkcon_enable,
166 .ctrlbit = S3C2410_CLKCON_UART0,
167 }, {
168 .name = "uart",
169 .id = 1,
170 .parent = &clk_p,
171 .enable = s3c2410_clkcon_enable,
172 .ctrlbit = S3C2410_CLKCON_UART1,
173 }, {
174 .name = "uart",
175 .id = 2,
176 .parent = &clk_p,
177 .enable = s3c2410_clkcon_enable,
178 .ctrlbit = S3C2410_CLKCON_UART2,
179 }, {
180 .name = "rtc",
181 .id = -1,
182 .parent = &clk_p,
183 .enable = s3c2410_clkcon_enable,
184 .ctrlbit = S3C2410_CLKCON_RTC,
185 }, {
186 .name = "watchdog",
187 .id = -1,
188 .parent = &clk_p,
189 .ctrlbit = 0,
190 }, {
191 .name = "usb-bus-host",
192 .id = -1,
193 .parent = &clk_usb_bus,
194 }, {
195 .name = "usb-bus-gadget",
196 .id = -1,
197 .parent = &clk_usb_bus,
198 },
199};
200
201/* s3c2410_baseclk_add()
202 *
203 * Add all the clocks used by the s3c2410 or compatible CPUs
204 * such as the S3C2440 and S3C2442.
205 *
206 * We cannot use a system device as we are needed before any
207 * of the init-calls that initialise the devices are actually
208 * done.
209*/
210
211int __init s3c2410_baseclk_add(void)
212{
213 unsigned long clkslow = __raw_readl(S3C2410_CLKSLOW);
214 unsigned long clkcon = __raw_readl(S3C2410_CLKCON);
215 struct clk *clkp;
216 struct clk *xtal;
217 int ret;
218 int ptr;
219
220 clk_upll.enable = s3c2410_upll_enable;
221
222 if (s3c24xx_register_clock(&clk_usb_bus) < 0)
223 printk(KERN_ERR "failed to register usb bus clock\n");
224
225 /* register clocks from clock array */
226
227 clkp = init_clocks;
228 for (ptr = 0; ptr < ARRAY_SIZE(init_clocks); ptr++, clkp++) {
229 /* ensure that we note the clock state */
230
231 clkp->usage = clkcon & clkp->ctrlbit ? 1 : 0;
232
233 ret = s3c24xx_register_clock(clkp);
234 if (ret < 0) {
235 printk(KERN_ERR "Failed to register clock %s (%d)\n",
236 clkp->name, ret);
237 }
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000238 }
239
Ben Dooksa21765a2007-02-11 18:31:01 +0100240 /* We must be careful disabling the clocks we are not intending to
Robert P. J. Day3a4fa0a2007-10-19 23:10:43 +0200241 * be using at boot time, as subsystems such as the LCD which do
Ben Dooksa21765a2007-02-11 18:31:01 +0100242 * their own DMA requests to the bus can cause the system to lockup
243 * if they where in the middle of requesting bus access.
244 *
245 * Disabling the LCD clock if the LCD is active is very dangerous,
246 * and therefore the bootloader should be careful to not enable
247 * the LCD clock if it is not needed.
248 */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000249
Ben Dooksa21765a2007-02-11 18:31:01 +0100250 /* install (and disable) the clocks we do not need immediately */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000251
Ben Dooksa21765a2007-02-11 18:31:01 +0100252 clkp = init_clocks_disable;
253 for (ptr = 0; ptr < ARRAY_SIZE(init_clocks_disable); ptr++, clkp++) {
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000254
Ben Dooksa21765a2007-02-11 18:31:01 +0100255 ret = s3c24xx_register_clock(clkp);
256 if (ret < 0) {
257 printk(KERN_ERR "Failed to register clock %s (%d)\n",
258 clkp->name, ret);
259 }
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000260
Ben Dooksa21765a2007-02-11 18:31:01 +0100261 s3c2410_clkcon_enable(clkp, 0);
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000262 }
263
Ben Dooksa21765a2007-02-11 18:31:01 +0100264 /* show the clock-slow value */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000265
Ben Dooksa21765a2007-02-11 18:31:01 +0100266 xtal = clk_get(NULL, "xtal");
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000267
Ben Dooksa21765a2007-02-11 18:31:01 +0100268 printk("CLOCK: Slow mode (%ld.%ld MHz), %s, MPLL %s, UPLL %s\n",
269 print_mhz(clk_get_rate(xtal) /
270 ( 2 * S3C2410_CLKSLOW_GET_SLOWVAL(clkslow))),
271 (clkslow & S3C2410_CLKSLOW_SLOW) ? "slow" : "fast",
272 (clkslow & S3C2410_CLKSLOW_MPLL_OFF) ? "off" : "on",
273 (clkslow & S3C2410_CLKSLOW_UCLK_OFF) ? "off" : "on");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700274
Linus Torvalds1da177e2005-04-16 15:20:36 -0700275 return 0;
276}