aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAnson Huang <b20788@freescale.com>2013-08-08 13:20:40 -0400
committerNitin Garg <nitin.garg@freescale.com>2014-04-16 09:01:05 -0400
commit71493e909a655d94f5fdfe33aaef99c09c7cfe01 (patch)
tree5d4494b00ef776944f8f7b8b5018f5f925b39982
parentef02c33a1bbb01105e6561f3e8dff31c171099a5 (diff)
ENGR00274056-1 thermal: add device cooling for thermal driver
cpu cooling is not enough when temperature is too hot, as some devices may contribute a lot of heat to SOC, such as GPU, so we need to add device cooling as well, when system is too hot, devices can also take their actions to lower SOC temperature. when temperature cross the passive trip, device cooling driver will send out notification, those devices who register this devfreq_cooling notification will take actions to lower SOC temperature. Signed-off-by: Anson Huang <b20788@freescale.com>
-rw-r--r--drivers/thermal/Kconfig7
-rw-r--r--drivers/thermal/Makefile1
-rw-r--r--drivers/thermal/device_cooling.c151
-rw-r--r--include/linux/device_cooling.h45
4 files changed, 204 insertions, 0 deletions
diff --git a/drivers/thermal/Kconfig b/drivers/thermal/Kconfig
index e91d78fff430..cb7d15c43f87 100644
--- a/drivers/thermal/Kconfig
+++ b/drivers/thermal/Kconfig
@@ -102,6 +102,13 @@ config IMX_THERMAL
102 cpufreq is used as the cooling device to throttle CPUs when the 102 cpufreq is used as the cooling device to throttle CPUs when the
103 passive trip is crossed. 103 passive trip is crossed.
104 104
105config DEVICE_THERMAL
106 tristate "generic device cooling support"
107 help
108 Support for device cooling.
109 It supports notification of crossing passive trip for devices,
110 devices need to do their own actions to cool down the SOC.
111
105config SPEAR_THERMAL 112config SPEAR_THERMAL
106 bool "SPEAr thermal sensor driver" 113 bool "SPEAr thermal sensor driver"
107 depends on PLAT_SPEAR 114 depends on PLAT_SPEAR
diff --git a/drivers/thermal/Makefile b/drivers/thermal/Makefile
index 6910b2dfb667..0177697798fa 100644
--- a/drivers/thermal/Makefile
+++ b/drivers/thermal/Makefile
@@ -22,6 +22,7 @@ obj-$(CONFIG_DOVE_THERMAL) += dove_thermal.o
22obj-$(CONFIG_DB8500_THERMAL) += db8500_thermal.o 22obj-$(CONFIG_DB8500_THERMAL) += db8500_thermal.o
23obj-$(CONFIG_ARMADA_THERMAL) += armada_thermal.o 23obj-$(CONFIG_ARMADA_THERMAL) += armada_thermal.o
24obj-$(CONFIG_IMX_THERMAL) += imx_thermal.o 24obj-$(CONFIG_IMX_THERMAL) += imx_thermal.o
25obj-$(CONFIG_DEVICE_THERMAL) += device_cooling.o
25obj-$(CONFIG_DB8500_CPUFREQ_COOLING) += db8500_cpufreq_cooling.o 26obj-$(CONFIG_DB8500_CPUFREQ_COOLING) += db8500_cpufreq_cooling.o
26obj-$(CONFIG_INTEL_POWERCLAMP) += intel_powerclamp.o 27obj-$(CONFIG_INTEL_POWERCLAMP) += intel_powerclamp.o
27 28
diff --git a/drivers/thermal/device_cooling.c b/drivers/thermal/device_cooling.c
new file mode 100644
index 000000000000..1c223a88bc1a
--- /dev/null
+++ b/drivers/thermal/device_cooling.c
@@ -0,0 +1,151 @@
1/*
2 * Copyright (C) 2013 Freescale Semiconductor, Inc.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 as
6 * published by the Free Software Foundation.
7 *
8 */
9
10#include <linux/module.h>
11#include <linux/thermal.h>
12#include <linux/err.h>
13#include <linux/slab.h>
14
15struct devfreq_cooling_device {
16 int id;
17 struct thermal_cooling_device *cool_dev;
18 unsigned int devfreq_state;
19};
20
21static DEFINE_IDR(devfreq_idr);
22static DEFINE_MUTEX(devfreq_cooling_lock);
23
24#define MAX_STATE 1
25
26static BLOCKING_NOTIFIER_HEAD(devfreq_cooling_chain_head);
27
28int register_devfreq_cooling_notifier(struct notifier_block *nb)
29{
30 return blocking_notifier_chain_register(
31 &devfreq_cooling_chain_head, nb);
32}
33EXPORT_SYMBOL_GPL(register_devfreq_cooling_notifier);
34
35int unregister_devfreq_cooling_notifier(struct notifier_block *nb)
36{
37 return blocking_notifier_chain_unregister(
38 &devfreq_cooling_chain_head, nb);
39}
40EXPORT_SYMBOL_GPL(unregister_devfreq_cooling_notifier);
41
42static int devfreq_cooling_notifier_call_chain(unsigned long val)
43{
44 return (blocking_notifier_call_chain(
45 &devfreq_cooling_chain_head, val, NULL)
46 == NOTIFY_BAD) ? -EINVAL : 0;
47}
48
49static int devfreq_set_cur_state(struct thermal_cooling_device *cdev,
50 unsigned long state)
51{
52 struct devfreq_cooling_device *devfreq_device = cdev->devdata;
53 int ret;
54
55 ret = devfreq_cooling_notifier_call_chain(state);
56 if (ret)
57 return -EINVAL;
58 devfreq_device->devfreq_state = state;
59
60 return 0;
61}
62
63static int devfreq_get_max_state(struct thermal_cooling_device *cdev,
64 unsigned long *state)
65{
66 *state = MAX_STATE;
67
68 return 0;
69}
70
71static int devfreq_get_cur_state(struct thermal_cooling_device *cdev,
72 unsigned long *state)
73{
74 struct devfreq_cooling_device *devfreq_device = cdev->devdata;
75
76 *state = devfreq_device->devfreq_state;
77
78 return 0;
79}
80
81static struct thermal_cooling_device_ops const devfreq_cooling_ops = {
82 .get_max_state = devfreq_get_max_state,
83 .get_cur_state = devfreq_get_cur_state,
84 .set_cur_state = devfreq_set_cur_state,
85};
86
87static int get_idr(struct idr *idr, int *id)
88{
89 int ret;
90
91 mutex_lock(&devfreq_cooling_lock);
92 ret = idr_alloc(idr, NULL, 0, 0, GFP_KERNEL);
93 mutex_unlock(&devfreq_cooling_lock);
94 if (unlikely(ret < 0))
95 return ret;
96 *id = ret;
97
98 return 0;
99}
100
101static void release_idr(struct idr *idr, int id)
102{
103 mutex_lock(&devfreq_cooling_lock);
104 idr_remove(idr, id);
105 mutex_unlock(&devfreq_cooling_lock);
106}
107
108struct thermal_cooling_device *devfreq_cooling_register(void)
109{
110 struct thermal_cooling_device *cool_dev;
111 struct devfreq_cooling_device *devfreq_dev = NULL;
112 char dev_name[THERMAL_NAME_LENGTH];
113 int ret = 0;
114
115 devfreq_dev = kzalloc(sizeof(struct devfreq_cooling_device),
116 GFP_KERNEL);
117 if (!devfreq_dev)
118 return ERR_PTR(-ENOMEM);
119
120 ret = get_idr(&devfreq_idr, &devfreq_dev->id);
121 if (ret) {
122 kfree(devfreq_dev);
123 return ERR_PTR(-EINVAL);
124 }
125
126 snprintf(dev_name, sizeof(dev_name), "thermal-devfreq-%d",
127 devfreq_dev->id);
128
129 cool_dev = thermal_cooling_device_register(dev_name, devfreq_dev,
130 &devfreq_cooling_ops);
131 if (!cool_dev) {
132 release_idr(&devfreq_idr, devfreq_dev->id);
133 kfree(devfreq_dev);
134 return ERR_PTR(-EINVAL);
135 }
136 devfreq_dev->cool_dev = cool_dev;
137 devfreq_dev->devfreq_state = 0;
138
139 return cool_dev;
140}
141EXPORT_SYMBOL_GPL(devfreq_cooling_register);
142
143void devfreq_cooling_unregister(struct thermal_cooling_device *cdev)
144{
145 struct devfreq_cooling_device *devfreq_dev = cdev->devdata;
146
147 thermal_cooling_device_unregister(devfreq_dev->cool_dev);
148 release_idr(&devfreq_idr, devfreq_dev->id);
149 kfree(devfreq_dev);
150}
151EXPORT_SYMBOL_GPL(devfreq_cooling_unregister);
diff --git a/include/linux/device_cooling.h b/include/linux/device_cooling.h
new file mode 100644
index 000000000000..ae40a451e4fa
--- /dev/null
+++ b/include/linux/device_cooling.h
@@ -0,0 +1,45 @@
1/*
2 * Copyright (C) 2013 Freescale Semiconductor, Inc.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 as
6 * published by the Free Software Foundation.
7 *
8 */
9
10#ifndef __DEVICE_THERMAL_H__
11#define __DEVICE_THERMAL_H__
12
13#include <linux/thermal.h>
14
15#ifdef CONFIG_DEVICE_THERMAL
16int register_devfreq_cooling_notifier(struct notifier_block *nb);
17int unregister_devfreq_cooling_notifier(struct notifier_block *nb);
18struct thermal_cooling_device *devfreq_cooling_register(void);
19void devfreq_cooling_unregister(struct thermal_cooling_device *cdev);
20#else
21static inline
22int register_devfreq_cooling_notifier(struct notifier_block *nb)
23{
24 return 0;
25}
26
27static inline
28int unregister_devfreq_cooling_notifier(struct notifier_block *nb)
29{
30 return 0;
31}
32
33static inline
34struct thermal_cooling_device *devfreq_cooling_register(void)
35{
36 return NULL;
37}
38
39static inline
40void devfreq_cooling_unregister(struct thermal_cooling_device *cdev)
41{
42 return;
43}
44#endif
45#endif /* __DEVICE_THERMAL_H__ */