| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Generic big.LITTLE CPUFreq Interface driver | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * It provides necessary ops to arm_big_little cpufreq driver and gets | 
					
						
							|  |  |  |  * Frequency information from Device Tree. Freq table in DT must be in KHz. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2013 Linaro. | 
					
						
							|  |  |  |  * Viresh Kumar <viresh.kumar@linaro.org> | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed "as is" WITHOUT ANY WARRANTY of any | 
					
						
							|  |  |  |  * kind, whether express or implied; without even the implied warranty | 
					
						
							|  |  |  |  * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/cpufreq.h>
 | 
					
						
							|  |  |  | #include <linux/device.h>
 | 
					
						
							|  |  |  | #include <linux/export.h>
 | 
					
						
							|  |  |  | #include <linux/module.h>
 | 
					
						
							| 
									
										
										
										
											2013-06-17 15:51:44 +01:00
										 |  |  | #include <linux/of_device.h>
 | 
					
						
							| 
									
										
										
										
											2013-09-19 16:03:52 -05:00
										 |  |  | #include <linux/pm_opp.h>
 | 
					
						
							| 
									
										
										
										
											2013-05-20 09:57:17 +05:30
										 |  |  | #include <linux/platform_device.h>
 | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | #include <linux/slab.h>
 | 
					
						
							|  |  |  | #include <linux/types.h>
 | 
					
						
							|  |  |  | #include "arm_big_little.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:25:11 +00:00
										 |  |  | /* get cpu node with valid operating-points */ | 
					
						
							|  |  |  | static struct device_node *get_cpu_node_with_valid_op(int cpu) | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-06-17 15:51:44 +01:00
										 |  |  | 	struct device_node *np = of_cpu_device_node_get(cpu); | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-17 15:51:44 +01:00
										 |  |  | 	if (!of_get_property(np, "operating-points", NULL)) { | 
					
						
							|  |  |  | 		of_node_put(np); | 
					
						
							|  |  |  | 		np = NULL; | 
					
						
							| 
									
										
										
										
											2013-04-15 07:05:24 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:25:11 +00:00
										 |  |  | 	return np; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int dt_init_opp_table(struct device *cpu_dev) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct device_node *np; | 
					
						
							|  |  |  | 	int ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-17 15:51:44 +01:00
										 |  |  | 	np = of_node_get(cpu_dev->of_node); | 
					
						
							|  |  |  | 	if (!np) { | 
					
						
							|  |  |  | 		pr_err("failed to find cpu%d node\n", cpu_dev->id); | 
					
						
							|  |  |  | 		return -ENOENT; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-05-17 11:25:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	ret = of_init_opp_table(cpu_dev); | 
					
						
							|  |  |  | 	of_node_put(np); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return ret; | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int dt_get_transition_latency(struct device *cpu_dev) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-17 11:25:11 +00:00
										 |  |  | 	struct device_node *np; | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | 	u32 transition_latency = CPUFREQ_ETERNAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-06-17 15:51:44 +01:00
										 |  |  | 	np = of_node_get(cpu_dev->of_node); | 
					
						
							|  |  |  | 	if (!np) { | 
					
						
							|  |  |  | 		pr_info("Failed to find cpu node. Use CPUFREQ_ETERNAL transition latency\n"); | 
					
						
							| 
									
										
										
										
											2013-04-29 13:24:46 +00:00
										 |  |  | 		return CPUFREQ_ETERNAL; | 
					
						
							| 
									
										
										
										
											2013-06-17 15:51:44 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2013-04-15 07:05:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:25:11 +00:00
										 |  |  | 	of_property_read_u32(np, "clock-latency", &transition_latency); | 
					
						
							|  |  |  | 	of_node_put(np); | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-17 11:25:11 +00:00
										 |  |  | 	pr_debug("%s: clock-latency: %d\n", __func__, transition_latency); | 
					
						
							|  |  |  | 	return transition_latency; | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct cpufreq_arm_bL_ops dt_bL_ops = { | 
					
						
							|  |  |  | 	.name	= "dt-bl", | 
					
						
							|  |  |  | 	.get_transition_latency = dt_get_transition_latency, | 
					
						
							|  |  |  | 	.init_opp_table = dt_init_opp_table, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-20 09:57:17 +05:30
										 |  |  | static int generic_bL_probe(struct platform_device *pdev) | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-17 11:25:11 +00:00
										 |  |  | 	struct device_node *np; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	np = get_cpu_node_with_valid_op(0); | 
					
						
							|  |  |  | 	if (!np) | 
					
						
							|  |  |  | 		return -ENODEV; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	of_node_put(np); | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | 	return bL_cpufreq_register(&dt_bL_ops); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-20 09:57:17 +05:30
										 |  |  | static int generic_bL_remove(struct platform_device *pdev) | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-05-20 09:57:17 +05:30
										 |  |  | 	bL_cpufreq_unregister(&dt_bL_ops); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-05-20 09:57:17 +05:30
										 |  |  | 
 | 
					
						
							|  |  |  | static struct platform_driver generic_bL_platdrv = { | 
					
						
							|  |  |  | 	.driver = { | 
					
						
							|  |  |  | 		.name	= "arm-bL-cpufreq-dt", | 
					
						
							|  |  |  | 		.owner	= THIS_MODULE, | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	.probe		= generic_bL_probe, | 
					
						
							|  |  |  | 	.remove		= generic_bL_remove, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | module_platform_driver(generic_bL_platdrv); | 
					
						
							| 
									
										
										
										
											2013-04-01 12:57:49 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | MODULE_AUTHOR("Viresh Kumar <viresh.kumar@linaro.org>"); | 
					
						
							|  |  |  | MODULE_DESCRIPTION("Generic ARM big LITTLE cpufreq driver via DT"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); |