Skip to content

Commit a686024

Browse files
linchuyuangroeck
authored andcommitted
hwmon: (max6697) Allow max6581 to create tempX_offset
Add tempX_offset attribute support to max6697 driver. Temperature offsets are only supported for MAX6581. Testing: echo 16250 > temp2_offset cat temp2_offset 16250 echo 17500 > temp3_offset cat temp3_offset 17500 cat temp4_offset 0 cat temp2_offset 17500 echo 0 > temp2_offset cat temp2_offset 0 cat temp3_offset 17500 echo -0 > temp2_offset cat temp2_offset 0 echo -100000 > temp2_offset cat temp2_input 4875 echo 10000 > temp2_offset cat temp2_input 47125 echo -2000 > temp2_offset cat temp2_input 34875 echo -0 > temp2_offset cat temp2_input 37000 Signed-off-by: Chu Lin <[email protected]> Link: https://lore.kernel.org/r/[email protected] [groeck: Fixed checkpatch warning (unnecessary { })] Signed-off-by: Guenter Roeck <[email protected]>
1 parent e263f2d commit a686024

File tree

1 file changed

+94
-2
lines changed

1 file changed

+94
-2
lines changed

drivers/hwmon/max6697.c

Lines changed: 94 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,8 @@ static const u8 MAX6697_REG_CRIT[] = {
5757
#define MAX6581_REG_IDEALITY_SELECT 0x4c
5858
#define MAX6581_REG_OFFSET 0x4d
5959
#define MAX6581_REG_OFFSET_SELECT 0x4e
60+
#define MAX6581_OFFSET_MIN -31750
61+
#define MAX6581_OFFSET_MAX 31750
6062

6163
#define MAX6697_CONV_TIME 156 /* ms per channel, worst case */
6264

@@ -172,6 +174,11 @@ static const struct max6697_chip_data max6697_chip_data[] = {
172174
},
173175
};
174176

177+
static inline int max6581_offset_to_millic(int val)
178+
{
179+
return sign_extend32(val, 7) * 250;
180+
}
181+
175182
static struct max6697_data *max6697_update_device(struct device *dev)
176183
{
177184
struct max6697_data *data = dev_get_drvdata(dev);
@@ -317,6 +324,70 @@ static ssize_t temp_store(struct device *dev,
317324
return ret < 0 ? ret : count;
318325
}
319326

327+
static ssize_t offset_store(struct device *dev, struct device_attribute *devattr, const char *buf,
328+
size_t count)
329+
{
330+
int val, ret, index, select;
331+
struct max6697_data *data;
332+
bool channel_enabled;
333+
long temp;
334+
335+
index = to_sensor_dev_attr(devattr)->index;
336+
data = dev_get_drvdata(dev);
337+
ret = kstrtol(buf, 10, &temp);
338+
if (ret < 0)
339+
return ret;
340+
341+
mutex_lock(&data->update_lock);
342+
select = i2c_smbus_read_byte_data(data->client, MAX6581_REG_OFFSET_SELECT);
343+
if (select < 0) {
344+
ret = select;
345+
goto abort;
346+
}
347+
channel_enabled = (select & (1 << (index - 1)));
348+
temp = clamp_val(temp, MAX6581_OFFSET_MIN, MAX6581_OFFSET_MAX);
349+
val = DIV_ROUND_CLOSEST(temp, 250);
350+
/* disable the offset for channel if the new offset is 0 */
351+
if (val == 0) {
352+
if (channel_enabled)
353+
ret = i2c_smbus_write_byte_data(data->client, MAX6581_REG_OFFSET_SELECT,
354+
select & ~(1 << (index - 1)));
355+
ret = ret < 0 ? ret : count;
356+
goto abort;
357+
}
358+
if (!channel_enabled) {
359+
ret = i2c_smbus_write_byte_data(data->client, MAX6581_REG_OFFSET_SELECT,
360+
select | (1 << (index - 1)));
361+
if (ret < 0)
362+
goto abort;
363+
}
364+
ret = i2c_smbus_write_byte_data(data->client, MAX6581_REG_OFFSET, val);
365+
ret = ret < 0 ? ret : count;
366+
367+
abort:
368+
mutex_unlock(&data->update_lock);
369+
return ret;
370+
}
371+
372+
static ssize_t offset_show(struct device *dev, struct device_attribute *devattr, char *buf)
373+
{
374+
struct max6697_data *data;
375+
int select, ret, index;
376+
377+
index = to_sensor_dev_attr(devattr)->index;
378+
data = dev_get_drvdata(dev);
379+
mutex_lock(&data->update_lock);
380+
select = i2c_smbus_read_byte_data(data->client, MAX6581_REG_OFFSET_SELECT);
381+
if (select < 0)
382+
ret = select;
383+
else if (select & (1 << (index - 1)))
384+
ret = i2c_smbus_read_byte_data(data->client, MAX6581_REG_OFFSET);
385+
else
386+
ret = 0;
387+
mutex_unlock(&data->update_lock);
388+
return ret < 0 ? ret : sprintf(buf, "%d\n", max6581_offset_to_millic(ret));
389+
}
390+
320391
static SENSOR_DEVICE_ATTR_RO(temp1_input, temp_input, 0);
321392
static SENSOR_DEVICE_ATTR_2_RW(temp1_max, temp, 0, MAX6697_TEMP_MAX);
322393
static SENSOR_DEVICE_ATTR_2_RW(temp1_crit, temp, 0, MAX6697_TEMP_CRIT);
@@ -375,6 +446,15 @@ static SENSOR_DEVICE_ATTR_RO(temp6_fault, alarm, 5);
375446
static SENSOR_DEVICE_ATTR_RO(temp7_fault, alarm, 6);
376447
static SENSOR_DEVICE_ATTR_RO(temp8_fault, alarm, 7);
377448

449+
/* There is no offset for local temperature so starting from temp2 */
450+
static SENSOR_DEVICE_ATTR_RW(temp2_offset, offset, 1);
451+
static SENSOR_DEVICE_ATTR_RW(temp3_offset, offset, 2);
452+
static SENSOR_DEVICE_ATTR_RW(temp4_offset, offset, 3);
453+
static SENSOR_DEVICE_ATTR_RW(temp5_offset, offset, 4);
454+
static SENSOR_DEVICE_ATTR_RW(temp6_offset, offset, 5);
455+
static SENSOR_DEVICE_ATTR_RW(temp7_offset, offset, 6);
456+
static SENSOR_DEVICE_ATTR_RW(temp8_offset, offset, 7);
457+
378458
static DEVICE_ATTR(dummy, 0, NULL, NULL);
379459

380460
static umode_t max6697_is_visible(struct kobject *kobj, struct attribute *attr,
@@ -383,8 +463,8 @@ static umode_t max6697_is_visible(struct kobject *kobj, struct attribute *attr,
383463
struct device *dev = container_of(kobj, struct device, kobj);
384464
struct max6697_data *data = dev_get_drvdata(dev);
385465
const struct max6697_chip_data *chip = data->chip;
386-
int channel = index / 6; /* channel number */
387-
int nr = index % 6; /* attribute index within channel */
466+
int channel = index / 7; /* channel number */
467+
int nr = index % 7; /* attribute index within channel */
388468

389469
if (channel >= chip->channels)
390470
return 0;
@@ -393,6 +473,10 @@ static umode_t max6697_is_visible(struct kobject *kobj, struct attribute *attr,
393473
return 0;
394474
if (nr == 5 && !(chip->have_fault & (1 << channel)))
395475
return 0;
476+
/* offset reg is only supported on max6581 remote channels */
477+
if (nr == 6)
478+
if (data->type != max6581 || channel == 0)
479+
return 0;
396480

397481
return attr->mode;
398482
}
@@ -409,55 +493,63 @@ static struct attribute *max6697_attributes[] = {
409493
&sensor_dev_attr_temp1_crit.dev_attr.attr,
410494
&sensor_dev_attr_temp1_crit_alarm.dev_attr.attr,
411495
&dev_attr_dummy.attr,
496+
&dev_attr_dummy.attr,
412497

413498
&sensor_dev_attr_temp2_input.dev_attr.attr,
414499
&sensor_dev_attr_temp2_max.dev_attr.attr,
415500
&sensor_dev_attr_temp2_max_alarm.dev_attr.attr,
416501
&sensor_dev_attr_temp2_crit.dev_attr.attr,
417502
&sensor_dev_attr_temp2_crit_alarm.dev_attr.attr,
418503
&sensor_dev_attr_temp2_fault.dev_attr.attr,
504+
&sensor_dev_attr_temp2_offset.dev_attr.attr,
419505

420506
&sensor_dev_attr_temp3_input.dev_attr.attr,
421507
&sensor_dev_attr_temp3_max.dev_attr.attr,
422508
&sensor_dev_attr_temp3_max_alarm.dev_attr.attr,
423509
&sensor_dev_attr_temp3_crit.dev_attr.attr,
424510
&sensor_dev_attr_temp3_crit_alarm.dev_attr.attr,
425511
&sensor_dev_attr_temp3_fault.dev_attr.attr,
512+
&sensor_dev_attr_temp3_offset.dev_attr.attr,
426513

427514
&sensor_dev_attr_temp4_input.dev_attr.attr,
428515
&sensor_dev_attr_temp4_max.dev_attr.attr,
429516
&sensor_dev_attr_temp4_max_alarm.dev_attr.attr,
430517
&sensor_dev_attr_temp4_crit.dev_attr.attr,
431518
&sensor_dev_attr_temp4_crit_alarm.dev_attr.attr,
432519
&sensor_dev_attr_temp4_fault.dev_attr.attr,
520+
&sensor_dev_attr_temp4_offset.dev_attr.attr,
433521

434522
&sensor_dev_attr_temp5_input.dev_attr.attr,
435523
&sensor_dev_attr_temp5_max.dev_attr.attr,
436524
&sensor_dev_attr_temp5_max_alarm.dev_attr.attr,
437525
&sensor_dev_attr_temp5_crit.dev_attr.attr,
438526
&sensor_dev_attr_temp5_crit_alarm.dev_attr.attr,
439527
&sensor_dev_attr_temp5_fault.dev_attr.attr,
528+
&sensor_dev_attr_temp5_offset.dev_attr.attr,
440529

441530
&sensor_dev_attr_temp6_input.dev_attr.attr,
442531
&sensor_dev_attr_temp6_max.dev_attr.attr,
443532
&sensor_dev_attr_temp6_max_alarm.dev_attr.attr,
444533
&sensor_dev_attr_temp6_crit.dev_attr.attr,
445534
&sensor_dev_attr_temp6_crit_alarm.dev_attr.attr,
446535
&sensor_dev_attr_temp6_fault.dev_attr.attr,
536+
&sensor_dev_attr_temp6_offset.dev_attr.attr,
447537

448538
&sensor_dev_attr_temp7_input.dev_attr.attr,
449539
&sensor_dev_attr_temp7_max.dev_attr.attr,
450540
&sensor_dev_attr_temp7_max_alarm.dev_attr.attr,
451541
&sensor_dev_attr_temp7_crit.dev_attr.attr,
452542
&sensor_dev_attr_temp7_crit_alarm.dev_attr.attr,
453543
&sensor_dev_attr_temp7_fault.dev_attr.attr,
544+
&sensor_dev_attr_temp7_offset.dev_attr.attr,
454545

455546
&sensor_dev_attr_temp8_input.dev_attr.attr,
456547
&sensor_dev_attr_temp8_max.dev_attr.attr,
457548
&sensor_dev_attr_temp8_max_alarm.dev_attr.attr,
458549
&sensor_dev_attr_temp8_crit.dev_attr.attr,
459550
&sensor_dev_attr_temp8_crit_alarm.dev_attr.attr,
460551
&sensor_dev_attr_temp8_fault.dev_attr.attr,
552+
&sensor_dev_attr_temp8_offset.dev_attr.attr,
461553
NULL
462554
};
463555

0 commit comments

Comments
 (0)