linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support
@ 2018-11-27  6:01 Anson Huang
  2018-11-27  6:01 ` [PATCH 2/2] dt-bindings: thermal: add binding doc for i.MX system controller thermal driver Anson Huang
  2018-11-27 13:16 ` [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support kbuild test robot
  0 siblings, 2 replies; 3+ messages in thread
From: Anson Huang @ 2018-11-27  6:01 UTC (permalink / raw)
  To: rui.zhang, edubezval, daniel.lezcano, robh+dt, mark.rutland,
	linux-pm, devicetree, linux-kernel
  Cc: dl-linux-imx

i.MX8QXP is an ARMv8 SoC which has a Cortex-M4 system controller
inside, the system controller is in charge of controlling power,
clock and thermal sensors etc..

This patch adds i.MX system controller thermal driver support,
Linux kernel has to communicate with system controller via MU
(message unit) IPC to get each thermal sensor's temperature,
it supports multiple sensors which are passed from device tree,
please see the binding doc for details.

Signed-off-by: Anson Huang <Anson.Huang@nxp.com>
---
 drivers/thermal/Kconfig          |  11 ++
 drivers/thermal/Makefile         |   1 +
 drivers/thermal/imx_sc_thermal.c | 221 +++++++++++++++++++++++++++++++++++++++
 3 files changed, 233 insertions(+)
 create mode 100644 drivers/thermal/imx_sc_thermal.c

diff --git a/drivers/thermal/Kconfig b/drivers/thermal/Kconfig
index 0e69edc..84e850c 100644
--- a/drivers/thermal/Kconfig
+++ b/drivers/thermal/Kconfig
@@ -222,6 +222,17 @@ config IMX_THERMAL
 	  cpufreq is used as the cooling device to throttle CPUs when the
 	  passive trip is crossed.
 
+config IMX_SC_THERMAL
+	tristate "Temperature sensor driver for NXP i.MX SoCs with System Controller"
+	depends on ARCH_MXC || COMPILE_TEST
+	depends on OF
+	help
+	  Support for Temperature Monitor (TEMPMON) found on NXP i.MX SoCs with
+	  system controller inside, Linux kernel has to communicate with system
+	  controller via MU (message unit) IPC to get temperature from thermal
+	  sensor. It supports one critical trip point and one
+	  passive trip point for each thermal sensor.
+
 config MAX77620_THERMAL
 	tristate "Temperature sensor driver for Maxim MAX77620 PMIC"
 	depends on MFD_MAX77620
diff --git a/drivers/thermal/Makefile b/drivers/thermal/Makefile
index 610344e..1b13f6a 100644
--- a/drivers/thermal/Makefile
+++ b/drivers/thermal/Makefile
@@ -41,6 +41,7 @@ obj-$(CONFIG_DB8500_THERMAL)	+= db8500_thermal.o
 obj-$(CONFIG_ARMADA_THERMAL)	+= armada_thermal.o
 obj-$(CONFIG_TANGO_THERMAL)	+= tango_thermal.o
 obj-$(CONFIG_IMX_THERMAL)	+= imx_thermal.o
+obj-$(CONFIG_IMX_SC_THERMAL)	+= imx_sc_thermal.o
 obj-$(CONFIG_MAX77620_THERMAL)	+= max77620_thermal.o
 obj-$(CONFIG_QORIQ_THERMAL)	+= qoriq_thermal.o
 obj-$(CONFIG_DA9062_THERMAL)	+= da9062-thermal.o
diff --git a/drivers/thermal/imx_sc_thermal.c b/drivers/thermal/imx_sc_thermal.c
new file mode 100644
index 0000000..2b34ce2
--- /dev/null
+++ b/drivers/thermal/imx_sc_thermal.c
@@ -0,0 +1,221 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Copyright 2018 NXP.
+ *
+ * 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.
+ *
+ */
+
+#include <linux/err.h>
+#include <linux/firmware/imx/sci.h>
+#include <linux/module.h>
+#include <linux/of.h>
+#include <linux/of_device.h>
+#include <linux/platform_device.h>
+#include <linux/slab.h>
+#include <linux/thermal.h>
+
+#include "thermal_core.h"
+
+#define IMX_SC_MISC_FUNC_GET_TEMP	13
+#define IMX_SC_C_TEMP			0
+
+struct imx_sc_ipc *thermal_ipc_handle;
+
+struct imx_sc_sensor {
+	struct thermal_zone_device *tzd;
+	unsigned int resource_id;
+};
+
+struct imx_sc_thermal_data {
+	struct imx_sc_sensor *sensor;
+};
+
+struct imx_sc_msg_req_misc_get_temp {
+	struct imx_sc_rpc_msg hdr;
+	u16 resource_id;
+	u8 type;
+} __packed;
+
+struct imx_sc_msg_resp_misc_get_temp {
+	struct imx_sc_rpc_msg hdr;
+	u16 celsius;
+	u8 tenths;
+} __packed;
+
+static int imx_sc_thermal_get_temp(void *data, int *temp)
+{
+	struct imx_sc_msg_resp_misc_get_temp *resp;
+	struct imx_sc_msg_req_misc_get_temp msg;
+	struct imx_sc_rpc_msg *hdr = &msg.hdr;
+	struct imx_sc_sensor *sensor = data;
+	int ret;
+
+	msg.resource_id = sensor->resource_id;
+	msg.type = IMX_SC_C_TEMP;
+
+	hdr->ver = IMX_SC_RPC_VERSION;
+	hdr->svc = IMX_SC_RPC_SVC_MISC;
+	hdr->func = IMX_SC_MISC_FUNC_GET_TEMP;
+	hdr->size = 2;
+
+	ret = imx_scu_call_rpc(thermal_ipc_handle, &msg, true);
+	if (ret) {
+		pr_err("read temp sensor %d failed, ret %d\n",
+			sensor->resource_id, ret);
+		return ret;
+	}
+
+	resp = (struct imx_sc_msg_resp_misc_get_temp *)&msg;
+	*temp = resp->celsius * 1000 + resp->tenths * 100;
+
+	return 0;
+}
+
+static const struct thermal_zone_of_device_ops imx_sc_thermal_ops = {
+	.get_temp = imx_sc_thermal_get_temp,
+};
+
+static int imx_sc_thermal_register_sensor(struct platform_device *pdev,
+					  struct imx_sc_sensor *sensor)
+{
+	struct thermal_zone_device *tzd;
+
+	tzd = devm_thermal_zone_of_sensor_register(&pdev->dev,
+						   sensor->resource_id,
+						   sensor,
+						   &imx_sc_thermal_ops);
+	if (IS_ERR(tzd)) {
+		dev_err(&pdev->dev, "failed to register sensor: %d\n",
+			sensor->resource_id);
+		return -EINVAL;
+	}
+
+	sensor->tzd = tzd;
+
+	return 0;
+}
+
+static int imx_sc_thermal_get_sensor_id(struct device_node *sensor_np)
+{
+	struct of_phandle_args sensor_specs;
+	int ret;
+
+	ret = of_parse_phandle_with_args(sensor_np, "thermal-sensors",
+			"#thermal-sensor-cells",
+			0, &sensor_specs);
+	if (ret)
+		return ret;
+
+	if (sensor_specs.args_count >= 1) {
+		ret = sensor_specs.args[0];
+		WARN(sensor_specs.args_count > 1,
+				"%pOFn: too many cells in sensor specifier %d\n",
+				sensor_specs.np, sensor_specs.args_count);
+	} else {
+		ret = 0;
+	}
+
+	return ret;
+}
+
+static int imx_sc_thermal_probe(struct platform_device *pdev)
+{
+	struct device_node *np = pdev->dev.of_node;
+	struct device_node *sensor_np = NULL;
+	struct imx_sc_thermal_data *data;
+	struct imx_sc_sensor *sensors;
+	u32 sensor_num;
+	int ret, i;
+
+	ret = imx_scu_get_handle(&thermal_ipc_handle);
+	if (ret) {
+		if (ret == -EPROBE_DEFER)
+			return ret;
+
+		dev_err(&pdev->dev, "failed to get ipc handle: %d!\n", ret);
+		return ret;
+	}
+
+	data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL);
+	if (!data)
+		return -ENOMEM;
+
+	ret = of_property_read_u32(np, "tsens-num", &sensor_num);
+	if (ret || !sensor_num) {
+		dev_err(&pdev->dev, "failed to get valid temp sensor number!\n");
+		ret = -EINVAL;
+		goto free_data;
+	}
+
+	sensors = devm_kzalloc(&pdev->dev, sizeof(*data->sensor) * sensor_num,
+			       GFP_KERNEL);
+	if (!sensors) {
+		ret = -ENOMEM;
+		goto free_data;
+	}
+
+	data->sensor = sensors;
+
+	np = of_find_node_by_name(NULL, "thermal-zones");
+	if (!np) {
+		ret = -ENODEV;
+		goto free_sensors;
+	}
+
+	for (i = 0; i < sensor_num; i++) {
+		struct imx_sc_sensor *sensor = &data->sensor[i];
+
+		sensor_np = of_get_next_child(np, sensor_np);
+		sensor->resource_id = imx_sc_thermal_get_sensor_id(sensor_np);
+		if (sensor->resource_id < 0) {
+			dev_err(&pdev->dev, "invalid sensor resource id: %d\n",
+				sensor->resource_id);
+			ret = sensor->resource_id;
+			goto put_node;
+		}
+
+		ret = imx_sc_thermal_register_sensor(pdev, sensor);
+		if (ret) {
+			dev_err(&pdev->dev, "failed to register thermal sensor: %d\n",
+				ret);
+			goto put_node;
+		}
+	}
+
+	of_node_put(sensor_np);
+	of_node_put(np);
+
+	return 0;
+
+put_node:
+	of_node_put(np);
+	of_node_put(sensor_np);
+free_sensors:
+	kfree(sensors);
+free_data:
+	kfree(data);
+
+	return ret;
+}
+
+static const struct of_device_id imx_sc_thermal_table[] = {
+	{ .compatible = "nxp,imx8qxp-sc-thermal", },
+	{},
+};
+MODULE_DEVICE_TABLE(of, imx_sc_thermal_table);
+
+static struct platform_driver imx_sc_thermal_driver = {
+		.probe = imx_sc_thermal_probe,
+		.driver = {
+			.name = "imx-sc-thermal",
+			.of_match_table = imx_sc_thermal_table,
+		},
+};
+module_platform_driver(imx_sc_thermal_driver);
+
+MODULE_AUTHOR("Anson Huang <Anson.Huang@nxp.com>");
+MODULE_DESCRIPTION("Thermal driver for NXP i.MX SoCs with system controller");
+MODULE_LICENSE("GPL v2");
-- 
2.7.4


^ permalink raw reply related	[flat|nested] 3+ messages in thread

* [PATCH 2/2] dt-bindings: thermal: add binding doc for i.MX system controller thermal driver
  2018-11-27  6:01 [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support Anson Huang
@ 2018-11-27  6:01 ` Anson Huang
  2018-11-27 13:16 ` [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support kbuild test robot
  1 sibling, 0 replies; 3+ messages in thread
From: Anson Huang @ 2018-11-27  6:01 UTC (permalink / raw)
  To: rui.zhang, edubezval, daniel.lezcano, robh+dt, mark.rutland,
	linux-pm, devicetree, linux-kernel
  Cc: dl-linux-imx

NXP i.MX8QXP is an ARMv8 SoC with a Cortex-M4 core inside as
system controller, the system controller is in charge of system
power, clock and thermal sensors etc. management, Linux kernel
has to communicate with system controller via MU (message unit)
IPC to get temperature from thermal sensors, this patch adds
binding doc for i.MX system controller thermal driver.

Signed-off-by: Anson Huang <Anson.Huang@nxp.com>
---
 .../devicetree/bindings/thermal/imx-sc-thermal.txt | 33 ++++++++++++++++++++++
 1 file changed, 33 insertions(+)
 create mode 100644 Documentation/devicetree/bindings/thermal/imx-sc-thermal.txt

diff --git a/Documentation/devicetree/bindings/thermal/imx-sc-thermal.txt b/Documentation/devicetree/bindings/thermal/imx-sc-thermal.txt
new file mode 100644
index 0000000..2f5b0a2
--- /dev/null
+++ b/Documentation/devicetree/bindings/thermal/imx-sc-thermal.txt
@@ -0,0 +1,33 @@
+* Temperature Monitor (TEMPMON) on NXP i.MX SoCs with System Controller
+
+Required properties:
+- compatible : Must be "nxp,imx8qxp-sc-thermal";
+- tsens-num : Total number of thermal sensors supported;
+- #thermal-sensor-cells : Should be 1. See ./thermal.txt for a description.
+
+Example:
+tsens: thermal-sensor {
+	compatible = "nxp,imx8qxp-sc-thermal";
+	tsens-num = <1>;
+	#thermal-sensor-cells = <1>;
+};
+
+thermal_zones: thermal-zones {
+	cpu-thermal0 {
+		polling-delay-passive = <250>;
+		polling-delay = <2000>;
+		thermal-sensors = <&tsens 355>;
+		trips {
+			cpu_alert0: trip0 {
+				temperature = <107000>;
+				hysteresis = <2000>;
+				type = "passive";
+			};
+			cpu_crit0: trip1 {
+				temperature = <127000>;
+				hysteresis = <2000>;
+				type = "critical";
+			};
+		};
+	};
+};
-- 
2.7.4


^ permalink raw reply related	[flat|nested] 3+ messages in thread

* Re: [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support
  2018-11-27  6:01 [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support Anson Huang
  2018-11-27  6:01 ` [PATCH 2/2] dt-bindings: thermal: add binding doc for i.MX system controller thermal driver Anson Huang
@ 2018-11-27 13:16 ` kbuild test robot
  1 sibling, 0 replies; 3+ messages in thread
From: kbuild test robot @ 2018-11-27 13:16 UTC (permalink / raw)
  To: Anson Huang
  Cc: kbuild-all, rui.zhang, edubezval, daniel.lezcano, robh+dt,
	mark.rutland, linux-pm, devicetree, linux-kernel, dl-linux-imx

[-- Attachment #1: Type: text/plain, Size: 7008 bytes --]

Hi Anson,

Thank you for the patch! Yet something to improve:

[auto build test ERROR on soc-thermal/next]
[also build test ERROR on v4.20-rc4 next-20181126]
[if your patch is applied to the wrong git tree, please drop us a note to help improve the system]

url:    https://github.com/0day-ci/linux/commits/Anson-Huang/thermal-imx_sc-add-i-MX-system-controller-thermal-support/20181127-141933
base:   https://git.kernel.org/pub/scm/linux/kernel/git/evalenti/linux-soc-thermal.git next
config: i386-allmodconfig (attached as .config)
compiler: gcc-7 (Debian 7.3.0-1) 7.3.0
reproduce:
        # save the attached .config to linux build tree
        make ARCH=i386 

All errors (new ones prefixed by >>):

>> drivers/thermal/imx_sc_thermal.c:12:10: fatal error: linux/firmware/imx/sci.h: No such file or directory
    #include <linux/firmware/imx/sci.h>
             ^~~~~~~~~~~~~~~~~~~~~~~~~~
   compilation terminated.

coccinelle warnings: (new ones prefixed by >>)

>> drivers/thermal/imx_sc_thermal.c:197:1-6: WARNING: invalid free of devm_ allocated data
   drivers/thermal/imx_sc_thermal.c:199:1-6: WARNING: invalid free of devm_ allocated data
--
>> drivers/thermal/imx_sc_thermal.c:173:6-25: WARNING: Unsigned expression compared with zero: sensor -> resource_id < 0

vim +12 drivers/thermal/imx_sc_thermal.c

  > 12	#include <linux/firmware/imx/sci.h>
    13	#include <linux/module.h>
    14	#include <linux/of.h>
    15	#include <linux/of_device.h>
    16	#include <linux/platform_device.h>
    17	#include <linux/slab.h>
    18	#include <linux/thermal.h>
    19	
    20	#include "thermal_core.h"
    21	
    22	#define IMX_SC_MISC_FUNC_GET_TEMP	13
    23	#define IMX_SC_C_TEMP			0
    24	
    25	struct imx_sc_ipc *thermal_ipc_handle;
    26	
    27	struct imx_sc_sensor {
    28		struct thermal_zone_device *tzd;
    29		unsigned int resource_id;
    30	};
    31	
    32	struct imx_sc_thermal_data {
    33		struct imx_sc_sensor *sensor;
    34	};
    35	
    36	struct imx_sc_msg_req_misc_get_temp {
    37		struct imx_sc_rpc_msg hdr;
    38		u16 resource_id;
    39		u8 type;
    40	} __packed;
    41	
    42	struct imx_sc_msg_resp_misc_get_temp {
    43		struct imx_sc_rpc_msg hdr;
    44		u16 celsius;
    45		u8 tenths;
    46	} __packed;
    47	
    48	static int imx_sc_thermal_get_temp(void *data, int *temp)
    49	{
    50		struct imx_sc_msg_resp_misc_get_temp *resp;
    51		struct imx_sc_msg_req_misc_get_temp msg;
    52		struct imx_sc_rpc_msg *hdr = &msg.hdr;
    53		struct imx_sc_sensor *sensor = data;
    54		int ret;
    55	
    56		msg.resource_id = sensor->resource_id;
    57		msg.type = IMX_SC_C_TEMP;
    58	
    59		hdr->ver = IMX_SC_RPC_VERSION;
    60		hdr->svc = IMX_SC_RPC_SVC_MISC;
    61		hdr->func = IMX_SC_MISC_FUNC_GET_TEMP;
    62		hdr->size = 2;
    63	
    64		ret = imx_scu_call_rpc(thermal_ipc_handle, &msg, true);
    65		if (ret) {
    66			pr_err("read temp sensor %d failed, ret %d\n",
    67				sensor->resource_id, ret);
    68			return ret;
    69		}
    70	
    71		resp = (struct imx_sc_msg_resp_misc_get_temp *)&msg;
    72		*temp = resp->celsius * 1000 + resp->tenths * 100;
    73	
    74		return 0;
    75	}
    76	
    77	static const struct thermal_zone_of_device_ops imx_sc_thermal_ops = {
    78		.get_temp = imx_sc_thermal_get_temp,
    79	};
    80	
    81	static int imx_sc_thermal_register_sensor(struct platform_device *pdev,
    82						  struct imx_sc_sensor *sensor)
    83	{
    84		struct thermal_zone_device *tzd;
    85	
    86		tzd = devm_thermal_zone_of_sensor_register(&pdev->dev,
    87							   sensor->resource_id,
    88							   sensor,
    89							   &imx_sc_thermal_ops);
    90		if (IS_ERR(tzd)) {
    91			dev_err(&pdev->dev, "failed to register sensor: %d\n",
    92				sensor->resource_id);
    93			return -EINVAL;
    94		}
    95	
    96		sensor->tzd = tzd;
    97	
    98		return 0;
    99	}
   100	
   101	static int imx_sc_thermal_get_sensor_id(struct device_node *sensor_np)
   102	{
   103		struct of_phandle_args sensor_specs;
   104		int ret;
   105	
   106		ret = of_parse_phandle_with_args(sensor_np, "thermal-sensors",
   107				"#thermal-sensor-cells",
   108				0, &sensor_specs);
   109		if (ret)
   110			return ret;
   111	
   112		if (sensor_specs.args_count >= 1) {
   113			ret = sensor_specs.args[0];
   114			WARN(sensor_specs.args_count > 1,
   115					"%pOFn: too many cells in sensor specifier %d\n",
   116					sensor_specs.np, sensor_specs.args_count);
   117		} else {
   118			ret = 0;
   119		}
   120	
   121		return ret;
   122	}
   123	
   124	static int imx_sc_thermal_probe(struct platform_device *pdev)
   125	{
   126		struct device_node *np = pdev->dev.of_node;
   127		struct device_node *sensor_np = NULL;
   128		struct imx_sc_thermal_data *data;
   129		struct imx_sc_sensor *sensors;
   130		u32 sensor_num;
   131		int ret, i;
   132	
   133		ret = imx_scu_get_handle(&thermal_ipc_handle);
   134		if (ret) {
   135			if (ret == -EPROBE_DEFER)
   136				return ret;
   137	
   138			dev_err(&pdev->dev, "failed to get ipc handle: %d!\n", ret);
   139			return ret;
   140		}
   141	
   142		data = devm_kzalloc(&pdev->dev, sizeof(*data), GFP_KERNEL);
   143		if (!data)
   144			return -ENOMEM;
   145	
   146		ret = of_property_read_u32(np, "tsens-num", &sensor_num);
   147		if (ret || !sensor_num) {
   148			dev_err(&pdev->dev, "failed to get valid temp sensor number!\n");
   149			ret = -EINVAL;
   150			goto free_data;
   151		}
   152	
   153		sensors = devm_kzalloc(&pdev->dev, sizeof(*data->sensor) * sensor_num,
   154				       GFP_KERNEL);
   155		if (!sensors) {
   156			ret = -ENOMEM;
   157			goto free_data;
   158		}
   159	
   160		data->sensor = sensors;
   161	
   162		np = of_find_node_by_name(NULL, "thermal-zones");
   163		if (!np) {
   164			ret = -ENODEV;
   165			goto free_sensors;
   166		}
   167	
   168		for (i = 0; i < sensor_num; i++) {
   169			struct imx_sc_sensor *sensor = &data->sensor[i];
   170	
   171			sensor_np = of_get_next_child(np, sensor_np);
   172			sensor->resource_id = imx_sc_thermal_get_sensor_id(sensor_np);
 > 173			if (sensor->resource_id < 0) {
   174				dev_err(&pdev->dev, "invalid sensor resource id: %d\n",
   175					sensor->resource_id);
   176				ret = sensor->resource_id;
   177				goto put_node;
   178			}
   179	
   180			ret = imx_sc_thermal_register_sensor(pdev, sensor);
   181			if (ret) {
   182				dev_err(&pdev->dev, "failed to register thermal sensor: %d\n",
   183					ret);
   184				goto put_node;
   185			}
   186		}
   187	
   188		of_node_put(sensor_np);
   189		of_node_put(np);
   190	
   191		return 0;
   192	
   193	put_node:
   194		of_node_put(np);
   195		of_node_put(sensor_np);
   196	free_sensors:
 > 197		kfree(sensors);
   198	free_data:
   199		kfree(data);
   200	
   201		return ret;
   202	}
   203	

---
0-DAY kernel test infrastructure                Open Source Technology Center
https://lists.01.org/pipermail/kbuild-all                   Intel Corporation

[-- Attachment #2: .config.gz --]
[-- Type: application/gzip, Size: 64928 bytes --]

^ permalink raw reply	[flat|nested] 3+ messages in thread

end of thread, other threads:[~2018-11-27 13:17 UTC | newest]

Thread overview: 3+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2018-11-27  6:01 [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support Anson Huang
2018-11-27  6:01 ` [PATCH 2/2] dt-bindings: thermal: add binding doc for i.MX system controller thermal driver Anson Huang
2018-11-27 13:16 ` [PATCH 1/2] thermal: imx_sc: add i.MX system controller thermal support kbuild test robot

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).