Blame view

drivers/cpufreq/exynos4210-cpufreq.c 4.58 KB
f7d770790   Kukjin Kim   [CPUFREQ] Move AR...
1
  /*
7d30e8b38   Kukjin Kim   ARM: EXYNOS4: Add...
2
   * Copyright (c) 2010-2011 Samsung Electronics Co., Ltd.
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
3
4
   *		http://www.samsung.com
   *
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
5
   * EXYNOS4210 - CPU frequency scaling support
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
6
7
8
9
10
   *
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of the GNU General Public License version 2 as
   * published by the Free Software Foundation.
  */
6c523c614   Jaecheol Lee   [CPUFREQ] EXYNOS:...
11
  #include <linux/module.h>
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
12
13
14
15
16
  #include <linux/kernel.h>
  #include <linux/err.h>
  #include <linux/clk.h>
  #include <linux/io.h>
  #include <linux/slab.h>
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
17
  #include <linux/cpufreq.h>
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
18
19
  #include <linux/of.h>
  #include <linux/of_address.h>
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
20

c4aaa2957   Kukjin Kim   cpufreq: exynos: ...
21
  #include "exynos-cpufreq.h"
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
22

f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
23
24
25
26
  static struct clk *cpu_clk;
  static struct clk *moutcore;
  static struct clk *mout_mpll;
  static struct clk *mout_apll;
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
27
  static struct exynos_dvfs_info *cpufreq;
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
28

9d0554fff   Jonghwan Choi   cpufreq: exynos: ...
29
  static unsigned int exynos4210_volt_table[] = {
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
30
  	1250000, 1150000, 1050000, 975000, 950000,
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
31
  };
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
32
  static struct cpufreq_frequency_table exynos4210_freq_table[] = {
7f4b04614   Viresh Kumar   cpufreq: create a...
33
34
35
36
37
38
  	{0, L0, 1200 * 1000},
  	{0, L1, 1000 * 1000},
  	{0, L2,  800 * 1000},
  	{0, L3,  500 * 1000},
  	{0, L4,  200 * 1000},
  	{0, 0, CPUFREQ_TABLE_END},
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
39
  };
9d0554fff   Jonghwan Choi   cpufreq: exynos: ...
40
  static struct apll_freq apll_freq_4210[] = {
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
41
  	/*
9d0554fff   Jonghwan Choi   cpufreq: exynos: ...
42
43
44
45
46
  	 * values:
  	 * freq
  	 * clock divider for CORE, COREM0, COREM1, PERIPH, ATB, PCLK_DBG, APLL, RESERVED
  	 * clock divider for COPY, HPM, RESERVED
  	 * PLL M, P, S
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
47
  	 */
9d0554fff   Jonghwan Choi   cpufreq: exynos: ...
48
49
50
51
52
  	APLL_FREQ(1200, 0, 3, 7, 3, 4, 1, 7, 0, 5, 0, 0, 150, 3, 1),
  	APLL_FREQ(1000, 0, 3, 7, 3, 4, 1, 7, 0, 4, 0, 0, 250, 6, 1),
  	APLL_FREQ(800,  0, 3, 7, 3, 3, 1, 7, 0, 3, 0, 0, 200, 6, 1),
  	APLL_FREQ(500,  0, 3, 7, 3, 3, 1, 7, 0, 3, 0, 0, 250, 6, 2),
  	APLL_FREQ(200,  0, 1, 3, 1, 3, 1, 0, 0, 3, 0, 0, 200, 6, 3),
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
53
  };
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
54
  static void exynos4210_set_clkdiv(unsigned int div_index)
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
55
56
57
58
  {
  	unsigned int tmp;
  
  	/* Change Divider - CPU0 */
9d0554fff   Jonghwan Choi   cpufreq: exynos: ...
59
  	tmp = apll_freq_4210[div_index].clk_div_cpu0;
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
60

4c8d81934   Tomasz Figa   cpufreq: exynos: ...
61
  	__raw_writel(tmp, cpufreq->cmu_regs + EXYNOS4_CLKDIV_CPU);
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
62
63
  
  	do {
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
64
  		tmp = __raw_readl(cpufreq->cmu_regs + EXYNOS4_CLKDIV_STATCPU);
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
65
  	} while (tmp & 0x1111111);
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
66
  	/* Change Divider - CPU1 */
9d0554fff   Jonghwan Choi   cpufreq: exynos: ...
67
  	tmp = apll_freq_4210[div_index].clk_div_cpu1;
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
68

4c8d81934   Tomasz Figa   cpufreq: exynos: ...
69
  	__raw_writel(tmp, cpufreq->cmu_regs + EXYNOS4_CLKDIV_CPU1);
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
70
71
  
  	do {
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
72
  		tmp = __raw_readl(cpufreq->cmu_regs + EXYNOS4_CLKDIV_STATCPU1);
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
73
  	} while (tmp & 0x11);
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
74
  }
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
75
  static void exynos4210_set_apll(unsigned int index)
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
76
  {
7ad65d592   Lukasz Majewski   cpufreq: exynos42...
77
  	unsigned int tmp, freq = apll_freq_4210[index].freq;
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
78

7ad65d592   Lukasz Majewski   cpufreq: exynos42...
79
  	/* MUX_CORE_SEL = MPLL, ARMCLK uses MPLL for lock time */
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
80
81
82
  	clk_set_parent(moutcore, mout_mpll);
  
  	do {
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
83
  		tmp = (__raw_readl(cpufreq->cmu_regs + EXYNOS4_CLKMUX_STATCPU)
09cee1ab0   Kukjin Kim   [CPUFREQ] EXYNOS4...
84
  			>> EXYNOS4_CLKSRC_CPU_MUXCORE_SHIFT);
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
85
86
  		tmp &= 0x7;
  	} while (tmp != 0x2);
7ad65d592   Lukasz Majewski   cpufreq: exynos42...
87
  	clk_set_rate(mout_apll, freq * 1000);
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
88

7ad65d592   Lukasz Majewski   cpufreq: exynos42...
89
  	/* MUX_CORE_SEL = APLL */
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
90
91
92
  	clk_set_parent(moutcore, mout_apll);
  
  	do {
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
93
  		tmp = __raw_readl(cpufreq->cmu_regs + EXYNOS4_CLKMUX_STATCPU);
09cee1ab0   Kukjin Kim   [CPUFREQ] EXYNOS4...
94
95
  		tmp &= EXYNOS4_CLKMUX_STATCPU_MUXCORE_MASK;
  	} while (tmp != (0x1 << EXYNOS4_CLKSRC_CPU_MUXCORE_SHIFT));
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
96
  }
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
97
98
  static void exynos4210_set_frequency(unsigned int old_index,
  				     unsigned int new_index)
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
99
  {
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
100
  	if (old_index > new_index) {
7ad65d592   Lukasz Majewski   cpufreq: exynos42...
101
102
  		exynos4210_set_clkdiv(new_index);
  		exynos4210_set_apll(new_index);
27f805dcb   Jaecheol Lee   [CPUFREQ] EXYNOS4...
103
  	} else if (old_index < new_index) {
7ad65d592   Lukasz Majewski   cpufreq: exynos42...
104
105
  		exynos4210_set_apll(new_index);
  		exynos4210_set_clkdiv(new_index);
bf5ce054f   Sangwook Ju   ARM: S5PV310: Upd...
106
107
  	}
  }
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
108
  int exynos4210_cpufreq_init(struct exynos_dvfs_info *info)
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
109
  {
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
110
  	struct device_node *np;
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
111
  	unsigned long rate;
27f805dcb   Jaecheol Lee   [CPUFREQ] EXYNOS4...
112

4c8d81934   Tomasz Figa   cpufreq: exynos: ...
113
114
115
116
117
118
  	/*
  	 * HACK: This is a temporary workaround to get access to clock
  	 * controller registers directly and remove static mappings and
  	 * dependencies on platform headers. It is necessary to enable
  	 * Exynos multi-platform support and will be removed together with
  	 * this whole driver as soon as Exynos gets migrated to use
bbcf07196   Viresh Kumar   cpufreq: cpu0: re...
119
  	 * cpufreq-dt driver.
4c8d81934   Tomasz Figa   cpufreq: exynos: ...
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
  	 */
  	np = of_find_compatible_node(NULL, NULL, "samsung,exynos4210-clock");
  	if (!np) {
  		pr_err("%s: failed to find clock controller DT node
  ",
  			__func__);
  		return -ENODEV;
  	}
  
  	info->cmu_regs = of_iomap(np, 0);
  	if (!info->cmu_regs) {
  		pr_err("%s: failed to map CMU registers
  ", __func__);
  		return -EFAULT;
  	}
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
135
136
137
138
139
140
  	cpu_clk = clk_get(NULL, "armclk");
  	if (IS_ERR(cpu_clk))
  		return PTR_ERR(cpu_clk);
  
  	moutcore = clk_get(NULL, "moutcore");
  	if (IS_ERR(moutcore))
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
141
  		goto err_moutcore;
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
142
143
144
  
  	mout_mpll = clk_get(NULL, "mout_mpll");
  	if (IS_ERR(mout_mpll))
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
145
146
147
  		goto err_mout_mpll;
  
  	rate = clk_get_rate(mout_mpll) / 1000;
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
148
149
150
  
  	mout_apll = clk_get(NULL, "mout_apll");
  	if (IS_ERR(mout_apll))
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
151
  		goto err_mout_apll;
0073f538c   MyungJoo Ham   [CPUFREQ] ARM Exy...
152

a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
153
  	info->mpll_freq_khz = rate;
9d0554fff   Jonghwan Choi   cpufreq: exynos: ...
154
  	/* 800Mhz */
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
155
  	info->pll_safe_idx = L2;
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
156
157
158
159
  	info->cpu_clk = cpu_clk;
  	info->volt_table = exynos4210_volt_table;
  	info->freq_table = exynos4210_freq_table;
  	info->set_freq = exynos4210_set_frequency;
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
160

4c8d81934   Tomasz Figa   cpufreq: exynos: ...
161
  	cpufreq = info;
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
162
  	return 0;
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
163

a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
164
  err_mout_apll:
184cddd1e   Jonghwan Choi   cpufreq: exynos: ...
165
  	clk_put(mout_mpll);
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
166
  err_mout_mpll:
184cddd1e   Jonghwan Choi   cpufreq: exynos: ...
167
  	clk_put(moutcore);
a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
168
  err_moutcore:
184cddd1e   Jonghwan Choi   cpufreq: exynos: ...
169
  	clk_put(cpu_clk);
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
170

a125a17fa   Jaecheol Lee   [CPUFREQ] EXYNOS:...
171
172
  	pr_debug("%s: failed initialization
  ", __func__);
f40f91fef   Sunyoung Kang   ARM: S5PV310: Add...
173
174
  	return -EINVAL;
  }