|  | /* | 
|  | * Marvell EBU Armada SoCs thermal sensor driver | 
|  | * | 
|  | * Copyright (C) 2013 Marvell | 
|  | * | 
|  | * This software is licensed under the terms of the GNU General Public | 
|  | * License version 2, as published by the Free Software Foundation, and | 
|  | * may be copied, distributed, and modified under those terms. | 
|  | * | 
|  | * 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. | 
|  | * | 
|  | */ | 
|  | #include <linux/device.h> | 
|  | #include <linux/err.h> | 
|  | #include <linux/io.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/of.h> | 
|  | #include <linux/module.h> | 
|  | #include <linux/delay.h> | 
|  | #include <linux/platform_device.h> | 
|  | #include <linux/of_device.h> | 
|  | #include <linux/thermal.h> | 
|  | #include <linux/iopoll.h> | 
|  | #include <linux/mfd/syscon.h> | 
|  | #include <linux/regmap.h> | 
|  |  | 
|  | /* Thermal Manager Control and Status Register */ | 
|  | #define PMU_TDC0_SW_RST_MASK		(0x1 << 1) | 
|  | #define PMU_TM_DISABLE_OFFS		0 | 
|  | #define PMU_TM_DISABLE_MASK		(0x1 << PMU_TM_DISABLE_OFFS) | 
|  | #define PMU_TDC0_REF_CAL_CNT_OFFS	11 | 
|  | #define PMU_TDC0_REF_CAL_CNT_MASK	(0x1ff << PMU_TDC0_REF_CAL_CNT_OFFS) | 
|  | #define PMU_TDC0_OTF_CAL_MASK		(0x1 << 30) | 
|  | #define PMU_TDC0_START_CAL_MASK		(0x1 << 25) | 
|  |  | 
|  | #define A375_UNIT_CONTROL_SHIFT		27 | 
|  | #define A375_UNIT_CONTROL_MASK		0x7 | 
|  | #define A375_READOUT_INVERT		BIT(15) | 
|  | #define A375_HW_RESETn			BIT(8) | 
|  |  | 
|  | /* Errata fields */ | 
|  | #define CONTROL0_TSEN_TC_TRIM_MASK	0x7 | 
|  | #define CONTROL0_TSEN_TC_TRIM_VAL	0x3 | 
|  |  | 
|  | #define CONTROL0_TSEN_START		BIT(0) | 
|  | #define CONTROL0_TSEN_RESET		BIT(1) | 
|  | #define CONTROL0_TSEN_ENABLE		BIT(2) | 
|  | #define CONTROL0_TSEN_AVG_BYPASS	BIT(6) | 
|  | #define CONTROL0_TSEN_CHAN_SHIFT	13 | 
|  | #define CONTROL0_TSEN_CHAN_MASK		0xF | 
|  | #define CONTROL0_TSEN_OSR_SHIFT		24 | 
|  | #define CONTROL0_TSEN_OSR_MAX		0x3 | 
|  | #define CONTROL0_TSEN_MODE_SHIFT	30 | 
|  | #define CONTROL0_TSEN_MODE_EXTERNAL	0x2 | 
|  | #define CONTROL0_TSEN_MODE_MASK		0x3 | 
|  |  | 
|  | #define CONTROL1_TSEN_AVG_SHIFT		0 | 
|  | #define CONTROL1_TSEN_AVG_MASK		0x7 | 
|  | #define CONTROL1_EXT_TSEN_SW_RESET	BIT(7) | 
|  | #define CONTROL1_EXT_TSEN_HW_RESETn	BIT(8) | 
|  |  | 
|  | #define STATUS_POLL_PERIOD_US		1000 | 
|  | #define STATUS_POLL_TIMEOUT_US		100000 | 
|  |  | 
|  | struct armada_thermal_data; | 
|  |  | 
|  | /* Marvell EBU Thermal Sensor Dev Structure */ | 
|  | struct armada_thermal_priv { | 
|  | struct device *dev; | 
|  | struct regmap *syscon; | 
|  | char zone_name[THERMAL_NAME_LENGTH]; | 
|  | /* serialize temperature reads/updates */ | 
|  | struct mutex update_lock; | 
|  | struct armada_thermal_data *data; | 
|  | int current_channel; | 
|  | }; | 
|  |  | 
|  | struct armada_thermal_data { | 
|  | /* Initialize the thermal IC */ | 
|  | void (*init)(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv); | 
|  |  | 
|  | /* Formula coeficients: temp = (b - m * reg) / div */ | 
|  | s64 coef_b; | 
|  | s64 coef_m; | 
|  | u32 coef_div; | 
|  | bool inverted; | 
|  | bool signed_sample; | 
|  |  | 
|  | /* Register shift and mask to access the sensor temperature */ | 
|  | unsigned int temp_shift; | 
|  | unsigned int temp_mask; | 
|  | u32 is_valid_bit; | 
|  |  | 
|  | /* Syscon access */ | 
|  | unsigned int syscon_control0_off; | 
|  | unsigned int syscon_control1_off; | 
|  | unsigned int syscon_status_off; | 
|  |  | 
|  | /* One sensor is in the thermal IC, the others are in the CPUs if any */ | 
|  | unsigned int cpu_nr; | 
|  | }; | 
|  |  | 
|  | struct armada_drvdata { | 
|  | enum drvtype { | 
|  | LEGACY, | 
|  | SYSCON | 
|  | } type; | 
|  | union { | 
|  | struct armada_thermal_priv *priv; | 
|  | struct thermal_zone_device *tz; | 
|  | } data; | 
|  | }; | 
|  |  | 
|  | /* | 
|  | * struct armada_thermal_sensor - hold the information of one thermal sensor | 
|  | * @thermal: pointer to the local private structure | 
|  | * @tzd: pointer to the thermal zone device | 
|  | * @id: identifier of the thermal sensor | 
|  | */ | 
|  | struct armada_thermal_sensor { | 
|  | struct armada_thermal_priv *priv; | 
|  | int id; | 
|  | }; | 
|  |  | 
|  | static void armadaxp_init(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | u32 reg; | 
|  |  | 
|  | regmap_read(priv->syscon, data->syscon_control1_off, ®); | 
|  | reg |= PMU_TDC0_OTF_CAL_MASK; | 
|  |  | 
|  | /* Reference calibration value */ | 
|  | reg &= ~PMU_TDC0_REF_CAL_CNT_MASK; | 
|  | reg |= (0xf1 << PMU_TDC0_REF_CAL_CNT_OFFS); | 
|  |  | 
|  | /* Reset the sensor */ | 
|  | reg |= PMU_TDC0_SW_RST_MASK; | 
|  |  | 
|  | regmap_write(priv->syscon, data->syscon_control1_off, reg); | 
|  |  | 
|  | /* Enable the sensor */ | 
|  | regmap_read(priv->syscon, data->syscon_status_off, ®); | 
|  | reg &= ~PMU_TM_DISABLE_MASK; | 
|  | regmap_write(priv->syscon, data->syscon_status_off, reg); | 
|  | } | 
|  |  | 
|  | static void armada370_init(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | u32 reg; | 
|  |  | 
|  | regmap_read(priv->syscon, data->syscon_control1_off, ®); | 
|  | reg |= PMU_TDC0_OTF_CAL_MASK; | 
|  |  | 
|  | /* Reference calibration value */ | 
|  | reg &= ~PMU_TDC0_REF_CAL_CNT_MASK; | 
|  | reg |= (0xf1 << PMU_TDC0_REF_CAL_CNT_OFFS); | 
|  |  | 
|  | /* Reset the sensor */ | 
|  | reg &= ~PMU_TDC0_START_CAL_MASK; | 
|  |  | 
|  | regmap_write(priv->syscon, data->syscon_control1_off, reg); | 
|  |  | 
|  | msleep(10); | 
|  | } | 
|  |  | 
|  | static void armada375_init(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | u32 reg; | 
|  |  | 
|  | regmap_read(priv->syscon, data->syscon_control1_off, ®); | 
|  | reg &= ~(A375_UNIT_CONTROL_MASK << A375_UNIT_CONTROL_SHIFT); | 
|  | reg &= ~A375_READOUT_INVERT; | 
|  | reg &= ~A375_HW_RESETn; | 
|  | regmap_write(priv->syscon, data->syscon_control1_off, reg); | 
|  |  | 
|  | msleep(20); | 
|  |  | 
|  | reg |= A375_HW_RESETn; | 
|  | regmap_write(priv->syscon, data->syscon_control1_off, reg); | 
|  |  | 
|  | msleep(50); | 
|  | } | 
|  |  | 
|  | static int armada_wait_sensor_validity(struct armada_thermal_priv *priv) | 
|  | { | 
|  | u32 reg; | 
|  |  | 
|  | return regmap_read_poll_timeout(priv->syscon, | 
|  | priv->data->syscon_status_off, reg, | 
|  | reg & priv->data->is_valid_bit, | 
|  | STATUS_POLL_PERIOD_US, | 
|  | STATUS_POLL_TIMEOUT_US); | 
|  | } | 
|  |  | 
|  | static void armada380_init(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | u32 reg; | 
|  |  | 
|  | /* Disable the HW/SW reset */ | 
|  | regmap_read(priv->syscon, data->syscon_control1_off, ®); | 
|  | reg |= CONTROL1_EXT_TSEN_HW_RESETn; | 
|  | reg &= ~CONTROL1_EXT_TSEN_SW_RESET; | 
|  | regmap_write(priv->syscon, data->syscon_control1_off, reg); | 
|  |  | 
|  | /* Set Tsen Tc Trim to correct default value (errata #132698) */ | 
|  | regmap_read(priv->syscon, data->syscon_control0_off, ®); | 
|  | reg &= ~CONTROL0_TSEN_TC_TRIM_MASK; | 
|  | reg |= CONTROL0_TSEN_TC_TRIM_VAL; | 
|  | regmap_write(priv->syscon, data->syscon_control0_off, reg); | 
|  | } | 
|  |  | 
|  | static void armada_ap806_init(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | u32 reg; | 
|  |  | 
|  | regmap_read(priv->syscon, data->syscon_control0_off, ®); | 
|  | reg &= ~CONTROL0_TSEN_RESET; | 
|  | reg |= CONTROL0_TSEN_START | CONTROL0_TSEN_ENABLE; | 
|  |  | 
|  | /* Sample every ~2ms */ | 
|  | reg |= CONTROL0_TSEN_OSR_MAX << CONTROL0_TSEN_OSR_SHIFT; | 
|  |  | 
|  | /* Enable average (2 samples by default) */ | 
|  | reg &= ~CONTROL0_TSEN_AVG_BYPASS; | 
|  |  | 
|  | regmap_write(priv->syscon, data->syscon_control0_off, reg); | 
|  | } | 
|  |  | 
|  | static void armada_cp110_init(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | u32 reg; | 
|  |  | 
|  | armada380_init(pdev, priv); | 
|  |  | 
|  | /* Sample every ~2ms */ | 
|  | regmap_read(priv->syscon, data->syscon_control0_off, ®); | 
|  | reg |= CONTROL0_TSEN_OSR_MAX << CONTROL0_TSEN_OSR_SHIFT; | 
|  | regmap_write(priv->syscon, data->syscon_control0_off, reg); | 
|  |  | 
|  | /* Average the output value over 2^1 = 2 samples */ | 
|  | regmap_read(priv->syscon, data->syscon_control1_off, ®); | 
|  | reg &= ~CONTROL1_TSEN_AVG_MASK << CONTROL1_TSEN_AVG_SHIFT; | 
|  | reg |= 1 << CONTROL1_TSEN_AVG_SHIFT; | 
|  | regmap_write(priv->syscon, data->syscon_control1_off, reg); | 
|  | } | 
|  |  | 
|  | static bool armada_is_valid(struct armada_thermal_priv *priv) | 
|  | { | 
|  | u32 reg; | 
|  |  | 
|  | if (!priv->data->is_valid_bit) | 
|  | return true; | 
|  |  | 
|  | regmap_read(priv->syscon, priv->data->syscon_status_off, ®); | 
|  |  | 
|  | return reg & priv->data->is_valid_bit; | 
|  | } | 
|  |  | 
|  | /* There is currently no board with more than one sensor per channel */ | 
|  | static int armada_select_channel(struct armada_thermal_priv *priv, int channel) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | u32 ctrl0; | 
|  |  | 
|  | if (channel < 0 || channel > priv->data->cpu_nr) | 
|  | return -EINVAL; | 
|  |  | 
|  | if (priv->current_channel == channel) | 
|  | return 0; | 
|  |  | 
|  | /* Stop the measurements */ | 
|  | regmap_read(priv->syscon, data->syscon_control0_off, &ctrl0); | 
|  | ctrl0 &= ~CONTROL0_TSEN_START; | 
|  | regmap_write(priv->syscon, data->syscon_control0_off, ctrl0); | 
|  |  | 
|  | /* Reset the mode, internal sensor will be automatically selected */ | 
|  | ctrl0 &= ~(CONTROL0_TSEN_MODE_MASK << CONTROL0_TSEN_MODE_SHIFT); | 
|  |  | 
|  | /* Other channels are external and should be selected accordingly */ | 
|  | if (channel) { | 
|  | /* Change the mode to external */ | 
|  | ctrl0 |= CONTROL0_TSEN_MODE_EXTERNAL << | 
|  | CONTROL0_TSEN_MODE_SHIFT; | 
|  | /* Select the sensor */ | 
|  | ctrl0 &= ~(CONTROL0_TSEN_CHAN_MASK << CONTROL0_TSEN_CHAN_SHIFT); | 
|  | ctrl0 |= (channel - 1) << CONTROL0_TSEN_CHAN_SHIFT; | 
|  | } | 
|  |  | 
|  | /* Actually set the mode/channel */ | 
|  | regmap_write(priv->syscon, data->syscon_control0_off, ctrl0); | 
|  | priv->current_channel = channel; | 
|  |  | 
|  | /* Re-start the measurements */ | 
|  | ctrl0 |= CONTROL0_TSEN_START; | 
|  | regmap_write(priv->syscon, data->syscon_control0_off, ctrl0); | 
|  |  | 
|  | /* | 
|  | * The IP has a latency of ~15ms, so after updating the selected source, | 
|  | * we must absolutely wait for the sensor validity bit to ensure we read | 
|  | * actual data. | 
|  | */ | 
|  | if (armada_wait_sensor_validity(priv)) { | 
|  | dev_err(priv->dev, | 
|  | "Temperature sensor reading not valid\n"); | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int armada_read_sensor(struct armada_thermal_priv *priv, int *temp) | 
|  | { | 
|  | u32 reg, div; | 
|  | s64 sample, b, m; | 
|  |  | 
|  | regmap_read(priv->syscon, priv->data->syscon_status_off, ®); | 
|  | reg = (reg >> priv->data->temp_shift) & priv->data->temp_mask; | 
|  | if (priv->data->signed_sample) | 
|  | /* The most significant bit is the sign bit */ | 
|  | sample = sign_extend32(reg, fls(priv->data->temp_mask) - 1); | 
|  | else | 
|  | sample = reg; | 
|  |  | 
|  | /* Get formula coeficients */ | 
|  | b = priv->data->coef_b; | 
|  | m = priv->data->coef_m; | 
|  | div = priv->data->coef_div; | 
|  |  | 
|  | if (priv->data->inverted) | 
|  | *temp = div_s64((m * sample) - b, div); | 
|  | else | 
|  | *temp = div_s64(b - (m * sample), div); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int armada_get_temp_legacy(struct thermal_zone_device *thermal, | 
|  | int *temp) | 
|  | { | 
|  | struct armada_thermal_priv *priv = thermal->devdata; | 
|  | int ret; | 
|  |  | 
|  | /* Valid check */ | 
|  | if (!armada_is_valid(priv)) { | 
|  | dev_err(priv->dev, | 
|  | "Temperature sensor reading not valid\n"); | 
|  | return -EIO; | 
|  | } | 
|  |  | 
|  | /* Do the actual reading */ | 
|  | ret = armada_read_sensor(priv, temp); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static struct thermal_zone_device_ops legacy_ops = { | 
|  | .get_temp = armada_get_temp_legacy, | 
|  | }; | 
|  |  | 
|  | static int armada_get_temp(void *_sensor, int *temp) | 
|  | { | 
|  | struct armada_thermal_sensor *sensor = _sensor; | 
|  | struct armada_thermal_priv *priv = sensor->priv; | 
|  | int ret; | 
|  |  | 
|  | mutex_lock(&priv->update_lock); | 
|  |  | 
|  | /* Select the desired channel */ | 
|  | ret = armada_select_channel(priv, sensor->id); | 
|  | if (ret) | 
|  | goto unlock_mutex; | 
|  |  | 
|  | /* Do the actual reading */ | 
|  | ret = armada_read_sensor(priv, temp); | 
|  |  | 
|  | unlock_mutex: | 
|  | mutex_unlock(&priv->update_lock); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static struct thermal_zone_of_device_ops of_ops = { | 
|  | .get_temp = armada_get_temp, | 
|  | }; | 
|  |  | 
|  | static const struct armada_thermal_data armadaxp_data = { | 
|  | .init = armadaxp_init, | 
|  | .temp_shift = 10, | 
|  | .temp_mask = 0x1ff, | 
|  | .coef_b = 3153000000ULL, | 
|  | .coef_m = 10000000ULL, | 
|  | .coef_div = 13825, | 
|  | .syscon_status_off = 0xb0, | 
|  | .syscon_control1_off = 0xd0, | 
|  | }; | 
|  |  | 
|  | static const struct armada_thermal_data armada370_data = { | 
|  | .init = armada370_init, | 
|  | .is_valid_bit = BIT(9), | 
|  | .temp_shift = 10, | 
|  | .temp_mask = 0x1ff, | 
|  | .coef_b = 3153000000ULL, | 
|  | .coef_m = 10000000ULL, | 
|  | .coef_div = 13825, | 
|  | .syscon_status_off = 0x0, | 
|  | .syscon_control1_off = 0x4, | 
|  | }; | 
|  |  | 
|  | static const struct armada_thermal_data armada375_data = { | 
|  | .init = armada375_init, | 
|  | .is_valid_bit = BIT(10), | 
|  | .temp_shift = 0, | 
|  | .temp_mask = 0x1ff, | 
|  | .coef_b = 3171900000ULL, | 
|  | .coef_m = 10000000ULL, | 
|  | .coef_div = 13616, | 
|  | .syscon_status_off = 0x78, | 
|  | .syscon_control0_off = 0x7c, | 
|  | .syscon_control1_off = 0x80, | 
|  | }; | 
|  |  | 
|  | static const struct armada_thermal_data armada380_data = { | 
|  | .init = armada380_init, | 
|  | .is_valid_bit = BIT(10), | 
|  | .temp_shift = 0, | 
|  | .temp_mask = 0x3ff, | 
|  | .coef_b = 1172499100ULL, | 
|  | .coef_m = 2000096ULL, | 
|  | .coef_div = 4201, | 
|  | .inverted = true, | 
|  | .syscon_control0_off = 0x70, | 
|  | .syscon_control1_off = 0x74, | 
|  | .syscon_status_off = 0x78, | 
|  | }; | 
|  |  | 
|  | static const struct armada_thermal_data armada_ap806_data = { | 
|  | .init = armada_ap806_init, | 
|  | .is_valid_bit = BIT(16), | 
|  | .temp_shift = 0, | 
|  | .temp_mask = 0x3ff, | 
|  | .coef_b = -150000LL, | 
|  | .coef_m = 423ULL, | 
|  | .coef_div = 1, | 
|  | .inverted = true, | 
|  | .signed_sample = true, | 
|  | .syscon_control0_off = 0x84, | 
|  | .syscon_control1_off = 0x88, | 
|  | .syscon_status_off = 0x8C, | 
|  | .cpu_nr = 4, | 
|  | }; | 
|  |  | 
|  | static const struct armada_thermal_data armada_cp110_data = { | 
|  | .init = armada_cp110_init, | 
|  | .is_valid_bit = BIT(10), | 
|  | .temp_shift = 0, | 
|  | .temp_mask = 0x3ff, | 
|  | .coef_b = 1172499100ULL, | 
|  | .coef_m = 2000096ULL, | 
|  | .coef_div = 4201, | 
|  | .inverted = true, | 
|  | .syscon_control0_off = 0x70, | 
|  | .syscon_control1_off = 0x74, | 
|  | .syscon_status_off = 0x78, | 
|  | }; | 
|  |  | 
|  | static const struct of_device_id armada_thermal_id_table[] = { | 
|  | { | 
|  | .compatible = "marvell,armadaxp-thermal", | 
|  | .data       = &armadaxp_data, | 
|  | }, | 
|  | { | 
|  | .compatible = "marvell,armada370-thermal", | 
|  | .data       = &armada370_data, | 
|  | }, | 
|  | { | 
|  | .compatible = "marvell,armada375-thermal", | 
|  | .data       = &armada375_data, | 
|  | }, | 
|  | { | 
|  | .compatible = "marvell,armada380-thermal", | 
|  | .data       = &armada380_data, | 
|  | }, | 
|  | { | 
|  | .compatible = "marvell,armada-ap806-thermal", | 
|  | .data       = &armada_ap806_data, | 
|  | }, | 
|  | { | 
|  | .compatible = "marvell,armada-cp110-thermal", | 
|  | .data       = &armada_cp110_data, | 
|  | }, | 
|  | { | 
|  | /* sentinel */ | 
|  | }, | 
|  | }; | 
|  | MODULE_DEVICE_TABLE(of, armada_thermal_id_table); | 
|  |  | 
|  | static const struct regmap_config armada_thermal_regmap_config = { | 
|  | .reg_bits = 32, | 
|  | .reg_stride = 4, | 
|  | .val_bits = 32, | 
|  | .fast_io = true, | 
|  | }; | 
|  |  | 
|  | static int armada_thermal_probe_legacy(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | struct armada_thermal_data *data = priv->data; | 
|  | struct resource *res; | 
|  | void __iomem *base; | 
|  |  | 
|  | /* First memory region points towards the status register */ | 
|  | res = platform_get_resource(pdev, IORESOURCE_MEM, 0); | 
|  | if (!res) | 
|  | return -EIO; | 
|  |  | 
|  | /* | 
|  | * Edit the resource start address and length to map over all the | 
|  | * registers, instead of pointing at them one by one. | 
|  | */ | 
|  | res->start -= data->syscon_status_off; | 
|  | res->end = res->start + max(data->syscon_status_off, | 
|  | max(data->syscon_control0_off, | 
|  | data->syscon_control1_off)) + | 
|  | sizeof(unsigned int) - 1; | 
|  |  | 
|  | base = devm_ioremap_resource(&pdev->dev, res); | 
|  | if (IS_ERR(base)) | 
|  | return PTR_ERR(base); | 
|  |  | 
|  | priv->syscon = devm_regmap_init_mmio(&pdev->dev, base, | 
|  | &armada_thermal_regmap_config); | 
|  | if (IS_ERR(priv->syscon)) | 
|  | return PTR_ERR(priv->syscon); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int armada_thermal_probe_syscon(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | priv->syscon = syscon_node_to_regmap(pdev->dev.parent->of_node); | 
|  | if (IS_ERR(priv->syscon)) | 
|  | return PTR_ERR(priv->syscon); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static void armada_set_sane_name(struct platform_device *pdev, | 
|  | struct armada_thermal_priv *priv) | 
|  | { | 
|  | const char *name = dev_name(&pdev->dev); | 
|  | char *insane_char; | 
|  |  | 
|  | if (strlen(name) > THERMAL_NAME_LENGTH) { | 
|  | /* | 
|  | * When inside a system controller, the device name has the | 
|  | * form: f06f8000.system-controller:ap-thermal so stripping | 
|  | * after the ':' should give us a shorter but meaningful name. | 
|  | */ | 
|  | name = strrchr(name, ':'); | 
|  | if (!name) | 
|  | name = "armada_thermal"; | 
|  | else | 
|  | name++; | 
|  | } | 
|  |  | 
|  | /* Save the name locally */ | 
|  | strncpy(priv->zone_name, name, THERMAL_NAME_LENGTH - 1); | 
|  | priv->zone_name[THERMAL_NAME_LENGTH - 1] = '\0'; | 
|  |  | 
|  | /* Then check there are no '-' or hwmon core will complain */ | 
|  | do { | 
|  | insane_char = strpbrk(priv->zone_name, "-"); | 
|  | if (insane_char) | 
|  | *insane_char = '_'; | 
|  | } while (insane_char); | 
|  | } | 
|  |  | 
|  | static int armada_thermal_probe(struct platform_device *pdev) | 
|  | { | 
|  | struct thermal_zone_device *tz; | 
|  | struct armada_thermal_sensor *sensor; | 
|  | struct armada_drvdata *drvdata; | 
|  | const struct of_device_id *match; | 
|  | struct armada_thermal_priv *priv; | 
|  | int sensor_id; | 
|  | int ret; | 
|  |  | 
|  | match = of_match_device(armada_thermal_id_table, &pdev->dev); | 
|  | if (!match) | 
|  | return -ENODEV; | 
|  |  | 
|  | priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL); | 
|  | if (!priv) | 
|  | return -ENOMEM; | 
|  |  | 
|  | drvdata = devm_kzalloc(&pdev->dev, sizeof(*drvdata), GFP_KERNEL); | 
|  | if (!drvdata) | 
|  | return -ENOMEM; | 
|  |  | 
|  | priv->dev = &pdev->dev; | 
|  | priv->data = (struct armada_thermal_data *)match->data; | 
|  |  | 
|  | mutex_init(&priv->update_lock); | 
|  |  | 
|  | /* | 
|  | * Legacy DT bindings only described "control1" register (also referred | 
|  | * as "control MSB" on old documentation). Then, bindings moved to cover | 
|  | * "control0/control LSB" and "control1/control MSB" registers within | 
|  | * the same resource, which was then of size 8 instead of 4. | 
|  | * | 
|  | * The logic of defining sporadic registers is broken. For instance, it | 
|  | * blocked the addition of the overheat interrupt feature that needed | 
|  | * another resource somewhere else in the same memory area. One solution | 
|  | * is to define an overall system controller and put the thermal node | 
|  | * into it, which requires the use of regmaps across all the driver. | 
|  | */ | 
|  | if (IS_ERR(syscon_node_to_regmap(pdev->dev.parent->of_node))) { | 
|  | /* Ensure device name is correct for the thermal core */ | 
|  | armada_set_sane_name(pdev, priv); | 
|  |  | 
|  | ret = armada_thermal_probe_legacy(pdev, priv); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | priv->data->init(pdev, priv); | 
|  |  | 
|  | /* Wait the sensors to be valid */ | 
|  | armada_wait_sensor_validity(priv); | 
|  |  | 
|  | tz = thermal_zone_device_register(priv->zone_name, 0, 0, priv, | 
|  | &legacy_ops, NULL, 0, 0); | 
|  | if (IS_ERR(tz)) { | 
|  | dev_err(&pdev->dev, | 
|  | "Failed to register thermal zone device\n"); | 
|  | return PTR_ERR(tz); | 
|  | } | 
|  |  | 
|  | drvdata->type = LEGACY; | 
|  | drvdata->data.tz = tz; | 
|  | platform_set_drvdata(pdev, drvdata); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | ret = armada_thermal_probe_syscon(pdev, priv); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | priv->current_channel = -1; | 
|  | priv->data->init(pdev, priv); | 
|  | drvdata->type = SYSCON; | 
|  | drvdata->data.priv = priv; | 
|  | platform_set_drvdata(pdev, drvdata); | 
|  |  | 
|  | /* | 
|  | * There is one channel for the IC and one per CPU (if any), each | 
|  | * channel has one sensor. | 
|  | */ | 
|  | for (sensor_id = 0; sensor_id <= priv->data->cpu_nr; sensor_id++) { | 
|  | sensor = devm_kzalloc(&pdev->dev, | 
|  | sizeof(struct armada_thermal_sensor), | 
|  | GFP_KERNEL); | 
|  | if (!sensor) | 
|  | return -ENOMEM; | 
|  |  | 
|  | /* Register the sensor */ | 
|  | sensor->priv = priv; | 
|  | sensor->id = sensor_id; | 
|  | tz = devm_thermal_zone_of_sensor_register(&pdev->dev, | 
|  | sensor->id, sensor, | 
|  | &of_ops); | 
|  | if (IS_ERR(tz)) { | 
|  | dev_info(&pdev->dev, "Thermal sensor %d unavailable\n", | 
|  | sensor_id); | 
|  | devm_kfree(&pdev->dev, sensor); | 
|  | continue; | 
|  | } | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int armada_thermal_exit(struct platform_device *pdev) | 
|  | { | 
|  | struct armada_drvdata *drvdata = platform_get_drvdata(pdev); | 
|  |  | 
|  | if (drvdata->type == LEGACY) | 
|  | thermal_zone_device_unregister(drvdata->data.tz); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static struct platform_driver armada_thermal_driver = { | 
|  | .probe = armada_thermal_probe, | 
|  | .remove = armada_thermal_exit, | 
|  | .driver = { | 
|  | .name = "armada_thermal", | 
|  | .of_match_table = armada_thermal_id_table, | 
|  | }, | 
|  | }; | 
|  |  | 
|  | module_platform_driver(armada_thermal_driver); | 
|  |  | 
|  | MODULE_AUTHOR("Ezequiel Garcia <ezequiel.garcia@free-electrons.com>"); | 
|  | MODULE_DESCRIPTION("Marvell EBU Armada SoCs thermal driver"); | 
|  | MODULE_LICENSE("GPL v2"); |