Source
61
61
static const u8 THMC50_REG_TEMP_MIN[] = { 0x3A, 0x38, 0x2C };
62
62
static const u8 THMC50_REG_TEMP_MAX[] = { 0x39, 0x37, 0x2B };
63
63
static const u8 THMC50_REG_TEMP_CRITICAL[] = { 0x13, 0x14, 0x14 };
64
64
static const u8 THMC50_REG_TEMP_DEFAULT[] = { 0x17, 0x18, 0x18 };
65
65
66
66
#define THMC50_REG_CONF_nFANOFF 0x20
67
67
#define THMC50_REG_CONF_PROGRAMMED 0x08
68
68
69
69
/* Each client has this additional data */
70
70
struct thmc50_data {
71
-
struct device *hwmon_dev;
71
+
struct i2c_client *client;
72
+
const struct attribute_group *groups[3];
72
73
73
74
struct mutex update_lock;
74
75
enum chips type;
75
76
unsigned long last_updated; /* In jiffies */
76
77
char has_temp3; /* !=0 if it is ADM1022 in temp3 mode */
77
78
char valid; /* !=0 if following fields are valid */
78
79
79
80
/* Register values */
80
81
s8 temp_input[3];
81
82
s8 temp_max[3];
82
83
s8 temp_min[3];
83
84
s8 temp_critical[3];
84
85
u8 analog_out;
85
86
u8 alarms;
86
87
};
87
88
88
89
static struct thmc50_data *thmc50_update_device(struct device *dev)
89
90
{
90
-
struct i2c_client *client = to_i2c_client(dev);
91
-
struct thmc50_data *data = i2c_get_clientdata(client);
91
+
struct thmc50_data *data = dev_get_drvdata(dev);
92
+
struct i2c_client *client = data->client;
92
93
int timeout = HZ / 5 + (data->type == thmc50 ? HZ : 0);
93
94
94
95
mutex_lock(&data->update_lock);
95
96
96
97
if (time_after(jiffies, data->last_updated + timeout)
97
98
|| !data->valid) {
98
99
99
100
int temps = data->has_temp3 ? 3 : 2;
100
101
int i;
101
102
int prog = i2c_smbus_read_byte_data(client, THMC50_REG_CONF);
131
132
struct device_attribute *attr, char *buf)
132
133
{
133
134
struct thmc50_data *data = thmc50_update_device(dev);
134
135
return sprintf(buf, "%d\n", data->analog_out);
135
136
}
136
137
137
138
static ssize_t set_analog_out(struct device *dev,
138
139
struct device_attribute *attr,
139
140
const char *buf, size_t count)
140
141
{
141
-
struct i2c_client *client = to_i2c_client(dev);
142
-
struct thmc50_data *data = i2c_get_clientdata(client);
142
+
struct thmc50_data *data = dev_get_drvdata(dev);
143
+
struct i2c_client *client = data->client;
143
144
int config;
144
145
unsigned long tmp;
145
146
int err;
146
147
147
148
err = kstrtoul(buf, 10, &tmp);
148
149
if (err)
149
150
return err;
150
151
151
152
mutex_lock(&data->update_lock);
152
153
data->analog_out = clamp_val(tmp, 0, 255);
185
186
{
186
187
int nr = to_sensor_dev_attr(attr)->index;
187
188
struct thmc50_data *data = thmc50_update_device(dev);
188
189
return sprintf(buf, "%d\n", data->temp_min[nr] * 1000);
189
190
}
190
191
191
192
static ssize_t set_temp_min(struct device *dev, struct device_attribute *attr,
192
193
const char *buf, size_t count)
193
194
{
194
195
int nr = to_sensor_dev_attr(attr)->index;
195
-
struct i2c_client *client = to_i2c_client(dev);
196
-
struct thmc50_data *data = i2c_get_clientdata(client);
196
+
struct thmc50_data *data = dev_get_drvdata(dev);
197
+
struct i2c_client *client = data->client;
197
198
long val;
198
199
int err;
199
200
200
201
err = kstrtol(buf, 10, &val);
201
202
if (err)
202
203
return err;
203
204
204
205
mutex_lock(&data->update_lock);
205
206
data->temp_min[nr] = clamp_val(val / 1000, -128, 127);
206
207
i2c_smbus_write_byte_data(client, THMC50_REG_TEMP_MIN[nr],
214
215
{
215
216
int nr = to_sensor_dev_attr(attr)->index;
216
217
struct thmc50_data *data = thmc50_update_device(dev);
217
218
return sprintf(buf, "%d\n", data->temp_max[nr] * 1000);
218
219
}
219
220
220
221
static ssize_t set_temp_max(struct device *dev, struct device_attribute *attr,
221
222
const char *buf, size_t count)
222
223
{
223
224
int nr = to_sensor_dev_attr(attr)->index;
224
-
struct i2c_client *client = to_i2c_client(dev);
225
-
struct thmc50_data *data = i2c_get_clientdata(client);
225
+
struct thmc50_data *data = dev_get_drvdata(dev);
226
+
struct i2c_client *client = data->client;
226
227
long val;
227
228
int err;
228
229
229
230
err = kstrtol(buf, 10, &val);
230
231
if (err)
231
232
return err;
232
233
233
234
mutex_lock(&data->update_lock);
234
235
data->temp_max[nr] = clamp_val(val / 1000, -128, 127);
235
236
i2c_smbus_write_byte_data(client, THMC50_REG_TEMP_MAX[nr],
363
364
}
364
365
365
366
pr_debug("thmc50: Detected %s (version %x, revision %x)\n",
366
367
type_name, (revision >> 4) - 0xc, revision & 0xf);
367
368
368
369
strlcpy(info->type, type_name, I2C_NAME_SIZE);
369
370
370
371
return 0;
371
372
}
372
373
373
-
static void thmc50_init_client(struct i2c_client *client)
374
+
static void thmc50_init_client(struct thmc50_data *data)
374
375
{
375
-
struct thmc50_data *data = i2c_get_clientdata(client);
376
+
struct i2c_client *client = data->client;
376
377
int config;
377
378
378
379
data->analog_out = i2c_smbus_read_byte_data(client,
379
380
THMC50_REG_ANALOG_OUT);
380
381
/* set up to at least 1 */
381
382
if (data->analog_out == 0) {
382
383
data->analog_out = 1;
383
384
i2c_smbus_write_byte_data(client, THMC50_REG_ANALOG_OUT,
384
385
data->analog_out);
385
386
}
386
387
config = i2c_smbus_read_byte_data(client, THMC50_REG_CONF);
387
388
config |= 0x1; /* start the chip if it is in standby mode */
388
389
if (data->type == adm1022 && (config & (1 << 7)))
389
390
data->has_temp3 = 1;
390
391
i2c_smbus_write_byte_data(client, THMC50_REG_CONF, config);
391
392
}
392
393
393
394
static int thmc50_probe(struct i2c_client *client,
394
395
const struct i2c_device_id *id)
395
396
{
397
+
struct device *dev = &client->dev;
396
398
struct thmc50_data *data;
397
-
int err;
399
+
struct device *hwmon_dev;
400
+
int idx = 0;
398
401
399
-
data = devm_kzalloc(&client->dev, sizeof(struct thmc50_data),
400
-
GFP_KERNEL);
402
+
data = devm_kzalloc(dev, sizeof(struct thmc50_data), GFP_KERNEL);
401
403
if (!data)
402
404
return -ENOMEM;
403
405
404
-
i2c_set_clientdata(client, data);
406
+
data->client = client;
405
407
data->type = id->driver_data;
406
408
mutex_init(&data->update_lock);
407
409
408
-
thmc50_init_client(client);
410
+
thmc50_init_client(data);
409
411
410
-
/* Register sysfs hooks */
411
-
err = sysfs_create_group(&client->dev.kobj, &thmc50_group);
412
-
if (err)
413
-
return err;
414
-
415
-
/* Register ADM1022 sysfs hooks */
416
-
if (data->has_temp3) {
417
-
err = sysfs_create_group(&client->dev.kobj, &temp3_group);
418
-
if (err)
419
-
goto exit_remove_sysfs_thmc50;
420
-
}
421
-
422
-
/* Register a new directory entry with module sensors */
423
-
data->hwmon_dev = hwmon_device_register(&client->dev);
424
-
if (IS_ERR(data->hwmon_dev)) {
425
-
err = PTR_ERR(data->hwmon_dev);
426
-
goto exit_remove_sysfs;
427
-
}
412
+
/* sysfs hooks */
413
+
data->groups[idx++] = &thmc50_group;
428
414
429
-
return 0;
430
-
431
-
exit_remove_sysfs:
432
-
if (data->has_temp3)
433
-
sysfs_remove_group(&client->dev.kobj, &temp3_group);
434
-
exit_remove_sysfs_thmc50:
435
-
sysfs_remove_group(&client->dev.kobj, &thmc50_group);
436
-
return err;
437
-
}
438
-
439
-
static int thmc50_remove(struct i2c_client *client)
440
-
{
441
-
struct thmc50_data *data = i2c_get_clientdata(client);
442
-
443
-
hwmon_device_unregister(data->hwmon_dev);
444
-
sysfs_remove_group(&client->dev.kobj, &thmc50_group);
415
+
/* Register additional ADM1022 sysfs hooks */
445
416
if (data->has_temp3)
446
-
sysfs_remove_group(&client->dev.kobj, &temp3_group);
417
+
data->groups[idx++] = &temp3_group;
447
418
448
-
return 0;
419
+
hwmon_dev = devm_hwmon_device_register_with_groups(dev, client->name,
420
+
data, data->groups);
421
+
return PTR_ERR_OR_ZERO(hwmon_dev);
449
422
}
450
423
451
424
static const struct i2c_device_id thmc50_id[] = {
452
425
{ "adm1022", adm1022 },
453
426
{ "thmc50", thmc50 },
454
427
{ }
455
428
};
456
429
MODULE_DEVICE_TABLE(i2c, thmc50_id);
457
430
458
431
static struct i2c_driver thmc50_driver = {
459
432
.class = I2C_CLASS_HWMON,
460
433
.driver = {
461
434
.name = "thmc50",
462
435
},
463
436
.probe = thmc50_probe,
464
-
.remove = thmc50_remove,
465
437
.id_table = thmc50_id,
466
438
.detect = thmc50_detect,
467
439
.address_list = normal_i2c,
468
440
};
469
441
470
442
module_i2c_driver(thmc50_driver);
471
443
472
444
MODULE_AUTHOR("Krzysztof Helt <krzysztof.h1@wp.pl>");
473
445
MODULE_DESCRIPTION("THMC50 driver");