1 // SPDX-License-Identifier: GPL-2.0
3 * System Control and Power Interface(SCPI) based hwmon sensor driver
5 * Copyright (C) 2015 ARM Ltd.
6 * Punit Agrawal <punit.agrawal@arm.com>
9 #include <linux/hwmon.h>
10 #include <linux/module.h>
12 #include <linux/platform_device.h>
13 #include <linux/scpi_protocol.h>
14 #include <linux/slab.h>
15 #include <linux/sysfs.h>
16 #include <linux/thermal.h>
20 struct scpi_sensor_info info
;
21 struct device_attribute dev_attr_input
;
22 struct device_attribute dev_attr_label
;
27 struct scpi_thermal_zone
{
29 struct scpi_sensors
*scpi_sensors
;
33 struct scpi_ops
*scpi_ops
;
34 struct sensor_data
*data
;
35 struct list_head thermal_zones
;
36 struct attribute
**attrs
;
37 struct attribute_group group
;
38 const struct attribute_group
*groups
[2];
41 static const u32 gxbb_scpi_scale
[] = {
42 [TEMPERATURE
] = 1, /* (celsius) */
43 [VOLTAGE
] = 1000, /* (millivolts) */
44 [CURRENT
] = 1000, /* (milliamperes) */
45 [POWER
] = 1000000, /* (microwatts) */
46 [ENERGY
] = 1000000, /* (microjoules) */
49 static const u32 scpi_scale
[] = {
50 [TEMPERATURE
] = 1000, /* (millicelsius) */
51 [VOLTAGE
] = 1000, /* (millivolts) */
52 [CURRENT
] = 1000, /* (milliamperes) */
53 [POWER
] = 1000000, /* (microwatts) */
54 [ENERGY
] = 1000000, /* (microjoules) */
57 static void scpi_scale_reading(u64
*value
, struct sensor_data
*sensor
)
59 if (scpi_scale
[sensor
->info
.class] != sensor
->scale
) {
60 *value
*= scpi_scale
[sensor
->info
.class];
61 do_div(*value
, sensor
->scale
);
65 static int scpi_read_temp(struct thermal_zone_device
*tz
, int *temp
)
67 struct scpi_thermal_zone
*zone
= thermal_zone_device_priv(tz
);
68 struct scpi_sensors
*scpi_sensors
= zone
->scpi_sensors
;
69 struct scpi_ops
*scpi_ops
= scpi_sensors
->scpi_ops
;
70 struct sensor_data
*sensor
= &scpi_sensors
->data
[zone
->sensor_id
];
74 ret
= scpi_ops
->sensor_get_value(sensor
->info
.sensor_id
, &value
);
78 scpi_scale_reading(&value
, sensor
);
84 /* hwmon callback functions */
86 scpi_show_sensor(struct device
*dev
, struct device_attribute
*attr
, char *buf
)
88 struct scpi_sensors
*scpi_sensors
= dev_get_drvdata(dev
);
89 struct scpi_ops
*scpi_ops
= scpi_sensors
->scpi_ops
;
90 struct sensor_data
*sensor
;
94 sensor
= container_of(attr
, struct sensor_data
, dev_attr_input
);
96 ret
= scpi_ops
->sensor_get_value(sensor
->info
.sensor_id
, &value
);
100 scpi_scale_reading(&value
, sensor
);
103 * Temperature sensor values are treated as signed values based on
104 * observation even though that is not explicitly specified, and
105 * because an unsigned u64 temperature does not really make practical
106 * sense especially when the temperature is below zero degrees Celsius.
108 if (sensor
->info
.class == TEMPERATURE
)
109 return sprintf(buf
, "%lld\n", (s64
)value
);
111 return sprintf(buf
, "%llu\n", value
);
115 scpi_show_label(struct device
*dev
, struct device_attribute
*attr
, char *buf
)
117 struct sensor_data
*sensor
;
119 sensor
= container_of(attr
, struct sensor_data
, dev_attr_label
);
121 return sprintf(buf
, "%s\n", sensor
->info
.name
);
124 static const struct thermal_zone_device_ops scpi_sensor_ops
= {
125 .get_temp
= scpi_read_temp
,
128 static const struct of_device_id scpi_of_match
[] = {
129 {.compatible
= "arm,scpi-sensors", .data
= &scpi_scale
},
130 {.compatible
= "amlogic,meson-gxbb-scpi-sensors", .data
= &gxbb_scpi_scale
},
133 MODULE_DEVICE_TABLE(of
, scpi_of_match
);
135 static int scpi_hwmon_probe(struct platform_device
*pdev
)
139 int num_temp
= 0, num_volt
= 0, num_current
= 0, num_power
= 0;
141 struct scpi_ops
*scpi_ops
;
142 struct device
*hwdev
, *dev
= &pdev
->dev
;
143 struct scpi_sensors
*scpi_sensors
;
146 scpi_ops
= get_scpi_ops();
148 return -EPROBE_DEFER
;
150 ret
= scpi_ops
->sensor_get_capability(&nr_sensors
);
157 scpi_sensors
= devm_kzalloc(dev
, sizeof(*scpi_sensors
), GFP_KERNEL
);
161 scpi_sensors
->data
= devm_kcalloc(dev
, nr_sensors
,
162 sizeof(*scpi_sensors
->data
), GFP_KERNEL
);
163 if (!scpi_sensors
->data
)
166 scpi_sensors
->attrs
= devm_kcalloc(dev
, (nr_sensors
* 2) + 1,
167 sizeof(*scpi_sensors
->attrs
), GFP_KERNEL
);
168 if (!scpi_sensors
->attrs
)
171 scpi_sensors
->scpi_ops
= scpi_ops
;
173 scale
= of_device_get_match_data(&pdev
->dev
);
175 dev_err(&pdev
->dev
, "Unable to initialize scpi-hwmon data\n");
179 for (i
= 0, idx
= 0; i
< nr_sensors
; i
++) {
180 struct sensor_data
*sensor
= &scpi_sensors
->data
[idx
];
182 ret
= scpi_ops
->sensor_get_info(i
, &sensor
->info
);
186 switch (sensor
->info
.class) {
188 snprintf(sensor
->input
, sizeof(sensor
->input
),
189 "temp%d_input", num_temp
+ 1);
190 snprintf(sensor
->label
, sizeof(sensor
->input
),
191 "temp%d_label", num_temp
+ 1);
195 snprintf(sensor
->input
, sizeof(sensor
->input
),
196 "in%d_input", num_volt
);
197 snprintf(sensor
->label
, sizeof(sensor
->input
),
198 "in%d_label", num_volt
);
202 snprintf(sensor
->input
, sizeof(sensor
->input
),
203 "curr%d_input", num_current
+ 1);
204 snprintf(sensor
->label
, sizeof(sensor
->input
),
205 "curr%d_label", num_current
+ 1);
209 snprintf(sensor
->input
, sizeof(sensor
->input
),
210 "power%d_input", num_power
+ 1);
211 snprintf(sensor
->label
, sizeof(sensor
->input
),
212 "power%d_label", num_power
+ 1);
216 snprintf(sensor
->input
, sizeof(sensor
->input
),
217 "energy%d_input", num_energy
+ 1);
218 snprintf(sensor
->label
, sizeof(sensor
->input
),
219 "energy%d_label", num_energy
+ 1);
226 sensor
->scale
= scale
[sensor
->info
.class];
228 sensor
->dev_attr_input
.attr
.mode
= 0444;
229 sensor
->dev_attr_input
.show
= scpi_show_sensor
;
230 sensor
->dev_attr_input
.attr
.name
= sensor
->input
;
232 sensor
->dev_attr_label
.attr
.mode
= 0444;
233 sensor
->dev_attr_label
.show
= scpi_show_label
;
234 sensor
->dev_attr_label
.attr
.name
= sensor
->label
;
236 scpi_sensors
->attrs
[idx
<< 1] = &sensor
->dev_attr_input
.attr
;
237 scpi_sensors
->attrs
[(idx
<< 1) + 1] = &sensor
->dev_attr_label
.attr
;
239 sysfs_attr_init(scpi_sensors
->attrs
[idx
<< 1]);
240 sysfs_attr_init(scpi_sensors
->attrs
[(idx
<< 1) + 1]);
244 scpi_sensors
->group
.attrs
= scpi_sensors
->attrs
;
245 scpi_sensors
->groups
[0] = &scpi_sensors
->group
;
247 platform_set_drvdata(pdev
, scpi_sensors
);
249 hwdev
= devm_hwmon_device_register_with_groups(dev
,
250 "scpi_sensors", scpi_sensors
, scpi_sensors
->groups
);
253 return PTR_ERR(hwdev
);
256 * Register the temperature sensors with the thermal framework
257 * to allow their usage in setting up the thermal zones from
260 * NOTE: Not all temperature sensors maybe used for thermal
263 INIT_LIST_HEAD(&scpi_sensors
->thermal_zones
);
264 for (i
= 0; i
< nr_sensors
; i
++) {
265 struct sensor_data
*sensor
= &scpi_sensors
->data
[i
];
266 struct thermal_zone_device
*z
;
267 struct scpi_thermal_zone
*zone
;
269 if (sensor
->info
.class != TEMPERATURE
)
272 zone
= devm_kzalloc(dev
, sizeof(*zone
), GFP_KERNEL
);
277 zone
->scpi_sensors
= scpi_sensors
;
278 z
= devm_thermal_of_zone_register(dev
,
279 sensor
->info
.sensor_id
,
283 * The call to thermal_zone_of_sensor_register returns
284 * an error for sensors that are not associated with
285 * any thermal zones or if the thermal subsystem is
289 devm_kfree(dev
, zone
);
295 static struct platform_driver scpi_hwmon_platdrv
= {
297 .name
= "scpi-hwmon",
298 .of_match_table
= scpi_of_match
,
300 .probe
= scpi_hwmon_probe
,
302 module_platform_driver(scpi_hwmon_platdrv
);
304 MODULE_AUTHOR("Punit Agrawal <punit.agrawal@arm.com>");
305 MODULE_DESCRIPTION("ARM SCPI HWMON interface driver");
306 MODULE_LICENSE("GPL v2");