Hi Eduardo, On Fri, Apr 12, 2013 at 2:34 AM, Eduardo Valentin <eduardo.valentin@xxxxxx> wrote: > > Amit, > > > On 26-03-2013 07:34, Amit Daniel Kachhap wrote: >> >> This sensor registers 3 instance of the tmu controller with the thermal >> zone >> and hence reports 3 temperature output. This driver supports upto five >> trip >> points. For critical threshold the driver uses the core driver thermal >> framework for shutdown and for non-critical threshold it invokes the hw >> based >> frequency clipping limits. Because of such differences with the existing >> 4210 >> tmu controller, exynos5440 tmu driver is added in a new file. >> >> Signed-off-by: Amit Daniel Kachhap <amit.daniel@xxxxxxxxxxx> >> >> --- >> drivers/thermal/samsung/Kconfig | 9 + >> drivers/thermal/samsung/Makefile | 1 + >> drivers/thermal/samsung/exynos5440_thermal.c | 713 >> ++++++++++++++++++++++++++ > > > This driver does not compile as module: > ERROR: "exynos_report_trigger" > [drivers/thermal/samsung/exynos5440_thermal.ko] undefined! > ERROR: "exynos_get_frequency_level" > [drivers/thermal/samsung/exynos5440_thermal.ko] undefined! > ERROR: "exynos_unregister_thermal" > [drivers/thermal/samsung/exynos5440_thermal.ko] undefined! Ok will fix it. > > > Besides, this driver is pretty similar to 4210 driver. Are you you cannot > isolate the difference into config data? Again, check the driver design for > TI SoC thermal (drivers/staging/ti-soc-thermal/ on linux-next) Yes I started with a 4210 file and srarted modifying it but then lot of conditional code started getting inserted and it become very complex. So I splitted this file itself. Will check your implementation. Thanks, Amit Daniel > > >> 3 files changed, 723 insertions(+), 0 deletions(-) >> create mode 100644 drivers/thermal/samsung/exynos5440_thermal.c >> >> diff --git a/drivers/thermal/samsung/Kconfig >> b/drivers/thermal/samsung/Kconfig >> index cefe693..0c7b4eb 100644 >> --- a/drivers/thermal/samsung/Kconfig >> +++ b/drivers/thermal/samsung/Kconfig >> @@ -20,4 +20,13 @@ config EXYNOS4210_THERMAL >> initialises the TMU controller and registers/unregisters with >> exynos >> common thermal layer. >> >> +config EXYNOS5440_THERMAL >> + tristate "Temperature sensor on Samsung EXYNOS 5440 SOC" >> + depends on SOC_EXYNOS5440 >> + help >> + If you say yes here you can enable TMU (Thermal Management Unit) >> + support on SAMSUNG EXYNOS 5440 series of SoC. This option >> initialises >> + the TMU controller and registers/unregisters with exynos common >> + thermal layer. >> + >> endif >> diff --git a/drivers/thermal/samsung/Makefile >> b/drivers/thermal/samsung/Makefile >> index d51d0c2..53230cf 100644 >> --- a/drivers/thermal/samsung/Makefile >> +++ b/drivers/thermal/samsung/Makefile >> @@ -3,3 +3,4 @@ >> # >> obj-$(CONFIG_EXYNOS_COMMON) += exynos_common.o >> obj-$(CONFIG_EXYNOS4210_THERMAL) += exynos4210_thermal.o >> +obj-$(CONFIG_EXYNOS5440_THERMAL) += exynos5440_thermal.o >> diff --git a/drivers/thermal/samsung/exynos5440_thermal.c >> b/drivers/thermal/samsung/exynos5440_thermal.c >> new file mode 100644 >> index 0000000..a3c75d3 >> --- /dev/null >> +++ b/drivers/thermal/samsung/exynos5440_thermal.c >> @@ -0,0 +1,713 @@ >> +/* >> + * exynos5440_thermal.c - Samsung EXYNOS 5440 TMU >> + * (Thermal Management Unit) >> + * >> + * Copyright (C) 2013 Samsung Electronics >> + * Amit Daniel Kachhap <amit.daniel@xxxxxxxxxxx> >> + * >> + * This program is free software; you can redistribute it and/or modify >> + * it under the terms of the GNU General Public License as published by >> + * the Free Software Foundation; either version 2 of the License, or >> + * (at your option) any later version. >> + * >> + * This program is distributed in the hope that it will be useful, >> + * but WITHOUT ANY WARRANTY; without even the implied warranty of >> + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the >> + * GNU General Public License for more details. >> + * >> + * You should have received a copy of the GNU General Public License >> + * along with this program; if not, write to the Free Software >> + * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 >> USA >> + * >> + */ >> + >> +#include <linux/clk.h> >> +#include <linux/cpufreq.h> >> +#include <linux/cpu_cooling.h> >> +#include <linux/err.h> >> +#include <linux/interrupt.h> >> +#include <linux/io.h> >> +#include <linux/kernel.h> >> +#include <linux/kobject.h> >> +#include <linux/module.h> >> +#include <linux/mutex.h> >> +#include <linux/of.h> >> +#include <linux/of_address.h> >> +#include <linux/of_irq.h> >> +#include <linux/platform_device.h> >> +#include <linux/slab.h> >> +#include <linux/thermal.h> >> +#include <linux/workqueue.h> >> +#include <linux/platform_data/exynos_thermal.h> >> + >> +#include "exynos_common.h" >> + >> + >> +/* Exynos5440 specific registers */ >> +#define TMU_S0_7_TRIM 0x0118 >> +#define TMU_S0_7_CTRL 0x0138 >> +#define TMU_S0_7_DEBUG 0x0158 >> +#define TMU_S0_7_STATUS 0x0178 >> +#define TMU_S0_7_COUNTER0 0x0198 >> +#define TMU_S0_7_COUNTER1 0x01b8 >> +#define TMU_S0_7_COUNTER2 0x01d8 >> +#define TMU_S0_7_COUNTER3 0x01f8 >> +#define TMU_S0_7_TEMP 0x0208 >> +#define TMU_S0_7_TH0 0x0228 >> +#define TMU_S0_7_TH1 0x0248 >> +#define TMU_S0_7_TH2 0x0268 >> +#define TMU_S0_7_PTEMP0 0x0288 >> +#define TMU_S0_7_PTEMP1 0x02a8 >> +#define TMU_S0_7_PTEMP2 0x02c8 >> +#define TMU_S0_7_PTEMP3 0x02e8 >> +#define TMU_S0_7_EVTEN 0x0308 >> +#define TMU_S0_7_IRQEN 0x0328 >> +#define TMU_S0_7_IRQ 0x0348 >> +#define TMU_IRQ_STATUS 0x0368 >> +#define TMU_PMIN 0x036c >> +#define TMU_TEMP 0x0370 >> +#define TMU_MISC 0x0374 >> + >> +/* Exynos5440 specific mask and shifts */ >> +#define TMU_TEMP_MASK 0xff >> + >> +#define TMU_TRIM_DATA_25C_SHIFT 0x0 >> +#define TMU_TRIM_DATA_85C_SHIFT 0x8 >> + >> +#define TMU_BUF_VREF_SEL_MASK 0x1f >> +#define TMU_BUF_VREF_SEL_SHIFT 24 >> +#define TMU_THERM_TRIP_MODE_MASK 0x7 >> +#define TMU_THERM_TRIP_MODE_SHIFT 13 >> +#define TMU_THERM_TRIP_EN_SHIFT 12 >> +#define TMU_BUF_SLOPE_SEL_MASK 0Xf >> +#define TMU_BUF_SLOPE_SEL_SHIFT 8 >> +#define TMU_THERM_IRQ_MODE_SHIFT 7 >> +#define TMU_CALIB_MODE_MASK 0x3 >> +#define TMU_CALIB_MODE_SHIFT 4 >> +#define TMU_FILTER_MODE_MASK 0x7 >> +#define TMU_FILTER_MODE_SHIFT 1 >> +#define TMU_SENSOR_EN_SHIFT 0 >> +#define TMU_SENSOR_ENABLE 0x1 >> + >> +#define TMU_EMU_EN_SHIFT 0 >> +#define TMU_TEMP_EMU_SHIFT 8 >> +#define TMU_EMUL_ENABLE 1 >> + >> +#define TMU_STATUS_IDLE_SHIFT 0 >> + >> +#define TMU_TIME_MASK 0xffff >> +#define TMU_TIME_OF_SHIFT 16 >> +#define TMU_TIME_ON_SHIFT 0 >> + >> +#define TMU_CURRENT_TEMP_SHIFT 0 >> +#define TMU_FILTERED_TEMP_SHIFT 8 >> +#define TMU_RAW_TEMP_SHIFT 16 >> +#define TMU_TEMP_SEQNUM 24 >> + >> +#define TMU_THRES_RISE0_SHIFT 0 >> +#define TMU_THRES_RISE1_SHIFT 8 >> +#define TMU_THRES_RISE2_SHIFT 16 >> +#define TMU_THRES_RISE3_SHIFT 24 >> + >> +#define TMU_THRES_FALL0_SHIFT 0 >> +#define TMU_THRES_FALL1_SHIFT 8 >> +#define TMU_THRES_FALL2_SHIFT 16 >> +#define TMU_THRES_FALL3_SHIFT 24 >> + >> +#define TMU_THRES_RISE4_SHIFT 24 >> + >> +#define TMU_RISE_EVTEN_MASK 0xf >> +#define TMU_RISE_EVTEN_SHIFT 0 >> +#define TMU_FALL_EVTEN_MASK 0xf >> +#define TMU_FALL_EVTEN_SHIFT 4 >> + >> +#define TMU_RISE_IRQEN_MASK 0xf >> +#define TMU_RISE_IRQEN_SHIFT 0 >> +#define TMU_FALL_IRQEN_MASK 0xf >> +#define TMU_FALL_IRQEN_SHIFT 4 >> +#define TMU_CLEAR_RISE_INT TMU_RISE_IRQEN_MASK >> +#define TMU_CLEAR_FALL_INT (TMU_FALL_IRQEN_MASK << 4) >> + >> +#define TMU_PMIN_MASK 0x7 >> +#define TMU_PMIN0_SHIFT 0 >> +#define TMU_PMIN1_SHIFT 4 >> +#define TMU_PMIN2_SHIFT 8 >> +#define TMU_PMIN3_SHIFT 12 >> +#define TMU_PMIN_SHIFT(x) (4 * x) >> +#define TMU_TPMIN_SHIFT 16 >> + >> +#define TMU_TEMP_MAX_SHIFT 0 >> +#define TMU_MAX_RISE_LEVEL 4 >> +#define TMU_MAX_FALL_LEVEL 4 >> +#define TMU_MAX_SENSOR 8 >> + >> +#define TMU_DEF_CODE_TO_TEMP_OFFSET 20 >> + >> +struct exynos_tmu_data { >> + int irq; >> + int id; >> + unsigned int shift; >> + enum soc_type soc; >> + void __iomem *base; >> + struct clk *clk; >> + struct work_struct irq_work; >> + u8 temp_error1, temp_error2; >> + struct mutex lock; >> + struct thermal_sensor_conf *reg_conf; >> + struct exynos_tmu_platform_data *pdata; >> +}; >> + >> +struct exynos_tmu_common { >> + int level[TMU_MAX_SENSOR]; >> + int sensor_count; >> +}; >> +static struct exynos_tmu_common tmu_common; >> +/* >> + * TMU treats temperature as a mapped temperature code. >> + * The temperature is converted differently depending on the calibration >> type. >> + */ >> +static int temp_to_code(struct exynos_tmu_data *data, u8 temp) >> +{ >> + struct exynos_tmu_platform_data *pdata = data->pdata; >> + int temp_code; >> + >> + if (pdata->cal_mode == HW_MODE) >> + return temp; >> + >> + switch (pdata->cal_type) { >> + case TYPE_TWO_POINT_TRIMMING: >> + temp_code = (temp - 25) * >> + (data->temp_error2 - data->temp_error1) / >> + (70 - 25) + data->temp_error1; >> + break; >> + case TYPE_ONE_POINT_TRIMMING: >> + temp_code = temp + data->temp_error1 - 25; >> + break; >> + default: >> + temp_code = temp + TMU_DEF_CODE_TO_TEMP_OFFSET; >> + break; >> + } >> + >> + return temp_code; >> +} >> + >> +/* >> + * Calculate a temperature value from a temperature code. >> + * The unit of the temperature is degree Celsius. >> + */ >> +static int code_to_temp(struct exynos_tmu_data *data, u8 temp_code) >> +{ >> + struct exynos_tmu_platform_data *pdata = data->pdata; >> + int temp; >> + >> + if (pdata->cal_mode == HW_MODE) >> + return temp_code; >> + >> + switch (pdata->cal_type) { >> + case TYPE_TWO_POINT_TRIMMING: >> + temp = (temp_code - data->temp_error1) * (70 - 25) / >> + (data->temp_error2 - data->temp_error1) + 25; >> + break; >> + case TYPE_ONE_POINT_TRIMMING: >> + temp = temp_code - data->temp_error1 + 25; >> + break; >> + default: >> + temp = temp_code - TMU_DEF_CODE_TO_TEMP_OFFSET; >> + break; >> + } >> + >> + return temp; >> +} >> + >> +static int exynos_tmu_initialize(struct platform_device *pdev) >> +{ >> + struct exynos_tmu_data *data = platform_get_drvdata(pdev); >> + struct exynos_tmu_platform_data *pdata = data->pdata; >> + unsigned int status, con, trim_info; >> + unsigned int rising_threshold = 0, falling_threshold = 0; >> + int ret = 0, threshold_code, i, trigger_levs = 0; >> + >> + status = readl(data->base + data->shift + TMU_S0_7_STATUS); >> + status &= 0x1; >> + if (!status) >> + dev_err(&pdev->dev, "Sensor Initial status is busy\n"); >> + >> + if (pdata->cal_mode == HW_MODE) >> + goto skip_calib_data; >> + >> + /* Save trimming info in order to perform calibration */ >> + trim_info = readl(data->base + data->shift + TMU_S0_7_TRIM); >> + data->temp_error1 = trim_info & TMU_TEMP_MASK; >> + data->temp_error2 = ((trim_info >> 8) & TMU_TEMP_MASK); >> + if (!data->temp_error1) >> + data->temp_error1 = pdata->efuse_value & TMU_TEMP_MASK; >> + if (!data->temp_error2) >> + data->temp_error2 = (pdata->efuse_value >> 8) & >> TMU_TEMP_MASK; >> + >> +skip_calib_data: >> + /* Count trigger levels to be enabled */ >> + for (i = 0; i < MAX_THRESHOLD_LEVS; i++) >> + if (pdata->trigger_levels[i]) >> + trigger_levs++; >> + >> + /* Write temperature code for rising and falling threshold */ >> + for (i = 0; (i < trigger_levs && i < TMU_MAX_RISE_LEVEL); i++) { >> + threshold_code = temp_to_code(data, >> + pdata->trigger_levels[i]); >> + if (threshold_code < 0) { >> + ret = threshold_code; >> + dev_err(&pdev->dev, "Invalid threshold=%d >> level=%d\n", >> + threshold_code, >> i); >> + goto out; >> + } >> + rising_threshold |= threshold_code << 8 * i; >> + if (pdata->threshold_falling) { >> + threshold_code = temp_to_code(data, >> + pdata->trigger_levels[i] - >> + pdata->threshold_falling); >> + if (threshold_code > 0) >> + falling_threshold |= >> + threshold_code << 8 * i; >> + } >> + } >> + writel(rising_threshold, >> + data->base + data->shift + TMU_S0_7_TH0); >> + writel(falling_threshold, >> + data->base + data->shift + TMU_S0_7_TH1); >> + >> + /* if 5th threshold limit is also present */ >> + if (i == TMU_MAX_RISE_LEVEL) { >> + threshold_code = temp_to_code(data, >> + pdata->trigger_levels[i]); >> + if (threshold_code < 0) { >> + ret = threshold_code; >> + dev_err(&pdev->dev, "Invalid threshold=%d >> level=%d\n", >> + threshold_code, >> i); >> + goto out; >> + } >> + rising_threshold = threshold_code << >> TMU_THRES_RISE4_SHIFT; >> + writel(rising_threshold, >> + data->base + data->shift + TMU_S0_7_TH2); >> + con = readl(data->base + data->shift + TMU_S0_7_CTRL); >> + con |= (1 << TMU_THERM_TRIP_EN_SHIFT); >> + writel(con, data->base + data->shift + TMU_S0_7_CTRL); >> + } >> + >> + writel(TMU_CLEAR_RISE_INT | TMU_CLEAR_FALL_INT, >> + data->base + data->shift + TMU_S0_7_IRQ); >> + >> + /* clear all PMIN */ >> + writel(0, data->base + TMU_PMIN); >> +out: >> + return ret; >> +} >> + >> +static void exynos_tmu_control(struct platform_device *pdev, bool on) >> +{ >> + struct exynos_tmu_data *data = platform_get_drvdata(pdev); >> + struct exynos_tmu_platform_data *pdata = data->pdata; >> + unsigned int con, interrupt_en; >> + >> + mutex_lock(&data->lock); >> + con = readl(data->base + data->shift + TMU_S0_7_CTRL); >> + con &= ~(TMU_BUF_VREF_SEL_MASK << TMU_BUF_VREF_SEL_SHIFT | >> + TMU_THERM_TRIP_MODE_MASK << TMU_THERM_TRIP_MODE_SHIFT | >> + TMU_BUF_SLOPE_SEL_MASK << TMU_BUF_SLOPE_SEL_SHIFT | >> + TMU_CALIB_MODE_MASK << TMU_CALIB_MODE_SHIFT | >> + TMU_FILTER_MODE_MASK << TMU_FILTER_MODE_SHIFT | >> + TMU_SENSOR_ENABLE << TMU_SENSOR_EN_SHIFT); >> + >> + con |= pdata->reference_voltage << TMU_BUF_VREF_SEL_SHIFT | >> + pdata->gain << TMU_BUF_SLOPE_SEL_SHIFT; >> + >> + if (pdata->cal_mode == HW_MODE) >> + con |= pdata->cal_type << TMU_CALIB_MODE_SHIFT; >> + >> + con |= pdata->noise_cancel_mode << TMU_THERM_TRIP_MODE_SHIFT; >> + >> + if (on) { >> + con |= TMU_SENSOR_ENABLE; >> + interrupt_en = >> + pdata->trigger_enable[3] << 3 | >> + pdata->trigger_enable[2] << 2 | >> + pdata->trigger_enable[1] << 1 | >> + pdata->trigger_enable[0] << 0; >> + if (pdata->threshold_falling) >> + interrupt_en |= interrupt_en << >> TMU_FALL_IRQEN_SHIFT; >> + } else { >> + interrupt_en = 0; /* Disable all interrupts */ >> + } >> + writel(interrupt_en, data->base + data->shift + TMU_S0_7_IRQEN); >> + writel(interrupt_en, data->base + data->shift + TMU_S0_7_EVTEN); >> + writel(con, data->base + data->shift + TMU_S0_7_CTRL); >> + >> + mutex_unlock(&data->lock); >> +} >> + >> +static int exynos_tmu_read(struct exynos_tmu_data *data) >> +{ >> + u8 temp_code; >> + int temp; >> + >> + mutex_lock(&data->lock); > > > Dont you need to enable clocks? > > >> + >> + temp_code = readl(data->base + data->shift + TMU_S0_7_TEMP); >> + temp_code >>= TMU_CURRENT_TEMP_SHIFT; >> + temp_code &= TMU_TEMP_MASK; >> + temp = code_to_temp(data, temp_code); >> + >> + mutex_unlock(&data->lock); >> + >> + return temp; >> +} >> + >> +#ifdef CONFIG_THERMAL_EMULATION >> +static int exynos_tmu_set_emulation(struct exynos_tmu_data *data, >> + unsigned long temp) >> +{ >> + unsigned int reg; >> + >> + if (temp && temp < MCELSIUS) >> + goto out; >> + >> + mutex_lock(&data->lock); >> + reg = readl(data->base + data->shift + TMU_S0_7_DEBUG); >> + >> + if (temp) { >> + temp /= MCELSIUS; >> + reg &= ~(TMU_TEMP_MASK << TMU_TEMP_EMU_SHIFT); >> + reg |= (temp_to_code(data, temp) << TMU_TEMP_EMU_SHIFT) | >> + TMU_EMUL_ENABLE; >> + } else { >> + reg &= ~TMU_EMUL_ENABLE; >> + } >> + >> + writel(reg, data->base + data->shift + TMU_S0_7_DEBUG); >> + mutex_unlock(&data->lock); >> + return 0; >> +out: >> + return -EINVAL; >> +} >> +#endif >> + >> +static void exynos_tmu_set_cooling(struct exynos_tmu_data *data, int >> level, >> + unsigned int cur_temp) >> +{ >> + struct exynos_tmu_platform_data *pdata = data->pdata; >> + bool check_rise, change; >> + unsigned int thres_temp, freq = 0, val; >> + int i, index, max_level = 0; >> + >> + /* Get the max level across all sensors except this */ >> + for (i = 0; i < tmu_common.sensor_count; i++) { >> + if (i == data->id) >> + continue; >> + if (tmu_common.level[i] > max_level) >> + max_level = tmu_common.level[i]; >> + } >> + change = false; >> + if (level < TMU_MAX_RISE_LEVEL) { >> + thres_temp = readl(data->base + data->shift + >> TMU_S0_7_TH0); >> + thres_temp = (thres_temp >> (level * 8) & TMU_TEMP_MASK); >> + check_rise = true; >> + tmu_common.level[data->id] = level + 1; >> + if (tmu_common.level[data->id] > max_level) >> + change = true; >> + } else { >> + level -= TMU_MAX_RISE_LEVEL; >> + thres_temp = readl(data->base + data->shift + >> TMU_S0_7_TH1); >> + thres_temp = (thres_temp >> (level * 8) & TMU_TEMP_MASK); >> + check_rise = false; >> + tmu_common.level[data->id] = level; >> + if (tmu_common.level[data->id] >= max_level) >> + change = true; >> + } >> + >> + if (change == false) >> + return; >> + >> + thres_temp = code_to_temp(data, thres_temp); >> + if (!check_rise) >> + thres_temp += pdata->threshold_falling; >> + >> + change = false; >> + /* find this threshold temp in the patform table cooling data */ >> + for (i = 0; i < pdata->freq_tab_count; i++) { >> + if (thres_temp != pdata->freq_tab[i].temp_level) >> + continue; >> + >> + if (check_rise && cur_temp >= thres_temp) { >> + freq = pdata->freq_tab[i].freq_clip_max; >> + change = true; >> + } >> + if (!check_rise && >> + (cur_temp <= (thres_temp - pdata->threshold_falling))) { >> + change = true; >> + freq = 0; >> + } >> + } >> + >> + /* critical threshold temp */ >> + if (thres_temp == pdata->trigger_levels[TMU_MAX_RISE_LEVEL - 1]) >> + exynos_report_trigger(data->reg_conf); >> + >> + if (change == false) >> + return; >> + >> + index = 0; >> + >> + if (freq) { >> + index = exynos_get_frequency_level(0, freq); >> + if (index < 0) >> + return; >> + } >> + >> + val = readl(data->base + TMU_PMIN); >> + val &= (~(TMU_PMIN_MASK << TMU_PMIN_SHIFT(level))); >> + val |= (index << TMU_PMIN_SHIFT(level)); >> + writel(val, data->base + TMU_PMIN); >> +} >> + >> +static void exynos_tmu_work(struct work_struct *work) >> +{ >> + struct exynos_tmu_data *data = container_of(work, >> + struct exynos_tmu_data, irq_work); >> + int i, cur_temp; >> + unsigned int val_type, val_irq; >> + >> + if (!data) >> + goto out; >> + >> + val_type = readl(data->base + TMU_IRQ_STATUS); >> + >> + /* Find which sensor generated this interrupt */ >> + if (!((val_type >> data->id) & 0x1)) >> + goto out; >> + >> + cur_temp = exynos_tmu_read(data); >> + val_irq = readl(data->base + data->shift + TMU_S0_7_IRQ); >> + for (i = 0; i < (TMU_MAX_RISE_LEVEL + TMU_MAX_FALL_LEVEL); i++) { >> + if (!((val_irq >> i) & 0x1)) >> + continue; >> + exynos_tmu_set_cooling(data, i, cur_temp); >> + } >> + /* clear the interrupts */ >> + writel(val_irq, data->base + data->shift + TMU_S0_7_IRQ); >> +out: >> + enable_irq(data->irq); >> +} >> + >> +static irqreturn_t exynos_tmu_irq(int irq, void *id) >> +{ >> + struct exynos_tmu_data *data = id; >> + >> + disable_irq_nosync(irq); >> + schedule_work(&data->irq_work); >> + >> + return IRQ_HANDLED; >> +} >> + >> +static const struct of_device_id exynos_tmu_match[] = { >> + { >> + .compatible = "samsung,exynos5440-tmu", >> + }, >> + {}, >> +}; >> +MODULE_DEVICE_TABLE(of, exynos_tmu_match); >> + >> +int exynos_map_dt_data(struct platform_device *pdev) >> +{ >> + struct exynos_tmu_data *data = platform_get_drvdata(pdev); >> + struct resource res; >> + >> + if (!data) >> + return -ENODEV; >> + >> + data->id = of_alias_get_id(pdev->dev.of_node, "tmuctrl"); >> + if (data->id < 0) >> + data->id = 0; >> + >> + data->shift = data->id * 4; >> + >> + data->irq = irq_of_parse_and_map(pdev->dev.of_node, 0); >> + if (data->irq <= 0) { >> + dev_err(&pdev->dev, "failed to get IRQ\n"); >> + return -ENODEV; >> + } >> + >> + if (of_address_to_resource(pdev->dev.of_node, 0, &res)) { >> + dev_err(&pdev->dev, "failed to get Resource\n"); >> + return -ENODEV; >> + } >> + >> + /* clear the last 16 bytes */ >> + res.start &= (~(0xFFFF)); >> + data->base = devm_ioremap(&pdev->dev, res.start, >> resource_size(&res)); >> + if (!data->base) { >> + dev_err(&pdev->dev, "Failed to ioremap memory\n"); >> + return -ENOMEM; >> + } >> + return 0; >> +} >> + >> +static int exynos_tmu_probe(struct platform_device *pdev) >> +{ >> + struct exynos_tmu_data *data; >> + struct exynos_tmu_platform_data *pdata; >> + struct thermal_sensor_conf *sensor_conf; >> + int ret, i; >> + >> + data = devm_kzalloc(&pdev->dev, sizeof(struct exynos_tmu_data), >> + GFP_KERNEL); >> + if (!data) { >> + dev_err(&pdev->dev, "Failed to allocate driver >> structure\n"); >> + return -ENOMEM; >> + } >> + >> + pdata = (struct exynos_tmu_platform_data *) >> + platform_get_device_id(pdev)->driver_data; >> + if (!pdata) { >> + dev_err(&pdev->dev, "No platform init data supplied.\n"); >> + return -ENODEV; >> + } >> + >> + data->pdata = pdata; >> + platform_set_drvdata(pdev, data); >> + >> + ret = exynos_map_dt_data(pdev); >> + if (ret) >> + goto unset_data; >> + >> + INIT_WORK(&data->irq_work, exynos_tmu_work); >> + >> + ret = devm_request_irq(&pdev->dev, data->irq, exynos_tmu_irq, >> + IRQF_TRIGGER_RISING|IRQF_SHARED, dev_name(&pdev->dev), >> data); >> + if (ret) { >> + dev_err(&pdev->dev, "Failed to request irq: %d\n", >> data->irq); >> + goto unset_data; >> + } >> + >> + data->clk = of_clk_get(pdev->dev.of_node, 0); >> + if (IS_ERR(data->clk)) { >> + dev_err(&pdev->dev, "Failed to get tmu clock\n"); >> + ret = PTR_ERR(data->clk); >> + goto unset_data; >> + } >> + clk_enable(data->clk); >> + > > > hmmm ok, you want it to be always running, right? > > >> + mutex_init(&data->lock); >> + >> + ret = exynos_tmu_initialize(pdev); >> + if (ret) { >> + dev_err(&pdev->dev, "Failed to initialize TMU\n"); >> + goto err_clk; >> + } >> + >> + exynos_tmu_control(pdev, true); >> + >> + /* Allocate a structure to register with the exynos core thermal >> */ >> + sensor_conf = devm_kzalloc(&pdev->dev, >> + sizeof(struct thermal_sensor_conf), >> GFP_KERNEL); >> + if (!sensor_conf) { >> + dev_err(&pdev->dev, "Failed to allocate registration >> struct\n"); >> + ret = -ENOMEM; >> + goto err_clk; >> + } >> + data->reg_conf = sensor_conf; >> + sprintf(sensor_conf->name, "therm_zone%d", data->id); >> + sensor_conf->read_temperature = (int (*)(void *))exynos_tmu_read; >> +#ifdef CONFIG_THERMAL_EMULATION >> + sensor_conf->write_emul_temp = >> + (int (*)(void *, unsigned long))exynos_tmu_set_emulation; >> +#endif > > > Do you really need this ifdef here? Cant you do same as you have done for > 4210? > > >> + sensor_conf->driver_data = data; >> + sensor_conf->trip_data.trip_count = pdata->trigger_enable[0] + >> + pdata->trigger_enable[1] + >> pdata->trigger_enable[2] + >> + pdata->trigger_enable[3]; >> + >> + for (i = 0; i < sensor_conf->trip_data.trip_count; i++) >> + sensor_conf->trip_data.trip_val[i] = >> pdata->trigger_levels[i]; >> + >> + sensor_conf->trip_data.trigger_falling = pdata->threshold_falling; >> + >> + /* Register the sensor with thermal management interface */ >> + ret = exynos_register_thermal(sensor_conf); >> + if (ret) { >> + dev_err(&pdev->dev, "Failed to register thermal >> interface\n"); >> + goto err_clk; >> + } >> + tmu_common.sensor_count++; >> + return 0; >> +err_clk: >> + clk_disable(data->clk); >> + clk_put(data->clk); >> +unset_data: >> + platform_set_drvdata(pdev, NULL); >> + return ret; >> +} >> + >> +static int exynos_tmu_remove(struct platform_device *pdev) >> +{ >> + struct exynos_tmu_data *data = platform_get_drvdata(pdev); >> + struct thermal_sensor_conf *sensor_conf = data->reg_conf; >> + >> + exynos_tmu_control(pdev, false); >> + clk_disable(data->clk); >> + >> + exynos_unregister_thermal(sensor_conf); >> + >> + clk_put(data->clk); >> + >> + platform_set_drvdata(pdev, NULL); >> + >> + return 0; >> +} >> + >> +#ifdef CONFIG_PM_SLEEP >> +static int exynos_tmu_suspend(struct device *dev) >> +{ >> + struct platform_device *pdev = to_platform_device(dev); >> + struct exynos_tmu_data *data = platform_get_drvdata(pdev); >> + >> + exynos_tmu_control(pdev, false); >> + clk_disable(data->clk); >> + >> + return 0; >> +} >> + >> +static int exynos_tmu_resume(struct device *dev) >> +{ >> + struct platform_device *pdev = to_platform_device(dev); >> + struct exynos_tmu_data *data = platform_get_drvdata(pdev); >> + >> + clk_enable(data->clk); >> + exynos_tmu_initialize(pdev); >> + exynos_tmu_control(pdev, true); >> + >> + return 0; >> +} >> + >> +static SIMPLE_DEV_PM_OPS(exynos_tmu_pm, >> + exynos_tmu_suspend, exynos_tmu_resume); >> +#define EXYNOS_TMU_PM (&exynos_tmu_pm) >> +#else >> +#define EXYNOS_TMU_PM NULL >> +#endif >> + >> +static struct platform_driver exynos_tmu_driver = { >> + .driver = { >> + .name = "exynos5440-tmu", >> + .owner = THIS_MODULE, >> + .pm = EXYNOS_TMU_PM, >> + .of_match_table = exynos_tmu_match, >> + }, >> + .probe = exynos_tmu_probe, >> + .remove = exynos_tmu_remove, >> +}; >> + >> +module_platform_driver(exynos_tmu_driver); >> + >> +MODULE_DESCRIPTION("EXYNOS5440 TMU Driver"); >> +MODULE_AUTHOR("Amit Daniel<amit.daniel@xxxxxxxxxxx>"); >> +MODULE_LICENSE("GPL"); > > GPL v2? > >> +MODULE_ALIAS("platform:exynos5440-tmu"); >> > -- To unsubscribe from this list: send the line "unsubscribe linux-samsung-soc" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html