1 // SPDX-License-Identifier: GPL-2.0-only
3 * Windfarm PowerMac thermal control. SMU "satellite" controller sensors.
5 * Copyright (C) 2005 Paul Mackerras, IBM Corp. <paulus@samba.org>
8 #include <linux/types.h>
9 #include <linux/errno.h>
10 #include <linux/kernel.h>
11 #include <linux/slab.h>
12 #include <linux/init.h>
13 #include <linux/wait.h>
14 #include <linux/i2c.h>
15 #include <linux/mutex.h>
18 #include <asm/pmac_low_i2c.h>
24 /* If the cache is older than 800ms we'll refetch it */
25 #define MAX_AGE msecs_to_jiffies(800)
31 unsigned long last_read
; /* jiffies when cache last updated */
33 struct list_head sensors
;
34 struct i2c_client
*i2c
;
35 struct device_node
*node
;
38 static struct wf_sat
*sats
[2];
40 struct wf_sat_sensor
{
41 struct list_head link
;
43 int index2
; /* used for power sensors */
46 struct wf_sensor sens
;
49 #define wf_to_sat(c) container_of(c, struct wf_sat_sensor, sens)
51 struct smu_sdbp_header
*smu_sat_get_sdb_partition(unsigned int sat_id
, int id
,
60 /* TODO: Add the resulting partition to the device-tree */
62 if (sat_id
> 1 || (sat
= sats
[sat_id
]) == NULL
)
65 err
= i2c_smbus_write_word_data(sat
->i2c
, 8, id
<< 8);
67 printk(KERN_ERR
"smu_sat_get_sdb_part wr error %d\n", err
);
71 err
= i2c_smbus_read_word_data(sat
->i2c
, 9);
73 printk(KERN_ERR
"smu_sat_get_sdb_part rd len error\n");
78 printk(KERN_ERR
"smu_sat_get_sdb_part no partition %x\n", id
);
82 len
= le16_to_cpu(len
);
84 buf
= kmalloc(len
, GFP_KERNEL
);
88 for (i
= 0; i
< len
; i
+= 4) {
89 err
= i2c_smbus_read_i2c_block_data(sat
->i2c
, 0xa, 4, data
);
91 printk(KERN_ERR
"smu_sat_get_sdb_part rd err %d\n",
101 printk(KERN_DEBUG
"sat %d partition %x:", sat_id
, id
);
102 print_hex_dump(KERN_DEBUG
, " ", DUMP_PREFIX_OFFSET
,
103 16, 1, buf
, len
, false);
106 return (struct smu_sdbp_header
*) buf
;
112 EXPORT_SYMBOL_GPL(smu_sat_get_sdb_partition
);
114 /* refresh the cache */
115 static int wf_sat_read_cache(struct wf_sat
*sat
)
119 err
= i2c_smbus_read_i2c_block_data(sat
->i2c
, 0x3f, 16, sat
->cache
);
122 sat
->last_read
= jiffies
;
127 printk(KERN_DEBUG
"wf_sat_get: data is");
128 print_hex_dump(KERN_DEBUG
, " ", DUMP_PREFIX_OFFSET
,
129 16, 1, sat
->cache
, 16, false);
135 static int wf_sat_sensor_get(struct wf_sensor
*sr
, s32
*value
)
137 struct wf_sat_sensor
*sens
= wf_to_sat(sr
);
138 struct wf_sat
*sat
= sens
->sat
;
142 if (sat
->i2c
== NULL
)
145 mutex_lock(&sat
->mutex
);
146 if (time_after(jiffies
, (sat
->last_read
+ MAX_AGE
))) {
147 err
= wf_sat_read_cache(sat
);
153 val
= ((sat
->cache
[i
] << 8) + sat
->cache
[i
+1]) << sens
->shift
;
154 if (sens
->index2
>= 0) {
155 i
= sens
->index2
* 2;
156 /* 4.12 * 8.8 -> 12.20; shift right 4 to get 16.16 */
157 val
= (val
* ((sat
->cache
[i
] << 8) + sat
->cache
[i
+1])) >> 4;
164 mutex_unlock(&sat
->mutex
);
168 static void wf_sat_release(struct kref
*ref
)
170 struct wf_sat
*sat
= container_of(ref
, struct wf_sat
, ref
);
173 sats
[sat
->nr
] = NULL
;
177 static void wf_sat_sensor_release(struct wf_sensor
*sr
)
179 struct wf_sat_sensor
*sens
= wf_to_sat(sr
);
180 struct wf_sat
*sat
= sens
->sat
;
183 kref_put(&sat
->ref
, wf_sat_release
);
186 static const struct wf_sensor_ops wf_sat_ops
= {
187 .get_value
= wf_sat_sensor_get
,
188 .release
= wf_sat_sensor_release
,
189 .owner
= THIS_MODULE
,
192 static int wf_sat_probe(struct i2c_client
*client
,
193 const struct i2c_device_id
*id
)
195 struct device_node
*dev
= client
->dev
.of_node
;
197 struct wf_sat_sensor
*sens
;
201 struct device_node
*child
;
202 int shift
, cpu
, index
;
204 int vsens
[2], isens
[2];
206 sat
= kzalloc(sizeof(struct wf_sat
), GFP_KERNEL
);
210 sat
->node
= of_node_get(dev
);
211 kref_init(&sat
->ref
);
212 mutex_init(&sat
->mutex
);
214 INIT_LIST_HEAD(&sat
->sensors
);
215 i2c_set_clientdata(client
, sat
);
217 vsens
[0] = vsens
[1] = -1;
218 isens
[0] = isens
[1] = -1;
220 while ((child
= of_get_next_child(dev
, child
)) != NULL
) {
221 reg
= of_get_property(child
, "reg", NULL
);
222 loc
= of_get_property(child
, "location", NULL
);
223 if (reg
== NULL
|| loc
== NULL
)
226 /* the cooked sensors are between 0x30 and 0x37 */
227 if (*reg
< 0x30 || *reg
> 0x37)
231 /* expect location to be CPU [AB][01] ... */
232 if (strncmp(loc
, "CPU ", 4) != 0)
236 if (chip
> 1 || core
> 1) {
237 printk(KERN_ERR
"wf_sat_create: don't understand "
238 "location %s for %pOF\n", loc
, child
);
241 cpu
= 2 * chip
+ core
;
244 else if (sat
->nr
!= chip
) {
245 printk(KERN_ERR
"wf_sat_create: can't cope with "
246 "multiple CPU chips on one SAT (%s)\n", loc
);
250 if (of_node_is_type(child
, "voltage-sensor")) {
251 name
= "cpu-voltage";
254 } else if (of_node_is_type(child
, "current-sensor")) {
255 name
= "cpu-current";
258 } else if (of_node_is_type(child
, "temp-sensor")) {
262 continue; /* hmmm shouldn't happen */
264 /* the +16 is enough for "cpu-voltage-n" */
265 sens
= kzalloc(sizeof(struct wf_sat_sensor
) + 16, GFP_KERNEL
);
267 printk(KERN_ERR
"wf_sat_create: couldn't create "
268 "%s sensor %d (no memory)\n", name
, cpu
);
275 sens
->sens
.ops
= &wf_sat_ops
;
276 sens
->sens
.name
= (char *) (sens
+ 1);
277 snprintf((char *)sens
->sens
.name
, 16, "%s-%d", name
, cpu
);
279 if (wf_register_sensor(&sens
->sens
))
282 list_add(&sens
->link
, &sat
->sensors
);
287 /* make the power sensors */
288 for (core
= 0; core
< 2; ++core
) {
289 if (vsens
[core
] < 0 || isens
[core
] < 0)
291 cpu
= 2 * sat
->nr
+ core
;
292 sens
= kzalloc(sizeof(struct wf_sat_sensor
) + 16, GFP_KERNEL
);
294 printk(KERN_ERR
"wf_sat_create: couldn't create power "
295 "sensor %d (no memory)\n", cpu
);
298 sens
->index
= vsens
[core
];
299 sens
->index2
= isens
[core
];
302 sens
->sens
.ops
= &wf_sat_ops
;
303 sens
->sens
.name
= (char *) (sens
+ 1);
304 snprintf((char *)sens
->sens
.name
, 16, "cpu-power-%d", cpu
);
306 if (wf_register_sensor(&sens
->sens
))
309 list_add(&sens
->link
, &sat
->sensors
);
320 static int wf_sat_remove(struct i2c_client
*client
)
322 struct wf_sat
*sat
= i2c_get_clientdata(client
);
323 struct wf_sat_sensor
*sens
;
325 /* release sensors */
326 while(!list_empty(&sat
->sensors
)) {
327 sens
= list_first_entry(&sat
->sensors
,
328 struct wf_sat_sensor
, link
);
329 list_del(&sens
->link
);
330 wf_unregister_sensor(&sens
->sens
);
333 kref_put(&sat
->ref
, wf_sat_release
);
338 static const struct i2c_device_id wf_sat_id
[] = {
339 { "MAC,smu-sat", 0 },
342 MODULE_DEVICE_TABLE(i2c
, wf_sat_id
);
344 static struct i2c_driver wf_sat_driver
= {
346 .name
= "wf_smu_sat",
348 .probe
= wf_sat_probe
,
349 .remove
= wf_sat_remove
,
350 .id_table
= wf_sat_id
,
353 module_i2c_driver(wf_sat_driver
);
355 MODULE_AUTHOR("Paul Mackerras <paulus@samba.org>");
356 MODULE_DESCRIPTION("SMU satellite sensors for PowerMac thermal control");
357 MODULE_LICENSE("GPL");