blob: d39d3c7875803e456c25c37babc8be9f9d30cef5 [file] [log] [blame]
Kukjin Kim09ec1d72013-01-31 16:54:38 -08001/*
Ben Dooksa21765a2007-02-11 18:31:01 +01002 * Copyright (c) 2006 Simtec Electronics
Linus Torvalds1da177e2005-04-16 15:20:36 -07003 * Ben Dooks <ben@simtec.co.uk>
4 *
Ben Dooksa21765a2007-02-11 18:31:01 +01005 * S3C2410,S3C2440,S3C2442 Clock control support
Linus Torvalds1da177e2005-04-16 15:20:36 -07006 *
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
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20*/
21
22#include <linux/init.h>
23#include <linux/module.h>
24#include <linux/kernel.h>
25#include <linux/list.h>
26#include <linux/errno.h>
27#include <linux/err.h>
Kay Sieversedbaa602011-12-21 16:26:03 -080028#include <linux/device.h>
Russell Kingf8ce2542006-01-07 16:15:52 +000029#include <linux/clk.h>
Arjan van de Ven00431702006-01-12 18:42:23 +000030#include <linux/mutex.h>
Ben Dooks8e40a2f2006-03-20 17:10:04 +000031#include <linux/delay.h>
Ben Dooksa21765a2007-02-11 18:31:01 +010032#include <linux/serial_core.h>
Russell Kingfced80c2008-09-06 12:10:45 +010033#include <linux/io.h>
Ben Dooksa21765a2007-02-11 18:31:01 +010034
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
Ben Dooksa2b7ba92008-10-07 22:26:09 +010039#include <plat/regs-serial.h>
Russell Kinga09e64f2008-08-05 16:14:15 +010040#include <mach/regs-clock.h>
41#include <mach/regs-gpio.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070042
Ben Dooksd5120ae2008-10-07 23:09:51 +010043#include <plat/clock.h>
Ben Dooksa2b7ba92008-10-07 22:26:09 +010044#include <plat/cpu.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070045
Ben Dooksa21765a2007-02-11 18:31:01 +010046int s3c2410_clkcon_enable(struct clk *clk, int enable)
Linus Torvalds1da177e2005-04-16 15:20:36 -070047{
Ben Dooksa21765a2007-02-11 18:31:01 +010048 unsigned int clocks = clk->ctrlbit;
49 unsigned long clkcon;
Linus Torvalds1da177e2005-04-16 15:20:36 -070050
Ben Dooksa21765a2007-02-11 18:31:01 +010051 clkcon = __raw_readl(S3C2410_CLKCON);
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000052
53 if (enable)
Ben Dooksa21765a2007-02-11 18:31:01 +010054 clkcon |= clocks;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000055 else
Ben Dooksa21765a2007-02-11 18:31:01 +010056 clkcon &= ~clocks;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000057
Ben Dooksa21765a2007-02-11 18:31:01 +010058 /* ensure none of the special function bits set */
59 clkcon &= ~(S3C2410_CLKCON_IDLE|S3C2410_CLKCON_POWER);
60
61 __raw_writel(clkcon, S3C2410_CLKCON);
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000062
63 return 0;
64}
65
Ben Dooksa21765a2007-02-11 18:31:01 +010066static int s3c2410_upll_enable(struct clk *clk, int enable)
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000067{
Ben Dooksa21765a2007-02-11 18:31:01 +010068 unsigned long clkslow = __raw_readl(S3C2410_CLKSLOW);
69 unsigned long orig = clkslow;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000070
Ben Dooksa21765a2007-02-11 18:31:01 +010071 if (enable)
72 clkslow &= ~S3C2410_CLKSLOW_UCLK_OFF;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000073 else
Ben Dooksa21765a2007-02-11 18:31:01 +010074 clkslow |= S3C2410_CLKSLOW_UCLK_OFF;
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000075
Ben Dooksa21765a2007-02-11 18:31:01 +010076 __raw_writel(clkslow, S3C2410_CLKSLOW);
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000077
Ben Dooksa21765a2007-02-11 18:31:01 +010078 /* if we started the UPLL, then allow to settle */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +000079
Ben Dooksa21765a2007-02-11 18:31:01 +010080 if (enable && (orig & S3C2410_CLKSLOW_UCLK_OFF))
81 udelay(200);
82
83 return 0;
84}
85
86/* standard clock definitions */
87
Ben Dooks4e046912010-04-28 12:58:13 +090088static struct clk init_clocks_off[] = {
Ben Dooksa21765a2007-02-11 18:31:01 +010089 {
90 .name = "nand",
Ben Dooksa21765a2007-02-11 18:31:01 +010091 .parent = &clk_h,
92 .enable = s3c2410_clkcon_enable,
93 .ctrlbit = S3C2410_CLKCON_NAND,
94 }, {
95 .name = "sdi",
Ben Dooksa21765a2007-02-11 18:31:01 +010096 .parent = &clk_p,
97 .enable = s3c2410_clkcon_enable,
98 .ctrlbit = S3C2410_CLKCON_SDI,
99 }, {
100 .name = "adc",
Ben Dooksa21765a2007-02-11 18:31:01 +0100101 .parent = &clk_p,
102 .enable = s3c2410_clkcon_enable,
103 .ctrlbit = S3C2410_CLKCON_ADC,
104 }, {
105 .name = "i2c",
Ben Dooksa21765a2007-02-11 18:31:01 +0100106 .parent = &clk_p,
107 .enable = s3c2410_clkcon_enable,
108 .ctrlbit = S3C2410_CLKCON_IIC,
109 }, {
110 .name = "iis",
Ben Dooksa21765a2007-02-11 18:31:01 +0100111 .parent = &clk_p,
112 .enable = s3c2410_clkcon_enable,
113 .ctrlbit = S3C2410_CLKCON_IIS,
114 }, {
115 .name = "spi",
Ben Dooksa21765a2007-02-11 18:31:01 +0100116 .parent = &clk_p,
117 .enable = s3c2410_clkcon_enable,
118 .ctrlbit = S3C2410_CLKCON_SPI,
119 }
120};
121
Sylwester Nawrockid8174682013-07-24 13:23:51 +0900122static struct clk clk_lcd = {
123 .name = "lcd",
124 .parent = &clk_h,
125 .enable = s3c2410_clkcon_enable,
126 .ctrlbit = S3C2410_CLKCON_LCDC,
127};
128
129static struct clk clk_gpio = {
130 .name = "gpio",
131 .parent = &clk_p,
132 .enable = s3c2410_clkcon_enable,
133 .ctrlbit = S3C2410_CLKCON_GPIO,
134};
135
136static struct clk clk_usb_host = {
137 .name = "usb-host",
138 .parent = &clk_h,
139 .enable = s3c2410_clkcon_enable,
140 .ctrlbit = S3C2410_CLKCON_USBH,
141};
142
143static struct clk clk_usb_device = {
144 .name = "usb-device",
145 .parent = &clk_h,
146 .enable = s3c2410_clkcon_enable,
147 .ctrlbit = S3C2410_CLKCON_USBD,
148};
149
150static struct clk clk_timers = {
151 .name = "timers",
152 .parent = &clk_p,
153 .enable = s3c2410_clkcon_enable,
154 .ctrlbit = S3C2410_CLKCON_PWMT,
155};
156
157struct clk s3c24xx_clk_uart0 = {
158 .name = "uart",
159 .devname = "s3c2410-uart.0",
160 .parent = &clk_p,
161 .enable = s3c2410_clkcon_enable,
162 .ctrlbit = S3C2410_CLKCON_UART0,
163};
164
165struct clk s3c24xx_clk_uart1 = {
166 .name = "uart",
167 .devname = "s3c2410-uart.1",
168 .parent = &clk_p,
169 .enable = s3c2410_clkcon_enable,
170 .ctrlbit = S3C2410_CLKCON_UART1,
171};
172
173struct clk s3c24xx_clk_uart2 = {
174 .name = "uart",
175 .devname = "s3c2410-uart.2",
176 .parent = &clk_p,
177 .enable = s3c2410_clkcon_enable,
178 .ctrlbit = S3C2410_CLKCON_UART2,
179};
180
181static struct clk clk_rtc = {
182 .name = "rtc",
183 .parent = &clk_p,
184 .enable = s3c2410_clkcon_enable,
185 .ctrlbit = S3C2410_CLKCON_RTC,
186};
187
188static struct clk clk_watchdog = {
189 .name = "watchdog",
190 .parent = &clk_p,
191 .ctrlbit = 0,
192};
193
194static struct clk clk_usb_bus_host = {
195 .name = "usb-bus-host",
196 .parent = &clk_usb_bus,
197};
198
199static struct clk clk_usb_bus_gadget = {
200 .name = "usb-bus-gadget",
201 .parent = &clk_usb_bus,
202};
203
204static struct clk *init_clocks[] = {
205 &clk_lcd,
206 &clk_gpio,
207 &clk_usb_host,
208 &clk_usb_device,
209 &clk_timers,
210 &s3c24xx_clk_uart0,
211 &s3c24xx_clk_uart1,
212 &s3c24xx_clk_uart2,
213 &clk_rtc,
214 &clk_watchdog,
215 &clk_usb_bus_host,
216 &clk_usb_bus_gadget,
Ben Dooksa21765a2007-02-11 18:31:01 +0100217};
218
219/* s3c2410_baseclk_add()
220 *
221 * Add all the clocks used by the s3c2410 or compatible CPUs
222 * such as the S3C2440 and S3C2442.
223 *
224 * We cannot use a system device as we are needed before any
225 * of the init-calls that initialise the devices are actually
226 * done.
227*/
228
229int __init s3c2410_baseclk_add(void)
230{
231 unsigned long clkslow = __raw_readl(S3C2410_CLKSLOW);
232 unsigned long clkcon = __raw_readl(S3C2410_CLKCON);
Ben Dooksa21765a2007-02-11 18:31:01 +0100233 struct clk *xtal;
234 int ret;
235 int ptr;
236
237 clk_upll.enable = s3c2410_upll_enable;
238
239 if (s3c24xx_register_clock(&clk_usb_bus) < 0)
240 printk(KERN_ERR "failed to register usb bus clock\n");
241
242 /* register clocks from clock array */
243
Sylwester Nawrockid8174682013-07-24 13:23:51 +0900244 for (ptr = 0; ptr < ARRAY_SIZE(init_clocks); ptr++) {
245 struct clk *clkp = init_clocks[ptr];
246
Ben Dooksa21765a2007-02-11 18:31:01 +0100247 /* ensure that we note the clock state */
248
249 clkp->usage = clkcon & clkp->ctrlbit ? 1 : 0;
250
251 ret = s3c24xx_register_clock(clkp);
252 if (ret < 0) {
253 printk(KERN_ERR "Failed to register clock %s (%d)\n",
254 clkp->name, ret);
255 }
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000256 }
257
Ben Dooksa21765a2007-02-11 18:31:01 +0100258 /* We must be careful disabling the clocks we are not intending to
Robert P. J. Day3a4fa0a2007-10-19 23:10:43 +0200259 * be using at boot time, as subsystems such as the LCD which do
Ben Dooksa21765a2007-02-11 18:31:01 +0100260 * their own DMA requests to the bus can cause the system to lockup
261 * if they where in the middle of requesting bus access.
262 *
263 * Disabling the LCD clock if the LCD is active is very dangerous,
264 * and therefore the bootloader should be careful to not enable
265 * the LCD clock if it is not needed.
266 */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000267
Ben Dooksa21765a2007-02-11 18:31:01 +0100268 /* install (and disable) the clocks we do not need immediately */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000269
Ben Dooks4e046912010-04-28 12:58:13 +0900270 s3c_register_clocks(init_clocks_off, ARRAY_SIZE(init_clocks_off));
271 s3c_disable_clocks(init_clocks_off, ARRAY_SIZE(init_clocks_off));
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000272
Ben Dooksa21765a2007-02-11 18:31:01 +0100273 /* show the clock-slow value */
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000274
Ben Dooksa21765a2007-02-11 18:31:01 +0100275 xtal = clk_get(NULL, "xtal");
Ben Dooks3fc3e1c2006-03-20 17:10:07 +0000276
Ben Dooksa21765a2007-02-11 18:31:01 +0100277 printk("CLOCK: Slow mode (%ld.%ld MHz), %s, MPLL %s, UPLL %s\n",
278 print_mhz(clk_get_rate(xtal) /
279 ( 2 * S3C2410_CLKSLOW_GET_SLOWVAL(clkslow))),
280 (clkslow & S3C2410_CLKSLOW_SLOW) ? "slow" : "fast",
281 (clkslow & S3C2410_CLKSLOW_MPLL_OFF) ? "off" : "on",
282 (clkslow & S3C2410_CLKSLOW_UCLK_OFF) ? "off" : "on");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700283
Linus Torvalds1da177e2005-04-16 15:20:36 -0700284 return 0;
285}