aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorLukasz Majewski <l.majewski@samsung.com>2015-01-23 07:10:00 -0500
committerEduardo Valentin <edubezval@gmail.com>2015-01-24 16:32:03 -0500
commit1fe391bf0234add380245dea2dd72220394fe5fd (patch)
treed1d86a97320bd4a4623d2623ae83ef08fdae4895
parentf5576e3a9ea48334289aaaa711080d09e3405099 (diff)
thermal: exynos: Modify exynos thermal code to use device tree for cpu cooling configuration
Up till now exynos_tmu_data.c was used for storing CPU cooling configuration data. Now the Exynos thermal core code uses device tree to get this data. For this purpose generic thermal code for configuring CPU cooling was used. Signed-off-by: Lukasz Majewski <l.majewski@samsung.com> Signed-off-by: Eduardo Valentin <edubezval@gmail.com>
-rw-r--r--drivers/thermal/samsung/exynos_thermal_common.c122
-rw-r--r--drivers/thermal/samsung/exynos_tmu.c7
-rw-r--r--drivers/thermal/samsung/exynos_tmu.h5
-rw-r--r--drivers/thermal/samsung/exynos_tmu_data.c42
4 files changed, 73 insertions, 103 deletions
diff --git a/drivers/thermal/samsung/exynos_thermal_common.c b/drivers/thermal/samsung/exynos_thermal_common.c
index 6dc3815cc73f..00aa68862a52 100644
--- a/drivers/thermal/samsung/exynos_thermal_common.c
+++ b/drivers/thermal/samsung/exynos_thermal_common.c
@@ -133,47 +133,62 @@ static int exynos_get_crit_temp(struct thermal_zone_device *thermal,
133static int exynos_bind(struct thermal_zone_device *thermal, 133static int exynos_bind(struct thermal_zone_device *thermal,
134 struct thermal_cooling_device *cdev) 134 struct thermal_cooling_device *cdev)
135{ 135{
136 int ret = 0, i, tab_size, level;
137 struct freq_clip_table *tab_ptr, *clip_data;
138 struct exynos_thermal_zone *th_zone = thermal->devdata; 136 struct exynos_thermal_zone *th_zone = thermal->devdata;
139 struct thermal_sensor_conf *data = th_zone->sensor_conf; 137 struct thermal_sensor_conf *data = th_zone->sensor_conf;
138 struct device_node *child, *gchild, *np;
139 struct of_phandle_args cooling_spec;
140 unsigned long max, state = 0;
141 int ret = 0, i = 0;
140 142
141 tab_ptr = (struct freq_clip_table *)data->cooling_data.freq_data; 143 /*
142 tab_size = data->cooling_data.freq_clip_count; 144 * Below code is necessary to skip binding when cpufreq's
143 145 * frequency table is not yet initialized.
144 if (tab_ptr == NULL || tab_size == 0) 146 */
147 cdev->ops->get_max_state(cdev, &state);
148 if (!state && !th_zone->cool_dev_size) {
149 th_zone->cool_dev_size = 1;
150 th_zone->cool_dev[0] = cdev;
151 th_zone->bind = false;
145 return 0; 152 return 0;
153 }
146 154
147 /* find the cooling device registered*/ 155 np = of_find_node_by_path("/thermal-zones/cpu-thermal");
148 for (i = 0; i < th_zone->cool_dev_size; i++) 156 if (!np) {
149 if (cdev == th_zone->cool_dev[i]) 157 pr_err("failed to find thmerla-zones/cpu-thermal node\n");
150 break; 158 return -ENOENT;
159 }
151 160
152 /* No matching cooling device */ 161 child = of_get_child_by_name(np, "cooling-maps");
153 if (i == th_zone->cool_dev_size)
154 return 0;
155 162
156 /* Bind the thermal zone to the cpufreq cooling device */ 163 for_each_child_of_node(child, gchild) {
157 for (i = 0; i < tab_size; i++) { 164 ret = of_parse_phandle_with_args(gchild, "cooling-device",
158 clip_data = (struct freq_clip_table *)&(tab_ptr[i]); 165 "#cooling-cells",
159 level = cpufreq_cooling_get_level(0, clip_data->freq_clip_max); 166 0, &cooling_spec);
160 if (level == THERMAL_CSTATE_INVALID) 167 if (ret < 0) {
161 return 0; 168 pr_err("missing cooling_device property\n");
162 switch (GET_ZONE(i)) { 169 goto end;
163 case MONITOR_ZONE: 170 }
164 case WARN_ZONE: 171
165 if (thermal_zone_bind_cooling_device(thermal, i, cdev, 172 if (cooling_spec.args_count < 2) {
166 level, 0)) {
167 dev_err(data->dev,
168 "error unbinding cdev inst=%d\n", i);
169 ret = -EINVAL;
170 }
171 th_zone->bind = true;
172 break;
173 default:
174 ret = -EINVAL; 173 ret = -EINVAL;
174 goto end;
175 } 175 }
176
177 max = cooling_spec.args[0];
178 if (thermal_zone_bind_cooling_device(thermal, i, cdev,
179 max, 0)) {
180 dev_err(data->dev,
181 "thermal error unbinding cdev inst=%d\n", i);
182
183 ret = -EINVAL;
184 goto end;
185 }
186 i++;
176 } 187 }
188 th_zone->bind = true;
189end:
190 of_node_put(child);
191 of_node_put(np);
177 192
178 return ret; 193 return ret;
179} 194}
@@ -182,16 +197,12 @@ static int exynos_bind(struct thermal_zone_device *thermal,
182static int exynos_unbind(struct thermal_zone_device *thermal, 197static int exynos_unbind(struct thermal_zone_device *thermal,
183 struct thermal_cooling_device *cdev) 198 struct thermal_cooling_device *cdev)
184{ 199{
185 int ret = 0, i, tab_size; 200 int ret = 0, i;
186 struct exynos_thermal_zone *th_zone = thermal->devdata; 201 struct exynos_thermal_zone *th_zone = thermal->devdata;
187 struct thermal_sensor_conf *data = th_zone->sensor_conf; 202 struct thermal_sensor_conf *data = th_zone->sensor_conf;
203 struct device_node *child, *gchild, *np;
188 204
189 if (th_zone->bind == false) 205 if (th_zone->bind == false || !th_zone->cool_dev_size)
190 return 0;
191
192 tab_size = data->cooling_data.freq_clip_count;
193
194 if (tab_size == 0)
195 return 0; 206 return 0;
196 207
197 /* find the cooling device registered*/ 208 /* find the cooling device registered*/
@@ -203,23 +214,30 @@ static int exynos_unbind(struct thermal_zone_device *thermal,
203 if (i == th_zone->cool_dev_size) 214 if (i == th_zone->cool_dev_size)
204 return 0; 215 return 0;
205 216
206 /* Bind the thermal zone to the cpufreq cooling device */ 217 np = of_find_node_by_path("/thermal-zones/cpu-thermal");
207 for (i = 0; i < tab_size; i++) { 218 if (!np) {
208 switch (GET_ZONE(i)) { 219 pr_err("failed to find thmerla-zones/cpu-thermal node\n");
209 case MONITOR_ZONE: 220 return -ENOENT;
210 case WARN_ZONE: 221 }
211 if (thermal_zone_unbind_cooling_device(thermal, i, 222
212 cdev)) { 223 child = of_get_child_by_name(np, "cooling-maps");
213 dev_err(data->dev, 224
214 "error unbinding cdev inst=%d\n", i); 225 i = 0;
215 ret = -EINVAL; 226 for_each_child_of_node(child, gchild) {
216 } 227 if (thermal_zone_unbind_cooling_device(thermal, i,
217 th_zone->bind = false; 228 cdev)) {
218 break; 229 dev_err(data->dev,
219 default: 230 "error unbinding cdev inst=%d\n", i);
220 ret = -EINVAL; 231 ret = -EINVAL;
232 goto end;
221 } 233 }
234 i++;
222 } 235 }
236 th_zone->bind = false;
237end:
238 of_node_put(child);
239 of_node_put(np);
240
223 return ret; 241 return ret;
224} 242}
225 243
diff --git a/drivers/thermal/samsung/exynos_tmu.c b/drivers/thermal/samsung/exynos_tmu.c
index 5000727c8c2b..ae30f6af05e0 100644
--- a/drivers/thermal/samsung/exynos_tmu.c
+++ b/drivers/thermal/samsung/exynos_tmu.c
@@ -916,13 +916,6 @@ static int exynos_tmu_probe(struct platform_device *pdev)
916 916
917 sensor_conf->trip_data.trigger_falling = pdata->threshold_falling; 917 sensor_conf->trip_data.trigger_falling = pdata->threshold_falling;
918 918
919 sensor_conf->cooling_data.freq_clip_count = pdata->freq_tab_count;
920 for (i = 0; i < pdata->freq_tab_count; i++) {
921 sensor_conf->cooling_data.freq_data[i].freq_clip_max =
922 pdata->freq_tab[i].freq_clip_max;
923 sensor_conf->cooling_data.freq_data[i].temp_level =
924 pdata->freq_tab[i].temp_level;
925 }
926 sensor_conf->dev = &pdev->dev; 919 sensor_conf->dev = &pdev->dev;
927 /* Register the sensor with thermal management interface */ 920 /* Register the sensor with thermal management interface */
928 ret = exynos_register_thermal(sensor_conf); 921 ret = exynos_register_thermal(sensor_conf);
diff --git a/drivers/thermal/samsung/exynos_tmu.h b/drivers/thermal/samsung/exynos_tmu.h
index 7f880d2e53f5..627dec92ec1b 100644
--- a/drivers/thermal/samsung/exynos_tmu.h
+++ b/drivers/thermal/samsung/exynos_tmu.h
@@ -83,9 +83,6 @@ enum soc_type {
83 * @second_point_trim: temp value of the second point trimming 83 * @second_point_trim: temp value of the second point trimming
84 * @default_temp_offset: default temperature offset in case of no trimming 84 * @default_temp_offset: default temperature offset in case of no trimming
85 * @cal_type: calibration type for temperature 85 * @cal_type: calibration type for temperature
86 * @freq_clip_table: Table representing frequency reduction percentage.
87 * @freq_tab_count: Count of the above table as frequency reduction may
88 * applicable to only some of the trigger levels.
89 * 86 *
90 * This structure is required for configuration of exynos_tmu driver. 87 * This structure is required for configuration of exynos_tmu driver.
91 */ 88 */
@@ -111,8 +108,6 @@ struct exynos_tmu_platform_data {
111 enum soc_type type; 108 enum soc_type type;
112 u32 cal_type; 109 u32 cal_type;
113 u32 cal_mode; 110 u32 cal_mode;
114 struct freq_clip_table freq_tab[4];
115 unsigned int freq_tab_count;
116}; 111};
117 112
118/** 113/**
diff --git a/drivers/thermal/samsung/exynos_tmu_data.c b/drivers/thermal/samsung/exynos_tmu_data.c
index b23910069f68..a993f3d33f9b 100644
--- a/drivers/thermal/samsung/exynos_tmu_data.c
+++ b/drivers/thermal/samsung/exynos_tmu_data.c
@@ -47,15 +47,6 @@ struct exynos_tmu_init_data const exynos4210_default_tmu_data = {
47 .first_point_trim = 25, 47 .first_point_trim = 25,
48 .second_point_trim = 85, 48 .second_point_trim = 85,
49 .default_temp_offset = 50, 49 .default_temp_offset = 50,
50 .freq_tab[0] = {
51 .freq_clip_max = 800 * 1000,
52 .temp_level = 85,
53 },
54 .freq_tab[1] = {
55 .freq_clip_max = 200 * 1000,
56 .temp_level = 100,
57 },
58 .freq_tab_count = 2,
59 .type = SOC_ARCH_EXYNOS4210, 50 .type = SOC_ARCH_EXYNOS4210,
60 }, 51 },
61 }, 52 },
@@ -87,16 +78,7 @@ struct exynos_tmu_init_data const exynos4210_default_tmu_data = {
87 .max_efuse_value = 100, \ 78 .max_efuse_value = 100, \
88 .first_point_trim = 25, \ 79 .first_point_trim = 25, \
89 .second_point_trim = 85, \ 80 .second_point_trim = 85, \
90 .default_temp_offset = 50, \ 81 .default_temp_offset = 50
91 .freq_tab[0] = { \
92 .freq_clip_max = 800 * 1000, \
93 .temp_level = 70, \
94 }, \
95 .freq_tab[1] = { \
96 .freq_clip_max = 400 * 1000, \
97 .temp_level = 95, \
98 }, \
99 .freq_tab_count = 2
100 82
101struct exynos_tmu_init_data const exynos3250_default_tmu_data = { 83struct exynos_tmu_init_data const exynos3250_default_tmu_data = {
102 .tmu_data = { 84 .tmu_data = {
@@ -133,16 +115,7 @@ struct exynos_tmu_init_data const exynos3250_default_tmu_data = {
133 .max_efuse_value = 100, \ 115 .max_efuse_value = 100, \
134 .first_point_trim = 25, \ 116 .first_point_trim = 25, \
135 .second_point_trim = 85, \ 117 .second_point_trim = 85, \
136 .default_temp_offset = 50, \ 118 .default_temp_offset = 50
137 .freq_tab[0] = { \
138 .freq_clip_max = 1400 * 1000, \
139 .temp_level = 70, \
140 }, \
141 .freq_tab[1] = { \
142 .freq_clip_max = 400 * 1000, \
143 .temp_level = 95, \
144 }, \
145 .freq_tab_count = 2
146 119
147struct exynos_tmu_init_data const exynos4412_default_tmu_data = { 120struct exynos_tmu_init_data const exynos4412_default_tmu_data = {
148 .tmu_data = { 121 .tmu_data = {
@@ -189,16 +162,7 @@ struct exynos_tmu_init_data const exynos5250_default_tmu_data = {
189 .max_efuse_value = 100, \ 162 .max_efuse_value = 100, \
190 .first_point_trim = 25, \ 163 .first_point_trim = 25, \
191 .second_point_trim = 85, \ 164 .second_point_trim = 85, \
192 .default_temp_offset = 50, \ 165 .default_temp_offset = 50,
193 .freq_tab[0] = { \
194 .freq_clip_max = 800 * 1000, \
195 .temp_level = 85, \
196 }, \
197 .freq_tab[1] = { \
198 .freq_clip_max = 200 * 1000, \
199 .temp_level = 103, \
200 }, \
201 .freq_tab_count = 2, \
202 166
203#define EXYNOS5260_TMU_DATA \ 167#define EXYNOS5260_TMU_DATA \
204 __EXYNOS5260_TMU_DATA \ 168 __EXYNOS5260_TMU_DATA \