Linux Audio

Check our new training course

Loading...
v3.15
 
  1/*
  2 * Copyright 2009 Wolfson Microelectronics plc
  3 *
  4 * S3C64xx CPUfreq Support
  5 *
  6 * This program is free software; you can redistribute it and/or modify
  7 * it under the terms of the GNU General Public License version 2 as
  8 * published by the Free Software Foundation.
  9 */
 10
 11#define pr_fmt(fmt) "cpufreq: " fmt
 12
 13#include <linux/kernel.h>
 14#include <linux/types.h>
 15#include <linux/init.h>
 16#include <linux/cpufreq.h>
 17#include <linux/clk.h>
 18#include <linux/err.h>
 19#include <linux/regulator/consumer.h>
 20#include <linux/module.h>
 21
 22static struct regulator *vddarm;
 23static unsigned long regulator_latency;
 24
 25#ifdef CONFIG_CPU_S3C6410
 26struct s3c64xx_dvfs {
 27	unsigned int vddarm_min;
 28	unsigned int vddarm_max;
 29};
 30
 
 31static struct s3c64xx_dvfs s3c64xx_dvfs_table[] = {
 32	[0] = { 1000000, 1150000 },
 33	[1] = { 1050000, 1150000 },
 34	[2] = { 1100000, 1150000 },
 35	[3] = { 1200000, 1350000 },
 36	[4] = { 1300000, 1350000 },
 37};
 
 38
 39static struct cpufreq_frequency_table s3c64xx_freq_table[] = {
 40	{ 0, 0,  66000 },
 41	{ 0, 0, 100000 },
 42	{ 0, 0, 133000 },
 43	{ 0, 1, 200000 },
 44	{ 0, 1, 222000 },
 45	{ 0, 1, 266000 },
 46	{ 0, 2, 333000 },
 47	{ 0, 2, 400000 },
 48	{ 0, 2, 532000 },
 49	{ 0, 2, 533000 },
 50	{ 0, 3, 667000 },
 51	{ 0, 4, 800000 },
 52	{ 0, 0, CPUFREQ_TABLE_END },
 53};
 54#endif
 55
 56static int s3c64xx_cpufreq_set_target(struct cpufreq_policy *policy,
 57				      unsigned int index)
 58{
 59	struct s3c64xx_dvfs *dvfs;
 60	unsigned int old_freq, new_freq;
 61	int ret;
 62
 
 
 
 
 63	old_freq = clk_get_rate(policy->clk) / 1000;
 64	new_freq = s3c64xx_freq_table[index].frequency;
 65	dvfs = &s3c64xx_dvfs_table[s3c64xx_freq_table[index].driver_data];
 66
 67#ifdef CONFIG_REGULATOR
 68	if (vddarm && new_freq > old_freq) {
 69		ret = regulator_set_voltage(vddarm,
 70					    dvfs->vddarm_min,
 71					    dvfs->vddarm_max);
 72		if (ret != 0) {
 73			pr_err("Failed to set VDDARM for %dkHz: %d\n",
 74			       new_freq, ret);
 75			return ret;
 76		}
 77	}
 78#endif
 79
 80	ret = clk_set_rate(policy->clk, new_freq * 1000);
 81	if (ret < 0) {
 82		pr_err("Failed to set rate %dkHz: %d\n",
 83		       new_freq, ret);
 84		return ret;
 85	}
 86
 87#ifdef CONFIG_REGULATOR
 88	if (vddarm && new_freq < old_freq) {
 89		ret = regulator_set_voltage(vddarm,
 90					    dvfs->vddarm_min,
 91					    dvfs->vddarm_max);
 92		if (ret != 0) {
 93			pr_err("Failed to set VDDARM for %dkHz: %d\n",
 94			       new_freq, ret);
 95			if (clk_set_rate(policy->clk, old_freq * 1000) < 0)
 96				pr_err("Failed to restore original clock rate\n");
 97
 98			return ret;
 99		}
100	}
101#endif
102
103	pr_debug("Set actual frequency %lukHz\n",
104		 clk_get_rate(policy->clk) / 1000);
105
106	return 0;
107}
108
109#ifdef CONFIG_REGULATOR
110static void __init s3c64xx_cpufreq_config_regulator(void)
111{
112	int count, v, i, found;
113	struct cpufreq_frequency_table *freq;
114	struct s3c64xx_dvfs *dvfs;
115
116	count = regulator_count_voltages(vddarm);
117	if (count < 0) {
118		pr_err("Unable to check supported voltages\n");
119	}
120
121	freq = s3c64xx_freq_table;
122	while (count > 0 && freq->frequency != CPUFREQ_TABLE_END) {
123		if (freq->frequency == CPUFREQ_ENTRY_INVALID)
124			continue;
125
 
126		dvfs = &s3c64xx_dvfs_table[freq->driver_data];
127		found = 0;
128
129		for (i = 0; i < count; i++) {
130			v = regulator_list_voltage(vddarm, i);
131			if (v >= dvfs->vddarm_min && v <= dvfs->vddarm_max)
132				found = 1;
133		}
134
135		if (!found) {
136			pr_debug("%dkHz unsupported by regulator\n",
137				 freq->frequency);
138			freq->frequency = CPUFREQ_ENTRY_INVALID;
139		}
140
141		freq++;
142	}
143
 
144	/* Guess based on having to do an I2C/SPI write; in future we
145	 * will be able to query the regulator performance here. */
146	regulator_latency = 1 * 1000 * 1000;
147}
148#endif
149
150static int s3c64xx_cpufreq_driver_init(struct cpufreq_policy *policy)
151{
152	int ret;
153	struct cpufreq_frequency_table *freq;
154
155	if (policy->cpu != 0)
156		return -EINVAL;
157
158	if (s3c64xx_freq_table == NULL) {
159		pr_err("No frequency information for this CPU\n");
160		return -ENODEV;
161	}
162
163	policy->clk = clk_get(NULL, "armclk");
164	if (IS_ERR(policy->clk)) {
165		pr_err("Unable to obtain ARMCLK: %ld\n",
166		       PTR_ERR(policy->clk));
167		return PTR_ERR(policy->clk);
168	}
169
170#ifdef CONFIG_REGULATOR
171	vddarm = regulator_get(NULL, "vddarm");
172	if (IS_ERR(vddarm)) {
173		ret = PTR_ERR(vddarm);
174		pr_err("Failed to obtain VDDARM: %d\n", ret);
175		pr_err("Only frequency scaling available\n");
176		vddarm = NULL;
177	} else {
178		s3c64xx_cpufreq_config_regulator();
179	}
180#endif
181
182	freq = s3c64xx_freq_table;
183	while (freq->frequency != CPUFREQ_TABLE_END) {
184		unsigned long r;
185
186		/* Check for frequencies we can generate */
187		r = clk_round_rate(policy->clk, freq->frequency * 1000);
188		r /= 1000;
189		if (r != freq->frequency) {
190			pr_debug("%dkHz unsupported by clock\n",
191				 freq->frequency);
192			freq->frequency = CPUFREQ_ENTRY_INVALID;
193		}
194
195		/* If we have no regulator then assume startup
196		 * frequency is the maximum we can support. */
197		if (!vddarm && freq->frequency > clk_get_rate(policy->clk) / 1000)
198			freq->frequency = CPUFREQ_ENTRY_INVALID;
199
200		freq++;
201	}
202
203	/* Datasheet says PLL stabalisation time (if we were to use
204	 * the PLLs, which we don't currently) is ~300us worst case,
205	 * but add some fudge.
206	 */
207	ret = cpufreq_generic_init(policy, s3c64xx_freq_table,
208			(500 * 1000) + regulator_latency);
209	if (ret != 0) {
210		pr_err("Failed to configure frequency table: %d\n",
211		       ret);
212		regulator_put(vddarm);
213		clk_put(policy->clk);
214	}
215
216	return ret;
217}
218
219static struct cpufreq_driver s3c64xx_cpufreq_driver = {
220	.flags		= CPUFREQ_NEED_INITIAL_FREQ_CHECK,
221	.verify		= cpufreq_generic_frequency_table_verify,
222	.target_index	= s3c64xx_cpufreq_set_target,
223	.get		= cpufreq_generic_get,
224	.init		= s3c64xx_cpufreq_driver_init,
225	.name		= "s3c",
226};
227
228static int __init s3c64xx_cpufreq_init(void)
229{
230	return cpufreq_register_driver(&s3c64xx_cpufreq_driver);
231}
232module_init(s3c64xx_cpufreq_init);
v6.13.7
  1// SPDX-License-Identifier: GPL-2.0-only
  2/*
  3 * Copyright 2009 Wolfson Microelectronics plc
  4 *
  5 * S3C64xx CPUfreq Support
 
 
 
 
  6 */
  7
  8#define pr_fmt(fmt) "cpufreq: " fmt
  9
 10#include <linux/kernel.h>
 11#include <linux/types.h>
 12#include <linux/init.h>
 13#include <linux/cpufreq.h>
 14#include <linux/clk.h>
 15#include <linux/err.h>
 16#include <linux/regulator/consumer.h>
 17#include <linux/module.h>
 18
 19static struct regulator *vddarm;
 20static unsigned long regulator_latency;
 21
 
 22struct s3c64xx_dvfs {
 23	unsigned int vddarm_min;
 24	unsigned int vddarm_max;
 25};
 26
 27#ifdef CONFIG_REGULATOR
 28static struct s3c64xx_dvfs s3c64xx_dvfs_table[] = {
 29	[0] = { 1000000, 1150000 },
 30	[1] = { 1050000, 1150000 },
 31	[2] = { 1100000, 1150000 },
 32	[3] = { 1200000, 1350000 },
 33	[4] = { 1300000, 1350000 },
 34};
 35#endif
 36
 37static struct cpufreq_frequency_table s3c64xx_freq_table[] = {
 38	{ 0, 0,  66000 },
 39	{ 0, 0, 100000 },
 40	{ 0, 0, 133000 },
 41	{ 0, 1, 200000 },
 42	{ 0, 1, 222000 },
 43	{ 0, 1, 266000 },
 44	{ 0, 2, 333000 },
 45	{ 0, 2, 400000 },
 46	{ 0, 2, 532000 },
 47	{ 0, 2, 533000 },
 48	{ 0, 3, 667000 },
 49	{ 0, 4, 800000 },
 50	{ 0, 0, CPUFREQ_TABLE_END },
 51};
 
 52
 53static int s3c64xx_cpufreq_set_target(struct cpufreq_policy *policy,
 54				      unsigned int index)
 55{
 56	unsigned int new_freq = s3c64xx_freq_table[index].frequency;
 
 57	int ret;
 58
 59#ifdef CONFIG_REGULATOR
 60	struct s3c64xx_dvfs *dvfs;
 61	unsigned int old_freq;
 62
 63	old_freq = clk_get_rate(policy->clk) / 1000;
 
 64	dvfs = &s3c64xx_dvfs_table[s3c64xx_freq_table[index].driver_data];
 65
 
 66	if (vddarm && new_freq > old_freq) {
 67		ret = regulator_set_voltage(vddarm,
 68					    dvfs->vddarm_min,
 69					    dvfs->vddarm_max);
 70		if (ret != 0) {
 71			pr_err("Failed to set VDDARM for %dkHz: %d\n",
 72			       new_freq, ret);
 73			return ret;
 74		}
 75	}
 76#endif
 77
 78	ret = clk_set_rate(policy->clk, new_freq * 1000);
 79	if (ret < 0) {
 80		pr_err("Failed to set rate %dkHz: %d\n",
 81		       new_freq, ret);
 82		return ret;
 83	}
 84
 85#ifdef CONFIG_REGULATOR
 86	if (vddarm && new_freq < old_freq) {
 87		ret = regulator_set_voltage(vddarm,
 88					    dvfs->vddarm_min,
 89					    dvfs->vddarm_max);
 90		if (ret != 0) {
 91			pr_err("Failed to set VDDARM for %dkHz: %d\n",
 92			       new_freq, ret);
 93			if (clk_set_rate(policy->clk, old_freq * 1000) < 0)
 94				pr_err("Failed to restore original clock rate\n");
 95
 96			return ret;
 97		}
 98	}
 99#endif
100
101	pr_debug("Set actual frequency %lukHz\n",
102		 clk_get_rate(policy->clk) / 1000);
103
104	return 0;
105}
106
107#ifdef CONFIG_REGULATOR
108static void s3c64xx_cpufreq_config_regulator(void)
109{
110	int count, v, i, found;
111	struct cpufreq_frequency_table *freq;
112	struct s3c64xx_dvfs *dvfs;
113
114	count = regulator_count_voltages(vddarm);
115	if (count < 0) {
116		pr_err("Unable to check supported voltages\n");
117	}
118
119	if (!count)
120		goto out;
 
 
121
122	cpufreq_for_each_valid_entry(freq, s3c64xx_freq_table) {
123		dvfs = &s3c64xx_dvfs_table[freq->driver_data];
124		found = 0;
125
126		for (i = 0; i < count; i++) {
127			v = regulator_list_voltage(vddarm, i);
128			if (v >= dvfs->vddarm_min && v <= dvfs->vddarm_max)
129				found = 1;
130		}
131
132		if (!found) {
133			pr_debug("%dkHz unsupported by regulator\n",
134				 freq->frequency);
135			freq->frequency = CPUFREQ_ENTRY_INVALID;
136		}
 
 
137	}
138
139out:
140	/* Guess based on having to do an I2C/SPI write; in future we
141	 * will be able to query the regulator performance here. */
142	regulator_latency = 1 * 1000 * 1000;
143}
144#endif
145
146static int s3c64xx_cpufreq_driver_init(struct cpufreq_policy *policy)
147{
 
148	struct cpufreq_frequency_table *freq;
149
150	if (policy->cpu != 0)
151		return -EINVAL;
152
 
 
 
 
 
153	policy->clk = clk_get(NULL, "armclk");
154	if (IS_ERR(policy->clk)) {
155		pr_err("Unable to obtain ARMCLK: %ld\n",
156		       PTR_ERR(policy->clk));
157		return PTR_ERR(policy->clk);
158	}
159
160#ifdef CONFIG_REGULATOR
161	vddarm = regulator_get(NULL, "vddarm");
162	if (IS_ERR(vddarm)) {
163		pr_err("Failed to obtain VDDARM: %ld\n", PTR_ERR(vddarm));
 
164		pr_err("Only frequency scaling available\n");
165		vddarm = NULL;
166	} else {
167		s3c64xx_cpufreq_config_regulator();
168	}
169#endif
170
171	cpufreq_for_each_entry(freq, s3c64xx_freq_table) {
 
172		unsigned long r;
173
174		/* Check for frequencies we can generate */
175		r = clk_round_rate(policy->clk, freq->frequency * 1000);
176		r /= 1000;
177		if (r != freq->frequency) {
178			pr_debug("%dkHz unsupported by clock\n",
179				 freq->frequency);
180			freq->frequency = CPUFREQ_ENTRY_INVALID;
181		}
182
183		/* If we have no regulator then assume startup
184		 * frequency is the maximum we can support. */
185		if (!vddarm && freq->frequency > clk_get_rate(policy->clk) / 1000)
186			freq->frequency = CPUFREQ_ENTRY_INVALID;
 
 
187	}
188
189	/* Datasheet says PLL stabalisation time (if we were to use
190	 * the PLLs, which we don't currently) is ~300us worst case,
191	 * but add some fudge.
192	 */
193	cpufreq_generic_init(policy, s3c64xx_freq_table,
194			(500 * 1000) + regulator_latency);
195	return 0;
 
 
 
 
 
 
 
196}
197
198static struct cpufreq_driver s3c64xx_cpufreq_driver = {
199	.flags		= CPUFREQ_NEED_INITIAL_FREQ_CHECK,
200	.verify		= cpufreq_generic_frequency_table_verify,
201	.target_index	= s3c64xx_cpufreq_set_target,
202	.get		= cpufreq_generic_get,
203	.init		= s3c64xx_cpufreq_driver_init,
204	.name		= "s3c",
205};
206
207static int __init s3c64xx_cpufreq_init(void)
208{
209	return cpufreq_register_driver(&s3c64xx_cpufreq_driver);
210}
211module_init(s3c64xx_cpufreq_init);