clk-tegra20.c revision 6d5b988e7dc56bb97c39bdcbc006fadcd6ca371b
1/*
2 * Copyright (c) 2012, NVIDIA CORPORATION.  All rights reserved.
3 *
4 * This program is free software; you can redistribute it and/or modify it
5 * under the terms and conditions of the GNU General Public License,
6 * version 2, as published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope it will be useful, but WITHOUT
9 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
10 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
11 * more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
15 */
16
17#include <linux/io.h>
18#include <linux/clk.h>
19#include <linux/clk-provider.h>
20#include <linux/clkdev.h>
21#include <linux/of.h>
22#include <linux/of_address.h>
23#include <linux/clk/tegra.h>
24#include <linux/delay.h>
25#include <dt-bindings/clock/tegra20-car.h>
26
27#include "clk.h"
28#include "clk-id.h"
29
30#define OSC_CTRL 0x50
31#define OSC_CTRL_OSC_FREQ_MASK (3<<30)
32#define OSC_CTRL_OSC_FREQ_13MHZ (0<<30)
33#define OSC_CTRL_OSC_FREQ_19_2MHZ (1<<30)
34#define OSC_CTRL_OSC_FREQ_12MHZ (2<<30)
35#define OSC_CTRL_OSC_FREQ_26MHZ (3<<30)
36#define OSC_CTRL_MASK (0x3f2 | OSC_CTRL_OSC_FREQ_MASK)
37
38#define OSC_CTRL_PLL_REF_DIV_MASK (3<<28)
39#define OSC_CTRL_PLL_REF_DIV_1		(0<<28)
40#define OSC_CTRL_PLL_REF_DIV_2		(1<<28)
41#define OSC_CTRL_PLL_REF_DIV_4		(2<<28)
42
43#define OSC_FREQ_DET 0x58
44#define OSC_FREQ_DET_TRIG (1<<31)
45
46#define OSC_FREQ_DET_STATUS 0x5c
47#define OSC_FREQ_DET_BUSY (1<<31)
48#define OSC_FREQ_DET_CNT_MASK 0xFFFF
49
50#define TEGRA20_CLK_PERIPH_BANKS	3
51
52#define PLLS_BASE 0xf0
53#define PLLS_MISC 0xf4
54#define PLLC_BASE 0x80
55#define PLLC_MISC 0x8c
56#define PLLM_BASE 0x90
57#define PLLM_MISC 0x9c
58#define PLLP_BASE 0xa0
59#define PLLP_MISC 0xac
60#define PLLA_BASE 0xb0
61#define PLLA_MISC 0xbc
62#define PLLU_BASE 0xc0
63#define PLLU_MISC 0xcc
64#define PLLD_BASE 0xd0
65#define PLLD_MISC 0xdc
66#define PLLX_BASE 0xe0
67#define PLLX_MISC 0xe4
68#define PLLE_BASE 0xe8
69#define PLLE_MISC 0xec
70
71#define PLL_BASE_LOCK BIT(27)
72#define PLLE_MISC_LOCK BIT(11)
73
74#define PLL_MISC_LOCK_ENABLE 18
75#define PLLDU_MISC_LOCK_ENABLE 22
76#define PLLE_MISC_LOCK_ENABLE 9
77
78#define PLLC_OUT 0x84
79#define PLLM_OUT 0x94
80#define PLLP_OUTA 0xa4
81#define PLLP_OUTB 0xa8
82#define PLLA_OUT 0xb4
83
84#define CCLK_BURST_POLICY 0x20
85#define SUPER_CCLK_DIVIDER 0x24
86#define SCLK_BURST_POLICY 0x28
87#define SUPER_SCLK_DIVIDER 0x2c
88#define CLK_SYSTEM_RATE 0x30
89
90#define CCLK_BURST_POLICY_SHIFT	28
91#define CCLK_RUN_POLICY_SHIFT	4
92#define CCLK_IDLE_POLICY_SHIFT	0
93#define CCLK_IDLE_POLICY	1
94#define CCLK_RUN_POLICY		2
95#define CCLK_BURST_POLICY_PLLX	8
96
97#define CLK_SOURCE_I2S1 0x100
98#define CLK_SOURCE_I2S2 0x104
99#define CLK_SOURCE_PWM 0x110
100#define CLK_SOURCE_SPI 0x114
101#define CLK_SOURCE_XIO 0x120
102#define CLK_SOURCE_TWC 0x12c
103#define CLK_SOURCE_IDE 0x144
104#define CLK_SOURCE_HDMI 0x18c
105#define CLK_SOURCE_DISP1 0x138
106#define CLK_SOURCE_DISP2 0x13c
107#define CLK_SOURCE_CSITE 0x1d4
108#define CLK_SOURCE_I2C1 0x124
109#define CLK_SOURCE_I2C2 0x198
110#define CLK_SOURCE_I2C3 0x1b8
111#define CLK_SOURCE_DVC 0x128
112#define CLK_SOURCE_UARTA 0x178
113#define CLK_SOURCE_UARTB 0x17c
114#define CLK_SOURCE_UARTC 0x1a0
115#define CLK_SOURCE_UARTD 0x1c0
116#define CLK_SOURCE_UARTE 0x1c4
117#define CLK_SOURCE_EMC 0x19c
118
119#define AUDIO_SYNC_CLK 0x38
120
121/* Tegra CPU clock and reset control regs */
122#define TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX		0x4c
123#define TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_SET	0x340
124#define TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_CLR	0x344
125
126#define CPU_CLOCK(cpu)	(0x1 << (8 + cpu))
127#define CPU_RESET(cpu)	(0x1111ul << (cpu))
128
129#ifdef CONFIG_PM_SLEEP
130static struct cpu_clk_suspend_context {
131	u32 pllx_misc;
132	u32 pllx_base;
133
134	u32 cpu_burst;
135	u32 clk_csite_src;
136	u32 cclk_divider;
137} tegra20_cpu_clk_sctx;
138#endif
139
140static void __iomem *clk_base;
141static void __iomem *pmc_base;
142
143#define TEGRA_INIT_DATA_MUX(_name, _parents, _offset,	\
144			    _clk_num, _gate_flags, _clk_id)	\
145	TEGRA_INIT_DATA(_name, NULL, NULL, _parents, _offset,	\
146			30, 2, 0, 0, 8, 1, TEGRA_DIVIDER_ROUND_UP,	\
147			_clk_num, \
148			_gate_flags, _clk_id)
149
150#define TEGRA_INIT_DATA_DIV16(_name, _parents, _offset, \
151			      _clk_num, _gate_flags, _clk_id)	\
152	TEGRA_INIT_DATA(_name, NULL, NULL, _parents, _offset,	\
153			30, 2, 0, 0, 16, 0, TEGRA_DIVIDER_ROUND_UP, \
154			_clk_num, _gate_flags,	\
155			_clk_id)
156
157#define TEGRA_INIT_DATA_NODIV(_name, _parents, _offset, \
158			      _mux_shift, _mux_width, _clk_num, \
159			      _gate_flags, _clk_id)			\
160	TEGRA_INIT_DATA(_name, NULL, NULL, _parents, _offset,	\
161			_mux_shift, _mux_width, 0, 0, 0, 0, 0, \
162			_clk_num, _gate_flags,	\
163			_clk_id)
164
165static struct clk **clks;
166
167static struct tegra_clk_pll_freq_table pll_c_freq_table[] = {
168	{ 12000000, 600000000, 600, 12, 0, 8 },
169	{ 13000000, 600000000, 600, 13, 0, 8 },
170	{ 19200000, 600000000, 500, 16, 0, 6 },
171	{ 26000000, 600000000, 600, 26, 0, 8 },
172	{ 0, 0, 0, 0, 0, 0 },
173};
174
175static struct tegra_clk_pll_freq_table pll_m_freq_table[] = {
176	{ 12000000, 666000000, 666, 12, 0, 8},
177	{ 13000000, 666000000, 666, 13, 0, 8},
178	{ 19200000, 666000000, 555, 16, 0, 8},
179	{ 26000000, 666000000, 666, 26, 0, 8},
180	{ 12000000, 600000000, 600, 12, 0, 8},
181	{ 13000000, 600000000, 600, 13, 0, 8},
182	{ 19200000, 600000000, 375, 12, 0, 6},
183	{ 26000000, 600000000, 600, 26, 0, 8},
184	{ 0, 0, 0, 0, 0, 0 },
185};
186
187static struct tegra_clk_pll_freq_table pll_p_freq_table[] = {
188	{ 12000000, 216000000, 432, 12, 1, 8},
189	{ 13000000, 216000000, 432, 13, 1, 8},
190	{ 19200000, 216000000, 90,   4, 1, 1},
191	{ 26000000, 216000000, 432, 26, 1, 8},
192	{ 12000000, 432000000, 432, 12, 0, 8},
193	{ 13000000, 432000000, 432, 13, 0, 8},
194	{ 19200000, 432000000, 90,   4, 0, 1},
195	{ 26000000, 432000000, 432, 26, 0, 8},
196	{ 0, 0, 0, 0, 0, 0 },
197};
198
199static struct tegra_clk_pll_freq_table pll_a_freq_table[] = {
200	{ 28800000, 56448000, 49, 25, 0, 1},
201	{ 28800000, 73728000, 64, 25, 0, 1},
202	{ 28800000, 24000000,  5,  6, 0, 1},
203	{ 0, 0, 0, 0, 0, 0 },
204};
205
206static struct tegra_clk_pll_freq_table pll_d_freq_table[] = {
207	{ 12000000, 216000000, 216, 12, 0, 4},
208	{ 13000000, 216000000, 216, 13, 0, 4},
209	{ 19200000, 216000000, 135, 12, 0, 3},
210	{ 26000000, 216000000, 216, 26, 0, 4},
211
212	{ 12000000, 594000000, 594, 12, 0, 8},
213	{ 13000000, 594000000, 594, 13, 0, 8},
214	{ 19200000, 594000000, 495, 16, 0, 8},
215	{ 26000000, 594000000, 594, 26, 0, 8},
216
217	{ 12000000, 1000000000, 1000, 12, 0, 12},
218	{ 13000000, 1000000000, 1000, 13, 0, 12},
219	{ 19200000, 1000000000, 625,  12, 0, 8},
220	{ 26000000, 1000000000, 1000, 26, 0, 12},
221
222	{ 0, 0, 0, 0, 0, 0 },
223};
224
225static struct tegra_clk_pll_freq_table pll_u_freq_table[] = {
226	{ 12000000, 480000000, 960, 12, 0, 0},
227	{ 13000000, 480000000, 960, 13, 0, 0},
228	{ 19200000, 480000000, 200, 4,  0, 0},
229	{ 26000000, 480000000, 960, 26, 0, 0},
230	{ 0, 0, 0, 0, 0, 0 },
231};
232
233static struct tegra_clk_pll_freq_table pll_x_freq_table[] = {
234	/* 1 GHz */
235	{ 12000000, 1000000000, 1000, 12, 0, 12},
236	{ 13000000, 1000000000, 1000, 13, 0, 12},
237	{ 19200000, 1000000000, 625,  12, 0, 8},
238	{ 26000000, 1000000000, 1000, 26, 0, 12},
239
240	/* 912 MHz */
241	{ 12000000, 912000000,  912,  12, 0, 12},
242	{ 13000000, 912000000,  912,  13, 0, 12},
243	{ 19200000, 912000000,  760,  16, 0, 8},
244	{ 26000000, 912000000,  912,  26, 0, 12},
245
246	/* 816 MHz */
247	{ 12000000, 816000000,  816,  12, 0, 12},
248	{ 13000000, 816000000,  816,  13, 0, 12},
249	{ 19200000, 816000000,  680,  16, 0, 8},
250	{ 26000000, 816000000,  816,  26, 0, 12},
251
252	/* 760 MHz */
253	{ 12000000, 760000000,  760,  12, 0, 12},
254	{ 13000000, 760000000,  760,  13, 0, 12},
255	{ 19200000, 760000000,  950,  24, 0, 8},
256	{ 26000000, 760000000,  760,  26, 0, 12},
257
258	/* 750 MHz */
259	{ 12000000, 750000000,  750,  12, 0, 12},
260	{ 13000000, 750000000,  750,  13, 0, 12},
261	{ 19200000, 750000000,  625,  16, 0, 8},
262	{ 26000000, 750000000,  750,  26, 0, 12},
263
264	/* 608 MHz */
265	{ 12000000, 608000000,  608,  12, 0, 12},
266	{ 13000000, 608000000,  608,  13, 0, 12},
267	{ 19200000, 608000000,  380,  12, 0, 8},
268	{ 26000000, 608000000,  608,  26, 0, 12},
269
270	/* 456 MHz */
271	{ 12000000, 456000000,  456,  12, 0, 12},
272	{ 13000000, 456000000,  456,  13, 0, 12},
273	{ 19200000, 456000000,  380,  16, 0, 8},
274	{ 26000000, 456000000,  456,  26, 0, 12},
275
276	/* 312 MHz */
277	{ 12000000, 312000000,  312,  12, 0, 12},
278	{ 13000000, 312000000,  312,  13, 0, 12},
279	{ 19200000, 312000000,  260,  16, 0, 8},
280	{ 26000000, 312000000,  312,  26, 0, 12},
281
282	{ 0, 0, 0, 0, 0, 0 },
283};
284
285static struct tegra_clk_pll_freq_table pll_e_freq_table[] = {
286	{ 12000000, 100000000,  200,  24, 0, 0 },
287	{ 0, 0, 0, 0, 0, 0 },
288};
289
290/* PLL parameters */
291static struct tegra_clk_pll_params pll_c_params = {
292	.input_min = 2000000,
293	.input_max = 31000000,
294	.cf_min = 1000000,
295	.cf_max = 6000000,
296	.vco_min = 20000000,
297	.vco_max = 1400000000,
298	.base_reg = PLLC_BASE,
299	.misc_reg = PLLC_MISC,
300	.lock_mask = PLL_BASE_LOCK,
301	.lock_enable_bit_idx = PLL_MISC_LOCK_ENABLE,
302	.lock_delay = 300,
303	.freq_table = pll_c_freq_table,
304	.flags = TEGRA_PLL_HAS_CPCON,
305};
306
307static struct tegra_clk_pll_params pll_m_params = {
308	.input_min = 2000000,
309	.input_max = 31000000,
310	.cf_min = 1000000,
311	.cf_max = 6000000,
312	.vco_min = 20000000,
313	.vco_max = 1200000000,
314	.base_reg = PLLM_BASE,
315	.misc_reg = PLLM_MISC,
316	.lock_mask = PLL_BASE_LOCK,
317	.lock_enable_bit_idx = PLL_MISC_LOCK_ENABLE,
318	.lock_delay = 300,
319	.freq_table = pll_m_freq_table,
320	.flags = TEGRA_PLL_HAS_CPCON,
321};
322
323static struct tegra_clk_pll_params pll_p_params = {
324	.input_min = 2000000,
325	.input_max = 31000000,
326	.cf_min = 1000000,
327	.cf_max = 6000000,
328	.vco_min = 20000000,
329	.vco_max = 1400000000,
330	.base_reg = PLLP_BASE,
331	.misc_reg = PLLP_MISC,
332	.lock_mask = PLL_BASE_LOCK,
333	.lock_enable_bit_idx = PLL_MISC_LOCK_ENABLE,
334	.lock_delay = 300,
335	.freq_table = pll_p_freq_table,
336	.flags = TEGRA_PLL_FIXED | TEGRA_PLL_HAS_CPCON,
337	.fixed_rate =  216000000,
338};
339
340static struct tegra_clk_pll_params pll_a_params = {
341	.input_min = 2000000,
342	.input_max = 31000000,
343	.cf_min = 1000000,
344	.cf_max = 6000000,
345	.vco_min = 20000000,
346	.vco_max = 1400000000,
347	.base_reg = PLLA_BASE,
348	.misc_reg = PLLA_MISC,
349	.lock_mask = PLL_BASE_LOCK,
350	.lock_enable_bit_idx = PLL_MISC_LOCK_ENABLE,
351	.lock_delay = 300,
352	.freq_table = pll_a_freq_table,
353	.flags = TEGRA_PLL_HAS_CPCON,
354};
355
356static struct tegra_clk_pll_params pll_d_params = {
357	.input_min = 2000000,
358	.input_max = 40000000,
359	.cf_min = 1000000,
360	.cf_max = 6000000,
361	.vco_min = 40000000,
362	.vco_max = 1000000000,
363	.base_reg = PLLD_BASE,
364	.misc_reg = PLLD_MISC,
365	.lock_mask = PLL_BASE_LOCK,
366	.lock_enable_bit_idx = PLLDU_MISC_LOCK_ENABLE,
367	.lock_delay = 1000,
368	.freq_table = pll_d_freq_table,
369	.flags = TEGRA_PLL_HAS_CPCON,
370};
371
372static struct pdiv_map pllu_p[] = {
373	{ .pdiv = 1, .hw_val = 1 },
374	{ .pdiv = 2, .hw_val = 0 },
375	{ .pdiv = 0, .hw_val = 0 },
376};
377
378static struct tegra_clk_pll_params pll_u_params = {
379	.input_min = 2000000,
380	.input_max = 40000000,
381	.cf_min = 1000000,
382	.cf_max = 6000000,
383	.vco_min = 48000000,
384	.vco_max = 960000000,
385	.base_reg = PLLU_BASE,
386	.misc_reg = PLLU_MISC,
387	.lock_mask = PLL_BASE_LOCK,
388	.lock_enable_bit_idx = PLLDU_MISC_LOCK_ENABLE,
389	.lock_delay = 1000,
390	.pdiv_tohw = pllu_p,
391	.freq_table = pll_u_freq_table,
392	.flags = TEGRA_PLLU | TEGRA_PLL_HAS_CPCON,
393};
394
395static struct tegra_clk_pll_params pll_x_params = {
396	.input_min = 2000000,
397	.input_max = 31000000,
398	.cf_min = 1000000,
399	.cf_max = 6000000,
400	.vco_min = 20000000,
401	.vco_max = 1200000000,
402	.base_reg = PLLX_BASE,
403	.misc_reg = PLLX_MISC,
404	.lock_mask = PLL_BASE_LOCK,
405	.lock_enable_bit_idx = PLL_MISC_LOCK_ENABLE,
406	.lock_delay = 300,
407	.freq_table = pll_x_freq_table,
408	.flags = TEGRA_PLL_HAS_CPCON,
409};
410
411static struct tegra_clk_pll_params pll_e_params = {
412	.input_min = 12000000,
413	.input_max = 12000000,
414	.cf_min = 0,
415	.cf_max = 0,
416	.vco_min = 0,
417	.vco_max = 0,
418	.base_reg = PLLE_BASE,
419	.misc_reg = PLLE_MISC,
420	.lock_mask = PLLE_MISC_LOCK,
421	.lock_enable_bit_idx = PLLE_MISC_LOCK_ENABLE,
422	.lock_delay = 0,
423	.freq_table = pll_e_freq_table,
424	.flags = TEGRA_PLL_FIXED,
425	.fixed_rate = 100000000,
426};
427
428static struct tegra_devclk devclks[] __initdata = {
429	{ .con_id = "pll_c", .dt_id = TEGRA20_CLK_PLL_C },
430	{ .con_id = "pll_c_out1", .dt_id = TEGRA20_CLK_PLL_C_OUT1 },
431	{ .con_id = "pll_p", .dt_id = TEGRA20_CLK_PLL_P },
432	{ .con_id = "pll_p_out1", .dt_id = TEGRA20_CLK_PLL_P_OUT1 },
433	{ .con_id = "pll_p_out2", .dt_id = TEGRA20_CLK_PLL_P_OUT2 },
434	{ .con_id = "pll_p_out3", .dt_id = TEGRA20_CLK_PLL_P_OUT3 },
435	{ .con_id = "pll_p_out4", .dt_id = TEGRA20_CLK_PLL_P_OUT4 },
436	{ .con_id = "pll_m", .dt_id = TEGRA20_CLK_PLL_M },
437	{ .con_id = "pll_m_out1", .dt_id = TEGRA20_CLK_PLL_M_OUT1 },
438	{ .con_id = "pll_x", .dt_id = TEGRA20_CLK_PLL_X },
439	{ .con_id = "pll_u", .dt_id = TEGRA20_CLK_PLL_U },
440	{ .con_id = "pll_d", .dt_id = TEGRA20_CLK_PLL_D },
441	{ .con_id = "pll_d_out0", .dt_id = TEGRA20_CLK_PLL_D_OUT0 },
442	{ .con_id = "pll_a", .dt_id = TEGRA20_CLK_PLL_A },
443	{ .con_id = "pll_a_out0", .dt_id = TEGRA20_CLK_PLL_A_OUT0 },
444	{ .con_id = "pll_e", .dt_id = TEGRA20_CLK_PLL_E },
445	{ .con_id = "cclk", .dt_id = TEGRA20_CLK_CCLK },
446	{ .con_id = "sclk", .dt_id = TEGRA20_CLK_SCLK },
447	{ .con_id = "hclk", .dt_id = TEGRA20_CLK_HCLK },
448	{ .con_id = "pclk", .dt_id = TEGRA20_CLK_PCLK },
449	{ .con_id = "fuse", .dt_id = TEGRA20_CLK_FUSE },
450	{ .con_id = "twd", .dt_id = TEGRA20_CLK_TWD },
451	{ .con_id = "audio", .dt_id = TEGRA20_CLK_AUDIO },
452	{ .con_id = "audio_2x", .dt_id = TEGRA20_CLK_AUDIO_2X },
453	{ .dev_id = "tegra20-ac97", .dt_id = TEGRA20_CLK_AC97 },
454	{ .dev_id = "tegra-apbdma", .dt_id = TEGRA20_CLK_APBDMA },
455	{ .dev_id = "rtc-tegra", .dt_id = TEGRA20_CLK_RTC },
456	{ .dev_id = "timer", .dt_id = TEGRA20_CLK_TIMER },
457	{ .dev_id = "tegra-kbc", .dt_id = TEGRA20_CLK_KBC },
458	{ .con_id = "csus", .dev_id =  "tegra_camera", .dt_id = TEGRA20_CLK_CSUS },
459	{ .con_id = "vcp", .dev_id = "tegra-avp", .dt_id = TEGRA20_CLK_VCP },
460	{ .con_id = "bsea", .dev_id = "tegra-avp", .dt_id = TEGRA20_CLK_BSEA },
461	{ .con_id = "bsev", .dev_id = "tegra-aes", .dt_id = TEGRA20_CLK_BSEV },
462	{ .con_id = "emc", .dt_id = TEGRA20_CLK_EMC },
463	{ .dev_id = "fsl-tegra-udc", .dt_id = TEGRA20_CLK_USBD },
464	{ .dev_id = "tegra-ehci.1", .dt_id = TEGRA20_CLK_USB2 },
465	{ .dev_id = "tegra-ehci.2", .dt_id = TEGRA20_CLK_USB3 },
466	{ .dev_id = "dsi", .dt_id = TEGRA20_CLK_DSI },
467	{ .con_id = "csi", .dev_id = "tegra_camera", .dt_id = TEGRA20_CLK_CSI },
468	{ .con_id = "isp", .dev_id = "tegra_camera", .dt_id = TEGRA20_CLK_ISP },
469	{ .con_id = "pex", .dt_id = TEGRA20_CLK_PEX },
470	{ .con_id = "afi", .dt_id = TEGRA20_CLK_AFI },
471	{ .con_id = "pcie_xclk", .dt_id = TEGRA20_CLK_PCIE_XCLK },
472	{ .con_id = "cdev1", .dt_id = TEGRA20_CLK_CDEV1 },
473	{ .con_id = "cdev2", .dt_id = TEGRA20_CLK_CDEV2 },
474	{ .con_id = "clk_32k", .dt_id = TEGRA20_CLK_CLK_32K },
475	{ .con_id = "blink", .dt_id = TEGRA20_CLK_BLINK },
476	{ .con_id = "clk_m", .dt_id = TEGRA20_CLK_CLK_M },
477	{ .con_id = "pll_ref", .dt_id = TEGRA20_CLK_PLL_REF },
478	{ .dev_id = "tegra20-i2s.0", .dt_id = TEGRA20_CLK_I2S1 },
479	{ .dev_id = "tegra20-i2s.1", .dt_id = TEGRA20_CLK_I2S2 },
480	{ .con_id = "spdif_out", .dev_id = "tegra20-spdif", .dt_id = TEGRA20_CLK_SPDIF_OUT },
481	{ .con_id = "spdif_in", .dev_id = "tegra20-spdif", .dt_id = TEGRA20_CLK_SPDIF_IN },
482	{ .dev_id = "spi_tegra.0", .dt_id = TEGRA20_CLK_SBC1 },
483	{ .dev_id = "spi_tegra.1", .dt_id = TEGRA20_CLK_SBC2 },
484	{ .dev_id = "spi_tegra.2", .dt_id = TEGRA20_CLK_SBC3 },
485	{ .dev_id = "spi_tegra.3", .dt_id = TEGRA20_CLK_SBC4 },
486	{ .dev_id = "spi", .dt_id = TEGRA20_CLK_SPI },
487	{ .dev_id = "xio", .dt_id = TEGRA20_CLK_XIO },
488	{ .dev_id = "twc", .dt_id = TEGRA20_CLK_TWC },
489	{ .dev_id = "ide", .dt_id = TEGRA20_CLK_IDE },
490	{ .dev_id = "tegra_nand", .dt_id = TEGRA20_CLK_NDFLASH },
491	{ .dev_id = "vfir", .dt_id = TEGRA20_CLK_VFIR },
492	{ .dev_id = "csite", .dt_id = TEGRA20_CLK_CSITE },
493	{ .dev_id = "la", .dt_id = TEGRA20_CLK_LA },
494	{ .dev_id = "tegra_w1", .dt_id = TEGRA20_CLK_OWR },
495	{ .dev_id = "mipi", .dt_id = TEGRA20_CLK_MIPI },
496	{ .dev_id = "vde", .dt_id = TEGRA20_CLK_VDE },
497	{ .con_id = "vi", .dev_id =  "tegra_camera", .dt_id = TEGRA20_CLK_VI },
498	{ .dev_id = "epp", .dt_id = TEGRA20_CLK_EPP },
499	{ .dev_id = "mpe", .dt_id = TEGRA20_CLK_MPE },
500	{ .dev_id = "host1x", .dt_id = TEGRA20_CLK_HOST1X },
501	{ .dev_id = "3d", .dt_id = TEGRA20_CLK_GR3D },
502	{ .dev_id = "2d", .dt_id = TEGRA20_CLK_GR2D },
503	{ .dev_id = "tegra-nor", .dt_id = TEGRA20_CLK_NOR },
504	{ .dev_id = "sdhci-tegra.0", .dt_id = TEGRA20_CLK_SDMMC1 },
505	{ .dev_id = "sdhci-tegra.1", .dt_id = TEGRA20_CLK_SDMMC2 },
506	{ .dev_id = "sdhci-tegra.2", .dt_id = TEGRA20_CLK_SDMMC3 },
507	{ .dev_id = "sdhci-tegra.3", .dt_id = TEGRA20_CLK_SDMMC4 },
508	{ .dev_id = "cve", .dt_id = TEGRA20_CLK_CVE },
509	{ .dev_id = "tvo", .dt_id = TEGRA20_CLK_TVO },
510	{ .dev_id = "tvdac", .dt_id = TEGRA20_CLK_TVDAC },
511	{ .con_id = "vi_sensor", .dev_id = "tegra_camera", .dt_id = TEGRA20_CLK_VI_SENSOR },
512	{ .dev_id = "hdmi", .dt_id = TEGRA20_CLK_HDMI },
513	{ .con_id = "div-clk", .dev_id = "tegra-i2c.0", .dt_id = TEGRA20_CLK_I2C1 },
514	{ .con_id = "div-clk", .dev_id = "tegra-i2c.1", .dt_id = TEGRA20_CLK_I2C2 },
515	{ .con_id = "div-clk", .dev_id = "tegra-i2c.2", .dt_id = TEGRA20_CLK_I2C3 },
516	{ .con_id = "div-clk", .dev_id = "tegra-i2c.3", .dt_id = TEGRA20_CLK_DVC },
517	{ .dev_id = "tegra-pwm", .dt_id = TEGRA20_CLK_PWM },
518	{ .dev_id = "tegra_uart.0", .dt_id = TEGRA20_CLK_UARTA },
519	{ .dev_id = "tegra_uart.1", .dt_id = TEGRA20_CLK_UARTB },
520	{ .dev_id = "tegra_uart.2", .dt_id = TEGRA20_CLK_UARTC },
521	{ .dev_id = "tegra_uart.3", .dt_id = TEGRA20_CLK_UARTD },
522	{ .dev_id = "tegra_uart.4", .dt_id = TEGRA20_CLK_UARTE },
523	{ .dev_id = "tegradc.0", .dt_id = TEGRA20_CLK_DISP1 },
524	{ .dev_id = "tegradc.1", .dt_id = TEGRA20_CLK_DISP2 },
525};
526
527static struct tegra_clk tegra20_clks[tegra_clk_max] __initdata = {
528	[tegra_clk_spdif_out] = { .dt_id = TEGRA20_CLK_SPDIF_OUT, .present = true },
529	[tegra_clk_spdif_in] = { .dt_id = TEGRA20_CLK_SPDIF_IN, .present = true },
530	[tegra_clk_sdmmc1] = { .dt_id = TEGRA20_CLK_SDMMC1, .present = true },
531	[tegra_clk_sdmmc2] = { .dt_id = TEGRA20_CLK_SDMMC2, .present = true },
532	[tegra_clk_sdmmc3] = { .dt_id = TEGRA20_CLK_SDMMC3, .present = true },
533	[tegra_clk_sdmmc4] = { .dt_id = TEGRA20_CLK_SDMMC4, .present = true },
534	[tegra_clk_la] = { .dt_id = TEGRA20_CLK_LA, .present = true },
535	[tegra_clk_csite] = { .dt_id = TEGRA20_CLK_CSITE, .present = true },
536	[tegra_clk_vfir] = { .dt_id = TEGRA20_CLK_VFIR, .present = true },
537	[tegra_clk_mipi] = { .dt_id = TEGRA20_CLK_MIPI, .present = true },
538	[tegra_clk_nor] = { .dt_id = TEGRA20_CLK_NOR, .present = true },
539	[tegra_clk_rtc] = { .dt_id = TEGRA20_CLK_RTC, .present = true },
540	[tegra_clk_timer] = { .dt_id = TEGRA20_CLK_TIMER, .present = true },
541	[tegra_clk_kbc] = { .dt_id = TEGRA20_CLK_KBC, .present = true },
542	[tegra_clk_csus] = { .dt_id = TEGRA20_CLK_CSUS, .present = true },
543	[tegra_clk_vcp] = { .dt_id = TEGRA20_CLK_VCP, .present = true },
544	[tegra_clk_bsea] = { .dt_id = TEGRA20_CLK_BSEA, .present = true },
545	[tegra_clk_bsev] = { .dt_id = TEGRA20_CLK_BSEV, .present = true },
546	[tegra_clk_usbd] = { .dt_id = TEGRA20_CLK_USBD, .present = true },
547	[tegra_clk_usb2] = { .dt_id = TEGRA20_CLK_USB2, .present = true },
548	[tegra_clk_usb3] = { .dt_id = TEGRA20_CLK_USB3, .present = true },
549	[tegra_clk_csi] = { .dt_id = TEGRA20_CLK_CSI, .present = true },
550	[tegra_clk_isp] = { .dt_id = TEGRA20_CLK_ISP, .present = true },
551	[tegra_clk_clk_32k] = { .dt_id = TEGRA20_CLK_CLK_32K, .present = true },
552	[tegra_clk_blink] = { .dt_id = TEGRA20_CLK_BLINK, .present = true },
553	[tegra_clk_hclk] = { .dt_id = TEGRA20_CLK_HCLK, .present = true },
554	[tegra_clk_pclk] = { .dt_id = TEGRA20_CLK_PCLK, .present = true },
555	[tegra_clk_pll_p_out1] = { .dt_id = TEGRA20_CLK_PLL_P_OUT1, .present = true },
556	[tegra_clk_pll_p_out2] = { .dt_id = TEGRA20_CLK_PLL_P_OUT2, .present = true },
557	[tegra_clk_pll_p_out3] = { .dt_id = TEGRA20_CLK_PLL_P_OUT3, .present = true },
558	[tegra_clk_pll_p_out4] = { .dt_id = TEGRA20_CLK_PLL_P_OUT4, .present = true },
559	[tegra_clk_pll_p] = { .dt_id = TEGRA20_CLK_PLL_P, .present = true },
560	[tegra_clk_owr] = { .dt_id = TEGRA20_CLK_OWR, .present = true },
561	[tegra_clk_sbc1] = { .dt_id = TEGRA20_CLK_SBC1, .present = true },
562	[tegra_clk_sbc2] = { .dt_id = TEGRA20_CLK_SBC2, .present = true },
563	[tegra_clk_sbc3] = { .dt_id = TEGRA20_CLK_SBC3, .present = true },
564	[tegra_clk_sbc4] = { .dt_id = TEGRA20_CLK_SBC4, .present = true },
565	[tegra_clk_vde] = { .dt_id = TEGRA20_CLK_VDE, .present = true },
566	[tegra_clk_vi] = { .dt_id = TEGRA20_CLK_VI, .present = true },
567	[tegra_clk_epp] = { .dt_id = TEGRA20_CLK_EPP, .present = true },
568	[tegra_clk_mpe] = { .dt_id = TEGRA20_CLK_MPE, .present = true },
569	[tegra_clk_host1x] = { .dt_id = TEGRA20_CLK_HOST1X, .present = true },
570	[tegra_clk_gr2d] = { .dt_id = TEGRA20_CLK_GR2D, .present = true },
571	[tegra_clk_gr3d] = { .dt_id = TEGRA20_CLK_GR3D, .present = true },
572	[tegra_clk_ndflash] = { .dt_id = TEGRA20_CLK_NDFLASH, .present = true },
573	[tegra_clk_cve] = { .dt_id = TEGRA20_CLK_CVE, .present = true },
574	[tegra_clk_tvo] = { .dt_id = TEGRA20_CLK_TVO, .present = true },
575	[tegra_clk_tvdac] = { .dt_id = TEGRA20_CLK_TVDAC, .present = true },
576	[tegra_clk_vi_sensor] = { .dt_id = TEGRA20_CLK_VI_SENSOR, .present = true },
577	[tegra_clk_afi] = { .dt_id = TEGRA20_CLK_AFI, .present = true },
578};
579
580static unsigned long tegra20_clk_measure_input_freq(void)
581{
582	u32 osc_ctrl = readl_relaxed(clk_base + OSC_CTRL);
583	u32 auto_clk_control = osc_ctrl & OSC_CTRL_OSC_FREQ_MASK;
584	u32 pll_ref_div = osc_ctrl & OSC_CTRL_PLL_REF_DIV_MASK;
585	unsigned long input_freq;
586
587	switch (auto_clk_control) {
588	case OSC_CTRL_OSC_FREQ_12MHZ:
589		BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1);
590		input_freq = 12000000;
591		break;
592	case OSC_CTRL_OSC_FREQ_13MHZ:
593		BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1);
594		input_freq = 13000000;
595		break;
596	case OSC_CTRL_OSC_FREQ_19_2MHZ:
597		BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1);
598		input_freq = 19200000;
599		break;
600	case OSC_CTRL_OSC_FREQ_26MHZ:
601		BUG_ON(pll_ref_div != OSC_CTRL_PLL_REF_DIV_1);
602		input_freq = 26000000;
603		break;
604	default:
605		pr_err("Unexpected clock autodetect value %d",
606		       auto_clk_control);
607		BUG();
608		return 0;
609	}
610
611	return input_freq;
612}
613
614static unsigned int tegra20_get_pll_ref_div(void)
615{
616	u32 pll_ref_div = readl_relaxed(clk_base + OSC_CTRL) &
617		OSC_CTRL_PLL_REF_DIV_MASK;
618
619	switch (pll_ref_div) {
620	case OSC_CTRL_PLL_REF_DIV_1:
621		return 1;
622	case OSC_CTRL_PLL_REF_DIV_2:
623		return 2;
624	case OSC_CTRL_PLL_REF_DIV_4:
625		return 4;
626	default:
627		pr_err("Invalied pll ref divider %d\n", pll_ref_div);
628		BUG();
629	}
630	return 0;
631}
632
633static void tegra20_pll_init(void)
634{
635	struct clk *clk;
636
637	/* PLLC */
638	clk = tegra_clk_register_pll("pll_c", "pll_ref", clk_base, NULL, 0,
639			    &pll_c_params, NULL);
640	clks[TEGRA20_CLK_PLL_C] = clk;
641
642	/* PLLC_OUT1 */
643	clk = tegra_clk_register_divider("pll_c_out1_div", "pll_c",
644				clk_base + PLLC_OUT, 0, TEGRA_DIVIDER_ROUND_UP,
645				8, 8, 1, NULL);
646	clk = tegra_clk_register_pll_out("pll_c_out1", "pll_c_out1_div",
647				clk_base + PLLC_OUT, 1, 0, CLK_SET_RATE_PARENT,
648				0, NULL);
649	clks[TEGRA20_CLK_PLL_C_OUT1] = clk;
650
651	/* PLLM */
652	clk = tegra_clk_register_pll("pll_m", "pll_ref", clk_base, NULL,
653			    CLK_IGNORE_UNUSED | CLK_SET_RATE_GATE,
654			    &pll_m_params, NULL);
655	clks[TEGRA20_CLK_PLL_M] = clk;
656
657	/* PLLM_OUT1 */
658	clk = tegra_clk_register_divider("pll_m_out1_div", "pll_m",
659				clk_base + PLLM_OUT, 0, TEGRA_DIVIDER_ROUND_UP,
660				8, 8, 1, NULL);
661	clk = tegra_clk_register_pll_out("pll_m_out1", "pll_m_out1_div",
662				clk_base + PLLM_OUT, 1, 0, CLK_IGNORE_UNUSED |
663				CLK_SET_RATE_PARENT, 0, NULL);
664	clks[TEGRA20_CLK_PLL_M_OUT1] = clk;
665
666	/* PLLX */
667	clk = tegra_clk_register_pll("pll_x", "pll_ref", clk_base, NULL, 0,
668			    &pll_x_params, NULL);
669	clks[TEGRA20_CLK_PLL_X] = clk;
670
671	/* PLLU */
672	clk = tegra_clk_register_pll("pll_u", "pll_ref", clk_base, NULL, 0,
673			    &pll_u_params, NULL);
674	clks[TEGRA20_CLK_PLL_U] = clk;
675
676	/* PLLD */
677	clk = tegra_clk_register_pll("pll_d", "pll_ref", clk_base, NULL, 0,
678			    &pll_d_params, NULL);
679	clks[TEGRA20_CLK_PLL_D] = clk;
680
681	/* PLLD_OUT0 */
682	clk = clk_register_fixed_factor(NULL, "pll_d_out0", "pll_d",
683					CLK_SET_RATE_PARENT, 1, 2);
684	clks[TEGRA20_CLK_PLL_D_OUT0] = clk;
685
686	/* PLLA */
687	clk = tegra_clk_register_pll("pll_a", "pll_p_out1", clk_base, NULL, 0,
688			    &pll_a_params, NULL);
689	clks[TEGRA20_CLK_PLL_A] = clk;
690
691	/* PLLA_OUT0 */
692	clk = tegra_clk_register_divider("pll_a_out0_div", "pll_a",
693				clk_base + PLLA_OUT, 0, TEGRA_DIVIDER_ROUND_UP,
694				8, 8, 1, NULL);
695	clk = tegra_clk_register_pll_out("pll_a_out0", "pll_a_out0_div",
696				clk_base + PLLA_OUT, 1, 0, CLK_IGNORE_UNUSED |
697				CLK_SET_RATE_PARENT, 0, NULL);
698	clks[TEGRA20_CLK_PLL_A_OUT0] = clk;
699
700	/* PLLE */
701	clk = tegra_clk_register_plle("pll_e", "pll_ref", clk_base, pmc_base,
702			     0, &pll_e_params, NULL);
703	clks[TEGRA20_CLK_PLL_E] = clk;
704}
705
706static const char *cclk_parents[] = { "clk_m", "pll_c", "clk_32k", "pll_m",
707				      "pll_p", "pll_p_out4",
708				      "pll_p_out3", "clk_d", "pll_x" };
709static const char *sclk_parents[] = { "clk_m", "pll_c_out1", "pll_p_out4",
710				      "pll_p_out3", "pll_p_out2", "clk_d",
711				      "clk_32k", "pll_m_out1" };
712
713static void tegra20_super_clk_init(void)
714{
715	struct clk *clk;
716
717	/* CCLK */
718	clk = tegra_clk_register_super_mux("cclk", cclk_parents,
719			      ARRAY_SIZE(cclk_parents), CLK_SET_RATE_PARENT,
720			      clk_base + CCLK_BURST_POLICY, 0, 4, 0, 0, NULL);
721	clks[TEGRA20_CLK_CCLK] = clk;
722
723	/* SCLK */
724	clk = tegra_clk_register_super_mux("sclk", sclk_parents,
725			      ARRAY_SIZE(sclk_parents), CLK_SET_RATE_PARENT,
726			      clk_base + SCLK_BURST_POLICY, 0, 4, 0, 0, NULL);
727	clks[TEGRA20_CLK_SCLK] = clk;
728
729	/* twd */
730	clk = clk_register_fixed_factor(NULL, "twd", "cclk", 0, 1, 4);
731	clks[TEGRA20_CLK_TWD] = clk;
732}
733
734static const char *audio_parents[] = {"spdif_in", "i2s1", "i2s2", "unused",
735				      "pll_a_out0", "unused", "unused",
736				      "unused"};
737
738static void __init tegra20_audio_clk_init(void)
739{
740	struct clk *clk;
741
742	/* audio */
743	clk = clk_register_mux(NULL, "audio_mux", audio_parents,
744				ARRAY_SIZE(audio_parents),
745				CLK_SET_RATE_NO_REPARENT,
746				clk_base + AUDIO_SYNC_CLK, 0, 3, 0, NULL);
747	clk = clk_register_gate(NULL, "audio", "audio_mux", 0,
748				clk_base + AUDIO_SYNC_CLK, 4,
749				CLK_GATE_SET_TO_DISABLE, NULL);
750	clks[TEGRA20_CLK_AUDIO] = clk;
751
752	/* audio_2x */
753	clk = clk_register_fixed_factor(NULL, "audio_doubler", "audio",
754					CLK_SET_RATE_PARENT, 2, 1);
755	clk = tegra_clk_register_periph_gate("audio_2x", "audio_doubler",
756				    TEGRA_PERIPH_NO_RESET, clk_base,
757				    CLK_SET_RATE_PARENT, 89,
758				    periph_clk_enb_refcnt);
759	clks[TEGRA20_CLK_AUDIO_2X] = clk;
760
761}
762
763static const char *i2s1_parents[] = {"pll_a_out0", "audio_2x", "pll_p",
764				     "clk_m"};
765static const char *i2s2_parents[] = {"pll_a_out0", "audio_2x", "pll_p",
766				     "clk_m"};
767static const char *pwm_parents[] = {"pll_p", "pll_c", "audio", "clk_m",
768				    "clk_32k"};
769static const char *mux_pllpcm_clkm[] = {"pll_p", "pll_c", "pll_m", "clk_m"};
770static const char *mux_pllpdc_clkm[] = {"pll_p", "pll_d_out0", "pll_c",
771					"clk_m"};
772static const char *mux_pllmcp_clkm[] = {"pll_m", "pll_c", "pll_p", "clk_m"};
773
774static struct tegra_periph_init_data tegra_periph_clk_list[] = {
775	TEGRA_INIT_DATA_MUX("i2s1", i2s1_parents,     CLK_SOURCE_I2S1,   11, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_I2S1),
776	TEGRA_INIT_DATA_MUX("i2s2", i2s2_parents,     CLK_SOURCE_I2S2,   18, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_I2S2),
777	TEGRA_INIT_DATA_MUX("spi",   mux_pllpcm_clkm,   CLK_SOURCE_SPI,   43, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_SPI),
778	TEGRA_INIT_DATA_MUX("xio",   mux_pllpcm_clkm,   CLK_SOURCE_XIO,   45, 0, TEGRA20_CLK_XIO),
779	TEGRA_INIT_DATA_MUX("twc",   mux_pllpcm_clkm,   CLK_SOURCE_TWC,   16, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_TWC),
780	TEGRA_INIT_DATA_MUX("ide",   mux_pllpcm_clkm,   CLK_SOURCE_XIO,   25, 0, TEGRA20_CLK_IDE),
781	TEGRA_INIT_DATA_DIV16("dvc", mux_pllpcm_clkm,   CLK_SOURCE_DVC,   47, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_DVC),
782	TEGRA_INIT_DATA_DIV16("i2c1", mux_pllpcm_clkm,   CLK_SOURCE_I2C1,   12, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_I2C1),
783	TEGRA_INIT_DATA_DIV16("i2c2", mux_pllpcm_clkm,   CLK_SOURCE_I2C2,   54, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_I2C2),
784	TEGRA_INIT_DATA_DIV16("i2c3", mux_pllpcm_clkm,   CLK_SOURCE_I2C3,   67, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_I2C3),
785	TEGRA_INIT_DATA_MUX("hdmi", mux_pllpdc_clkm,   CLK_SOURCE_HDMI,   51, 0, TEGRA20_CLK_HDMI),
786	TEGRA_INIT_DATA("pwm", NULL, NULL, pwm_parents,     CLK_SOURCE_PWM,   28, 3, 0, 0, 8, 1, 0, 17, TEGRA_PERIPH_ON_APB, TEGRA20_CLK_PWM),
787};
788
789static struct tegra_periph_init_data tegra_periph_nodiv_clk_list[] = {
790	TEGRA_INIT_DATA_NODIV("uarta",	mux_pllpcm_clkm, CLK_SOURCE_UARTA, 30, 2, 6,   TEGRA_PERIPH_ON_APB, TEGRA20_CLK_UARTA),
791	TEGRA_INIT_DATA_NODIV("uartb",	mux_pllpcm_clkm, CLK_SOURCE_UARTB, 30, 2, 7,   TEGRA_PERIPH_ON_APB, TEGRA20_CLK_UARTB),
792	TEGRA_INIT_DATA_NODIV("uartc",	mux_pllpcm_clkm, CLK_SOURCE_UARTC, 30, 2, 55,  TEGRA_PERIPH_ON_APB, TEGRA20_CLK_UARTC),
793	TEGRA_INIT_DATA_NODIV("uartd",	mux_pllpcm_clkm, CLK_SOURCE_UARTD, 30, 2, 65,  TEGRA_PERIPH_ON_APB, TEGRA20_CLK_UARTD),
794	TEGRA_INIT_DATA_NODIV("uarte",	mux_pllpcm_clkm, CLK_SOURCE_UARTE, 30, 2, 66,  TEGRA_PERIPH_ON_APB, TEGRA20_CLK_UARTE),
795	TEGRA_INIT_DATA_NODIV("disp1",	mux_pllpdc_clkm, CLK_SOURCE_DISP1, 30, 2, 27,  0, TEGRA20_CLK_DISP1),
796	TEGRA_INIT_DATA_NODIV("disp2",	mux_pllpdc_clkm, CLK_SOURCE_DISP2, 30, 2, 26,  0, TEGRA20_CLK_DISP2),
797};
798
799static void __init tegra20_periph_clk_init(void)
800{
801	struct tegra_periph_init_data *data;
802	struct clk *clk;
803	int i;
804
805	/* ac97 */
806	clk = tegra_clk_register_periph_gate("ac97", "pll_a_out0",
807				    TEGRA_PERIPH_ON_APB,
808				    clk_base, 0, 3, periph_clk_enb_refcnt);
809	clks[TEGRA20_CLK_AC97] = clk;
810
811	/* apbdma */
812	clk = tegra_clk_register_periph_gate("apbdma", "pclk", 0, clk_base,
813				    0, 34, periph_clk_enb_refcnt);
814	clks[TEGRA20_CLK_APBDMA] = clk;
815
816	/* emc */
817	clk = clk_register_mux(NULL, "emc_mux", mux_pllmcp_clkm,
818			       ARRAY_SIZE(mux_pllmcp_clkm),
819			       CLK_SET_RATE_NO_REPARENT,
820			       clk_base + CLK_SOURCE_EMC,
821			       30, 2, 0, NULL);
822	clk = tegra_clk_register_periph_gate("emc", "emc_mux", 0, clk_base, 0,
823				    57, periph_clk_enb_refcnt);
824	clks[TEGRA20_CLK_EMC] = clk;
825
826	/* dsi */
827	clk = tegra_clk_register_periph_gate("dsi", "pll_d", 0, clk_base, 0,
828				    48, periph_clk_enb_refcnt);
829	clk_register_clkdev(clk, NULL, "dsi");
830	clks[TEGRA20_CLK_DSI] = clk;
831
832	/* pex */
833	clk = tegra_clk_register_periph_gate("pex", "clk_m", 0, clk_base, 0, 70,
834				    periph_clk_enb_refcnt);
835	clks[TEGRA20_CLK_PEX] = clk;
836
837	/* pcie_xclk */
838	clk = tegra_clk_register_periph_gate("pcie_xclk", "clk_m", 0, clk_base,
839				    0, 74, periph_clk_enb_refcnt);
840	clks[TEGRA20_CLK_PCIE_XCLK] = clk;
841
842	/* cdev1 */
843	clk = clk_register_fixed_rate(NULL, "cdev1_fixed", NULL, CLK_IS_ROOT,
844				      26000000);
845	clk = tegra_clk_register_periph_gate("cdev1", "cdev1_fixed", 0,
846				    clk_base, 0, 94, periph_clk_enb_refcnt);
847	clks[TEGRA20_CLK_CDEV1] = clk;
848
849	/* cdev2 */
850	clk = clk_register_fixed_rate(NULL, "cdev2_fixed", NULL, CLK_IS_ROOT,
851				      26000000);
852	clk = tegra_clk_register_periph_gate("cdev2", "cdev2_fixed", 0,
853				    clk_base, 0, 93, periph_clk_enb_refcnt);
854	clks[TEGRA20_CLK_CDEV2] = clk;
855
856	for (i = 0; i < ARRAY_SIZE(tegra_periph_clk_list); i++) {
857		data = &tegra_periph_clk_list[i];
858		clk = tegra_clk_register_periph(data->name, data->p.parent_names,
859				data->num_parents, &data->periph,
860				clk_base, data->offset, data->flags);
861		clks[data->clk_id] = clk;
862	}
863
864	for (i = 0; i < ARRAY_SIZE(tegra_periph_nodiv_clk_list); i++) {
865		data = &tegra_periph_nodiv_clk_list[i];
866		clk = tegra_clk_register_periph_nodiv(data->name,
867					data->p.parent_names,
868					data->num_parents, &data->periph,
869					clk_base, data->offset);
870		clks[data->clk_id] = clk;
871	}
872
873	tegra_periph_clk_init(clk_base, pmc_base, tegra20_clks, &pll_p_params);
874}
875
876static void __init tegra20_osc_clk_init(void)
877{
878	struct clk *clk;
879	unsigned long input_freq;
880	unsigned int pll_ref_div;
881
882	input_freq = tegra20_clk_measure_input_freq();
883
884	/* clk_m */
885	clk = clk_register_fixed_rate(NULL, "clk_m", NULL, CLK_IS_ROOT |
886				      CLK_IGNORE_UNUSED, input_freq);
887	clks[TEGRA20_CLK_CLK_M] = clk;
888
889	/* pll_ref */
890	pll_ref_div = tegra20_get_pll_ref_div();
891	clk = clk_register_fixed_factor(NULL, "pll_ref", "clk_m",
892					CLK_SET_RATE_PARENT, 1, pll_ref_div);
893	clks[TEGRA20_CLK_PLL_REF] = clk;
894}
895
896/* Tegra20 CPU clock and reset control functions */
897static void tegra20_wait_cpu_in_reset(u32 cpu)
898{
899	unsigned int reg;
900
901	do {
902		reg = readl(clk_base +
903			    TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_SET);
904		cpu_relax();
905	} while (!(reg & (1 << cpu)));	/* check CPU been reset or not */
906
907	return;
908}
909
910static void tegra20_put_cpu_in_reset(u32 cpu)
911{
912	writel(CPU_RESET(cpu),
913	       clk_base + TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_SET);
914	dmb();
915}
916
917static void tegra20_cpu_out_of_reset(u32 cpu)
918{
919	writel(CPU_RESET(cpu),
920	       clk_base + TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_CLR);
921	wmb();
922}
923
924static void tegra20_enable_cpu_clock(u32 cpu)
925{
926	unsigned int reg;
927
928	reg = readl(clk_base + TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX);
929	writel(reg & ~CPU_CLOCK(cpu),
930	       clk_base + TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX);
931	barrier();
932	reg = readl(clk_base + TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX);
933}
934
935static void tegra20_disable_cpu_clock(u32 cpu)
936{
937	unsigned int reg;
938
939	reg = readl(clk_base + TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX);
940	writel(reg | CPU_CLOCK(cpu),
941	       clk_base + TEGRA_CLK_RST_CONTROLLER_CLK_CPU_CMPLX);
942}
943
944#ifdef CONFIG_PM_SLEEP
945static bool tegra20_cpu_rail_off_ready(void)
946{
947	unsigned int cpu_rst_status;
948
949	cpu_rst_status = readl(clk_base +
950			       TEGRA_CLK_RST_CONTROLLER_RST_CPU_CMPLX_SET);
951
952	return !!(cpu_rst_status & 0x2);
953}
954
955static void tegra20_cpu_clock_suspend(void)
956{
957	/* switch coresite to clk_m, save off original source */
958	tegra20_cpu_clk_sctx.clk_csite_src =
959				readl(clk_base + CLK_SOURCE_CSITE);
960	writel(3<<30, clk_base + CLK_SOURCE_CSITE);
961
962	tegra20_cpu_clk_sctx.cpu_burst =
963				readl(clk_base + CCLK_BURST_POLICY);
964	tegra20_cpu_clk_sctx.pllx_base =
965				readl(clk_base + PLLX_BASE);
966	tegra20_cpu_clk_sctx.pllx_misc =
967				readl(clk_base + PLLX_MISC);
968	tegra20_cpu_clk_sctx.cclk_divider =
969				readl(clk_base + SUPER_CCLK_DIVIDER);
970}
971
972static void tegra20_cpu_clock_resume(void)
973{
974	unsigned int reg, policy;
975
976	/* Is CPU complex already running on PLLX? */
977	reg = readl(clk_base + CCLK_BURST_POLICY);
978	policy = (reg >> CCLK_BURST_POLICY_SHIFT) & 0xF;
979
980	if (policy == CCLK_IDLE_POLICY)
981		reg = (reg >> CCLK_IDLE_POLICY_SHIFT) & 0xF;
982	else if (policy == CCLK_RUN_POLICY)
983		reg = (reg >> CCLK_RUN_POLICY_SHIFT) & 0xF;
984	else
985		BUG();
986
987	if (reg != CCLK_BURST_POLICY_PLLX) {
988		/* restore PLLX settings if CPU is on different PLL */
989		writel(tegra20_cpu_clk_sctx.pllx_misc,
990					clk_base + PLLX_MISC);
991		writel(tegra20_cpu_clk_sctx.pllx_base,
992					clk_base + PLLX_BASE);
993
994		/* wait for PLL stabilization if PLLX was enabled */
995		if (tegra20_cpu_clk_sctx.pllx_base & (1 << 30))
996			udelay(300);
997	}
998
999	/*
1000	 * Restore original burst policy setting for calls resulting from CPU
1001	 * LP2 in idle or system suspend.
1002	 */
1003	writel(tegra20_cpu_clk_sctx.cclk_divider,
1004					clk_base + SUPER_CCLK_DIVIDER);
1005	writel(tegra20_cpu_clk_sctx.cpu_burst,
1006					clk_base + CCLK_BURST_POLICY);
1007
1008	writel(tegra20_cpu_clk_sctx.clk_csite_src,
1009					clk_base + CLK_SOURCE_CSITE);
1010}
1011#endif
1012
1013static struct tegra_cpu_car_ops tegra20_cpu_car_ops = {
1014	.wait_for_reset	= tegra20_wait_cpu_in_reset,
1015	.put_in_reset	= tegra20_put_cpu_in_reset,
1016	.out_of_reset	= tegra20_cpu_out_of_reset,
1017	.enable_clock	= tegra20_enable_cpu_clock,
1018	.disable_clock	= tegra20_disable_cpu_clock,
1019#ifdef CONFIG_PM_SLEEP
1020	.rail_off_ready = tegra20_cpu_rail_off_ready,
1021	.suspend	= tegra20_cpu_clock_suspend,
1022	.resume		= tegra20_cpu_clock_resume,
1023#endif
1024};
1025
1026static struct tegra_clk_init_table init_table[] __initdata = {
1027	{TEGRA20_CLK_PLL_P, TEGRA20_CLK_CLK_MAX, 216000000, 1},
1028	{TEGRA20_CLK_PLL_P_OUT1, TEGRA20_CLK_CLK_MAX, 28800000, 1},
1029	{TEGRA20_CLK_PLL_P_OUT2, TEGRA20_CLK_CLK_MAX, 48000000, 1},
1030	{TEGRA20_CLK_PLL_P_OUT3, TEGRA20_CLK_CLK_MAX, 72000000, 1},
1031	{TEGRA20_CLK_PLL_P_OUT4, TEGRA20_CLK_CLK_MAX, 24000000, 1},
1032	{TEGRA20_CLK_PLL_C, TEGRA20_CLK_CLK_MAX, 600000000, 1},
1033	{TEGRA20_CLK_PLL_C_OUT1, TEGRA20_CLK_CLK_MAX, 120000000, 1},
1034	{TEGRA20_CLK_SCLK, TEGRA20_CLK_PLL_C_OUT1, 0, 1},
1035	{TEGRA20_CLK_HCLK, TEGRA20_CLK_CLK_MAX, 0, 1},
1036	{TEGRA20_CLK_PCLK, TEGRA20_CLK_CLK_MAX, 60000000, 1},
1037	{TEGRA20_CLK_CSITE, TEGRA20_CLK_CLK_MAX, 0, 1},
1038	{TEGRA20_CLK_EMC, TEGRA20_CLK_CLK_MAX, 0, 1},
1039	{TEGRA20_CLK_CCLK, TEGRA20_CLK_CLK_MAX, 0, 1},
1040	{TEGRA20_CLK_UARTA, TEGRA20_CLK_PLL_P, 0, 0},
1041	{TEGRA20_CLK_UARTB, TEGRA20_CLK_PLL_P, 0, 0},
1042	{TEGRA20_CLK_UARTC, TEGRA20_CLK_PLL_P, 0, 0},
1043	{TEGRA20_CLK_UARTD, TEGRA20_CLK_PLL_P, 0, 0},
1044	{TEGRA20_CLK_UARTE, TEGRA20_CLK_PLL_P, 0, 0},
1045	{TEGRA20_CLK_PLL_A, TEGRA20_CLK_CLK_MAX, 56448000, 1},
1046	{TEGRA20_CLK_PLL_A_OUT0, TEGRA20_CLK_CLK_MAX, 11289600, 1},
1047	{TEGRA20_CLK_CDEV1, TEGRA20_CLK_CLK_MAX, 0, 1},
1048	{TEGRA20_CLK_BLINK, TEGRA20_CLK_CLK_MAX, 32768, 1},
1049	{TEGRA20_CLK_I2S1, TEGRA20_CLK_PLL_A_OUT0, 11289600, 0},
1050	{TEGRA20_CLK_I2S2, TEGRA20_CLK_PLL_A_OUT0, 11289600, 0},
1051	{TEGRA20_CLK_SDMMC1, TEGRA20_CLK_PLL_P, 48000000, 0},
1052	{TEGRA20_CLK_SDMMC3, TEGRA20_CLK_PLL_P, 48000000, 0},
1053	{TEGRA20_CLK_SDMMC4, TEGRA20_CLK_PLL_P, 48000000, 0},
1054	{TEGRA20_CLK_SPI, TEGRA20_CLK_PLL_P, 20000000, 0},
1055	{TEGRA20_CLK_SBC1, TEGRA20_CLK_PLL_P, 100000000, 0},
1056	{TEGRA20_CLK_SBC2, TEGRA20_CLK_PLL_P, 100000000, 0},
1057	{TEGRA20_CLK_SBC3, TEGRA20_CLK_PLL_P, 100000000, 0},
1058	{TEGRA20_CLK_SBC4, TEGRA20_CLK_PLL_P, 100000000, 0},
1059	{TEGRA20_CLK_HOST1X, TEGRA20_CLK_PLL_C, 150000000, 0},
1060	{TEGRA20_CLK_DISP1, TEGRA20_CLK_PLL_P, 600000000, 0},
1061	{TEGRA20_CLK_DISP2, TEGRA20_CLK_PLL_P, 600000000, 0},
1062	{TEGRA20_CLK_GR2D, TEGRA20_CLK_PLL_C, 300000000, 0},
1063	{TEGRA20_CLK_GR3D, TEGRA20_CLK_PLL_C, 300000000, 0},
1064	{TEGRA20_CLK_CLK_MAX, TEGRA20_CLK_CLK_MAX, 0, 0}, /* This MUST be the last entry */
1065};
1066
1067static void __init tegra20_clock_apply_init_table(void)
1068{
1069	tegra_init_from_table(init_table, clks, TEGRA20_CLK_CLK_MAX);
1070}
1071
1072/*
1073 * Some clocks may be used by different drivers depending on the board
1074 * configuration.  List those here to register them twice in the clock lookup
1075 * table under two names.
1076 */
1077static struct tegra_clk_duplicate tegra_clk_duplicates[] = {
1078	TEGRA_CLK_DUPLICATE(TEGRA20_CLK_USBD,   "utmip-pad",    NULL),
1079	TEGRA_CLK_DUPLICATE(TEGRA20_CLK_USBD,   "tegra-ehci.0", NULL),
1080	TEGRA_CLK_DUPLICATE(TEGRA20_CLK_USBD,   "tegra-otg",    NULL),
1081	TEGRA_CLK_DUPLICATE(TEGRA20_CLK_CCLK,   NULL,           "cpu"),
1082	TEGRA_CLK_DUPLICATE(TEGRA20_CLK_CLK_MAX, NULL, NULL), /* Must be the last entry */
1083};
1084
1085static const struct of_device_id pmc_match[] __initconst = {
1086	{ .compatible = "nvidia,tegra20-pmc" },
1087	{},
1088};
1089
1090static void __init tegra20_clock_init(struct device_node *np)
1091{
1092	struct device_node *node;
1093
1094	clk_base = of_iomap(np, 0);
1095	if (!clk_base) {
1096		pr_err("Can't map CAR registers\n");
1097		BUG();
1098	}
1099
1100	node = of_find_matching_node(NULL, pmc_match);
1101	if (!node) {
1102		pr_err("Failed to find pmc node\n");
1103		BUG();
1104	}
1105
1106	pmc_base = of_iomap(node, 0);
1107	if (!pmc_base) {
1108		pr_err("Can't map pmc registers\n");
1109		BUG();
1110	}
1111
1112	clks = tegra_clk_init(clk_base, TEGRA20_CLK_CLK_MAX,
1113				TEGRA20_CLK_PERIPH_BANKS);
1114	if (!clks)
1115		return;
1116
1117	tegra20_osc_clk_init();
1118	tegra_fixed_clk_init(tegra20_clks);
1119	tegra20_pll_init();
1120	tegra20_super_clk_init();
1121	tegra_super_clk_gen4_init(clk_base, pmc_base, tegra20_clks, NULL);
1122	tegra20_periph_clk_init();
1123	tegra20_audio_clk_init();
1124	tegra_pmc_clk_init(pmc_base, tegra20_clks);
1125
1126	tegra_init_dup_clks(tegra_clk_duplicates, clks, TEGRA20_CLK_CLK_MAX);
1127
1128	tegra_add_of_provider(np);
1129	tegra_register_devclks(devclks, ARRAY_SIZE(devclks));
1130
1131	tegra_clk_apply_init_table = tegra20_clock_apply_init_table;
1132
1133	tegra_cpu_car_ops = &tegra20_cpu_car_ops;
1134}
1135CLK_OF_DECLARE(tegra20, "nvidia,tegra20-car", tegra20_clock_init);
1136