162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * LM73 Sensor driver 462306a36Sopenharmony_ci * Based on LM75 562306a36Sopenharmony_ci * 662306a36Sopenharmony_ci * Copyright (C) 2007, CenoSYS (www.cenosys.com). 762306a36Sopenharmony_ci * Copyright (C) 2009, Bollore telecom (www.bolloretelecom.eu). 862306a36Sopenharmony_ci * 962306a36Sopenharmony_ci * Guillaume Ligneul <guillaume.ligneul@gmail.com> 1062306a36Sopenharmony_ci * Adrien Demarez <adrien.demarez@bolloretelecom.eu> 1162306a36Sopenharmony_ci * Jeremy Laine <jeremy.laine@bolloretelecom.eu> 1262306a36Sopenharmony_ci * Chris Verges <kg4ysn@gmail.com> 1362306a36Sopenharmony_ci */ 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci#include <linux/module.h> 1662306a36Sopenharmony_ci#include <linux/init.h> 1762306a36Sopenharmony_ci#include <linux/i2c.h> 1862306a36Sopenharmony_ci#include <linux/hwmon.h> 1962306a36Sopenharmony_ci#include <linux/hwmon-sysfs.h> 2062306a36Sopenharmony_ci#include <linux/err.h> 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_ci/* Addresses scanned */ 2462306a36Sopenharmony_cistatic const unsigned short normal_i2c[] = { 0x48, 0x49, 0x4a, 0x4c, 2562306a36Sopenharmony_ci 0x4d, 0x4e, I2C_CLIENT_END }; 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci/* LM73 registers */ 2862306a36Sopenharmony_ci#define LM73_REG_INPUT 0x00 2962306a36Sopenharmony_ci#define LM73_REG_CONF 0x01 3062306a36Sopenharmony_ci#define LM73_REG_MAX 0x02 3162306a36Sopenharmony_ci#define LM73_REG_MIN 0x03 3262306a36Sopenharmony_ci#define LM73_REG_CTRL 0x04 3362306a36Sopenharmony_ci#define LM73_REG_ID 0x07 3462306a36Sopenharmony_ci 3562306a36Sopenharmony_ci#define LM73_ID 0x9001 /* 0x0190, byte-swapped */ 3662306a36Sopenharmony_ci#define DRVNAME "lm73" 3762306a36Sopenharmony_ci#define LM73_TEMP_MIN (-256000 / 250) 3862306a36Sopenharmony_ci#define LM73_TEMP_MAX (255750 / 250) 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci#define LM73_CTRL_RES_SHIFT 5 4162306a36Sopenharmony_ci#define LM73_CTRL_RES_MASK (BIT(5) | BIT(6)) 4262306a36Sopenharmony_ci#define LM73_CTRL_TO_MASK BIT(7) 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci#define LM73_CTRL_HI_SHIFT 2 4562306a36Sopenharmony_ci#define LM73_CTRL_LO_SHIFT 1 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_cistatic const unsigned short lm73_convrates[] = { 4862306a36Sopenharmony_ci 14, /* 11-bits (0.25000 C/LSB): RES1 Bit = 0, RES0 Bit = 0 */ 4962306a36Sopenharmony_ci 28, /* 12-bits (0.12500 C/LSB): RES1 Bit = 0, RES0 Bit = 1 */ 5062306a36Sopenharmony_ci 56, /* 13-bits (0.06250 C/LSB): RES1 Bit = 1, RES0 Bit = 0 */ 5162306a36Sopenharmony_ci 112, /* 14-bits (0.03125 C/LSB): RES1 Bit = 1, RES0 Bit = 1 */ 5262306a36Sopenharmony_ci}; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_cistruct lm73_data { 5562306a36Sopenharmony_ci struct i2c_client *client; 5662306a36Sopenharmony_ci struct mutex lock; 5762306a36Sopenharmony_ci u8 ctrl; /* control register value */ 5862306a36Sopenharmony_ci}; 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci/*-----------------------------------------------------------------------*/ 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_cistatic ssize_t temp_store(struct device *dev, struct device_attribute *da, 6362306a36Sopenharmony_ci const char *buf, size_t count) 6462306a36Sopenharmony_ci{ 6562306a36Sopenharmony_ci struct sensor_device_attribute *attr = to_sensor_dev_attr(da); 6662306a36Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 6762306a36Sopenharmony_ci long temp; 6862306a36Sopenharmony_ci short value; 6962306a36Sopenharmony_ci s32 err; 7062306a36Sopenharmony_ci 7162306a36Sopenharmony_ci int status = kstrtol(buf, 10, &temp); 7262306a36Sopenharmony_ci if (status < 0) 7362306a36Sopenharmony_ci return status; 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci /* Write value */ 7662306a36Sopenharmony_ci value = clamp_val(temp / 250, LM73_TEMP_MIN, LM73_TEMP_MAX) << 5; 7762306a36Sopenharmony_ci err = i2c_smbus_write_word_swapped(data->client, attr->index, value); 7862306a36Sopenharmony_ci return (err < 0) ? err : count; 7962306a36Sopenharmony_ci} 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_cistatic ssize_t temp_show(struct device *dev, struct device_attribute *da, 8262306a36Sopenharmony_ci char *buf) 8362306a36Sopenharmony_ci{ 8462306a36Sopenharmony_ci struct sensor_device_attribute *attr = to_sensor_dev_attr(da); 8562306a36Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 8662306a36Sopenharmony_ci int temp; 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci s32 err = i2c_smbus_read_word_swapped(data->client, attr->index); 8962306a36Sopenharmony_ci if (err < 0) 9062306a36Sopenharmony_ci return err; 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci /* use integer division instead of equivalent right shift to 9362306a36Sopenharmony_ci guarantee arithmetic shift and preserve the sign */ 9462306a36Sopenharmony_ci temp = (((s16) err) * 250) / 32; 9562306a36Sopenharmony_ci return sysfs_emit(buf, "%d\n", temp); 9662306a36Sopenharmony_ci} 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_cistatic ssize_t convrate_store(struct device *dev, struct device_attribute *da, 9962306a36Sopenharmony_ci const char *buf, size_t count) 10062306a36Sopenharmony_ci{ 10162306a36Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 10262306a36Sopenharmony_ci unsigned long convrate; 10362306a36Sopenharmony_ci s32 err; 10462306a36Sopenharmony_ci int res = 0; 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ci err = kstrtoul(buf, 10, &convrate); 10762306a36Sopenharmony_ci if (err < 0) 10862306a36Sopenharmony_ci return err; 10962306a36Sopenharmony_ci 11062306a36Sopenharmony_ci /* 11162306a36Sopenharmony_ci * Convert the desired conversion rate into register bits. 11262306a36Sopenharmony_ci * res is already initialized, and everything past the second-to-last 11362306a36Sopenharmony_ci * value in the array is treated as belonging to the last value 11462306a36Sopenharmony_ci * in the array. 11562306a36Sopenharmony_ci */ 11662306a36Sopenharmony_ci while (res < (ARRAY_SIZE(lm73_convrates) - 1) && 11762306a36Sopenharmony_ci convrate > lm73_convrates[res]) 11862306a36Sopenharmony_ci res++; 11962306a36Sopenharmony_ci 12062306a36Sopenharmony_ci mutex_lock(&data->lock); 12162306a36Sopenharmony_ci data->ctrl &= LM73_CTRL_TO_MASK; 12262306a36Sopenharmony_ci data->ctrl |= res << LM73_CTRL_RES_SHIFT; 12362306a36Sopenharmony_ci err = i2c_smbus_write_byte_data(data->client, LM73_REG_CTRL, 12462306a36Sopenharmony_ci data->ctrl); 12562306a36Sopenharmony_ci mutex_unlock(&data->lock); 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_ci if (err < 0) 12862306a36Sopenharmony_ci return err; 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_ci return count; 13162306a36Sopenharmony_ci} 13262306a36Sopenharmony_ci 13362306a36Sopenharmony_cistatic ssize_t convrate_show(struct device *dev, struct device_attribute *da, 13462306a36Sopenharmony_ci char *buf) 13562306a36Sopenharmony_ci{ 13662306a36Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 13762306a36Sopenharmony_ci int res; 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_ci res = (data->ctrl & LM73_CTRL_RES_MASK) >> LM73_CTRL_RES_SHIFT; 14062306a36Sopenharmony_ci return sysfs_emit(buf, "%hu\n", lm73_convrates[res]); 14162306a36Sopenharmony_ci} 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_cistatic ssize_t maxmin_alarm_show(struct device *dev, 14462306a36Sopenharmony_ci struct device_attribute *da, char *buf) 14562306a36Sopenharmony_ci{ 14662306a36Sopenharmony_ci struct sensor_device_attribute *attr = to_sensor_dev_attr(da); 14762306a36Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 14862306a36Sopenharmony_ci s32 ctrl; 14962306a36Sopenharmony_ci 15062306a36Sopenharmony_ci mutex_lock(&data->lock); 15162306a36Sopenharmony_ci ctrl = i2c_smbus_read_byte_data(data->client, LM73_REG_CTRL); 15262306a36Sopenharmony_ci if (ctrl < 0) 15362306a36Sopenharmony_ci goto abort; 15462306a36Sopenharmony_ci data->ctrl = ctrl; 15562306a36Sopenharmony_ci mutex_unlock(&data->lock); 15662306a36Sopenharmony_ci 15762306a36Sopenharmony_ci return sysfs_emit(buf, "%d\n", (ctrl >> attr->index) & 1); 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ciabort: 16062306a36Sopenharmony_ci mutex_unlock(&data->lock); 16162306a36Sopenharmony_ci return ctrl; 16262306a36Sopenharmony_ci} 16362306a36Sopenharmony_ci 16462306a36Sopenharmony_ci/*-----------------------------------------------------------------------*/ 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci/* sysfs attributes for hwmon */ 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RW(temp1_max, temp, LM73_REG_MAX); 16962306a36Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RW(temp1_min, temp, LM73_REG_MIN); 17062306a36Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RO(temp1_input, temp, LM73_REG_INPUT); 17162306a36Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RW(update_interval, convrate, 0); 17262306a36Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RO(temp1_max_alarm, maxmin_alarm, 17362306a36Sopenharmony_ci LM73_CTRL_HI_SHIFT); 17462306a36Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RO(temp1_min_alarm, maxmin_alarm, 17562306a36Sopenharmony_ci LM73_CTRL_LO_SHIFT); 17662306a36Sopenharmony_ci 17762306a36Sopenharmony_cistatic struct attribute *lm73_attrs[] = { 17862306a36Sopenharmony_ci &sensor_dev_attr_temp1_input.dev_attr.attr, 17962306a36Sopenharmony_ci &sensor_dev_attr_temp1_max.dev_attr.attr, 18062306a36Sopenharmony_ci &sensor_dev_attr_temp1_min.dev_attr.attr, 18162306a36Sopenharmony_ci &sensor_dev_attr_update_interval.dev_attr.attr, 18262306a36Sopenharmony_ci &sensor_dev_attr_temp1_max_alarm.dev_attr.attr, 18362306a36Sopenharmony_ci &sensor_dev_attr_temp1_min_alarm.dev_attr.attr, 18462306a36Sopenharmony_ci NULL 18562306a36Sopenharmony_ci}; 18662306a36Sopenharmony_ciATTRIBUTE_GROUPS(lm73); 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_ci/*-----------------------------------------------------------------------*/ 18962306a36Sopenharmony_ci 19062306a36Sopenharmony_ci/* device probe and removal */ 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_cistatic int 19362306a36Sopenharmony_cilm73_probe(struct i2c_client *client) 19462306a36Sopenharmony_ci{ 19562306a36Sopenharmony_ci struct device *dev = &client->dev; 19662306a36Sopenharmony_ci struct device *hwmon_dev; 19762306a36Sopenharmony_ci struct lm73_data *data; 19862306a36Sopenharmony_ci int ctrl; 19962306a36Sopenharmony_ci 20062306a36Sopenharmony_ci data = devm_kzalloc(dev, sizeof(struct lm73_data), GFP_KERNEL); 20162306a36Sopenharmony_ci if (!data) 20262306a36Sopenharmony_ci return -ENOMEM; 20362306a36Sopenharmony_ci 20462306a36Sopenharmony_ci data->client = client; 20562306a36Sopenharmony_ci mutex_init(&data->lock); 20662306a36Sopenharmony_ci 20762306a36Sopenharmony_ci ctrl = i2c_smbus_read_byte_data(client, LM73_REG_CTRL); 20862306a36Sopenharmony_ci if (ctrl < 0) 20962306a36Sopenharmony_ci return ctrl; 21062306a36Sopenharmony_ci data->ctrl = ctrl; 21162306a36Sopenharmony_ci 21262306a36Sopenharmony_ci hwmon_dev = devm_hwmon_device_register_with_groups(dev, client->name, 21362306a36Sopenharmony_ci data, lm73_groups); 21462306a36Sopenharmony_ci if (IS_ERR(hwmon_dev)) 21562306a36Sopenharmony_ci return PTR_ERR(hwmon_dev); 21662306a36Sopenharmony_ci 21762306a36Sopenharmony_ci dev_info(dev, "sensor '%s'\n", client->name); 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_ci return 0; 22062306a36Sopenharmony_ci} 22162306a36Sopenharmony_ci 22262306a36Sopenharmony_cistatic const struct i2c_device_id lm73_ids[] = { 22362306a36Sopenharmony_ci { "lm73", 0 }, 22462306a36Sopenharmony_ci { /* LIST END */ } 22562306a36Sopenharmony_ci}; 22662306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, lm73_ids); 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_ci/* Return 0 if detection is successful, -ENODEV otherwise */ 22962306a36Sopenharmony_cistatic int lm73_detect(struct i2c_client *new_client, 23062306a36Sopenharmony_ci struct i2c_board_info *info) 23162306a36Sopenharmony_ci{ 23262306a36Sopenharmony_ci struct i2c_adapter *adapter = new_client->adapter; 23362306a36Sopenharmony_ci int id, ctrl, conf; 23462306a36Sopenharmony_ci 23562306a36Sopenharmony_ci if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE_DATA | 23662306a36Sopenharmony_ci I2C_FUNC_SMBUS_WORD_DATA)) 23762306a36Sopenharmony_ci return -ENODEV; 23862306a36Sopenharmony_ci 23962306a36Sopenharmony_ci /* 24062306a36Sopenharmony_ci * Do as much detection as possible with byte reads first, as word 24162306a36Sopenharmony_ci * reads can confuse other devices. 24262306a36Sopenharmony_ci */ 24362306a36Sopenharmony_ci ctrl = i2c_smbus_read_byte_data(new_client, LM73_REG_CTRL); 24462306a36Sopenharmony_ci if (ctrl < 0 || (ctrl & 0x10)) 24562306a36Sopenharmony_ci return -ENODEV; 24662306a36Sopenharmony_ci 24762306a36Sopenharmony_ci conf = i2c_smbus_read_byte_data(new_client, LM73_REG_CONF); 24862306a36Sopenharmony_ci if (conf < 0 || (conf & 0x0c)) 24962306a36Sopenharmony_ci return -ENODEV; 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci id = i2c_smbus_read_byte_data(new_client, LM73_REG_ID); 25262306a36Sopenharmony_ci if (id < 0 || id != (LM73_ID & 0xff)) 25362306a36Sopenharmony_ci return -ENODEV; 25462306a36Sopenharmony_ci 25562306a36Sopenharmony_ci /* Check device ID */ 25662306a36Sopenharmony_ci id = i2c_smbus_read_word_data(new_client, LM73_REG_ID); 25762306a36Sopenharmony_ci if (id < 0 || id != LM73_ID) 25862306a36Sopenharmony_ci return -ENODEV; 25962306a36Sopenharmony_ci 26062306a36Sopenharmony_ci strscpy(info->type, "lm73", I2C_NAME_SIZE); 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_ci return 0; 26362306a36Sopenharmony_ci} 26462306a36Sopenharmony_ci 26562306a36Sopenharmony_cistatic const struct of_device_id lm73_of_match[] = { 26662306a36Sopenharmony_ci { 26762306a36Sopenharmony_ci .compatible = "ti,lm73", 26862306a36Sopenharmony_ci }, 26962306a36Sopenharmony_ci { }, 27062306a36Sopenharmony_ci}; 27162306a36Sopenharmony_ci 27262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, lm73_of_match); 27362306a36Sopenharmony_ci 27462306a36Sopenharmony_cistatic struct i2c_driver lm73_driver = { 27562306a36Sopenharmony_ci .class = I2C_CLASS_HWMON, 27662306a36Sopenharmony_ci .driver = { 27762306a36Sopenharmony_ci .name = "lm73", 27862306a36Sopenharmony_ci .of_match_table = lm73_of_match, 27962306a36Sopenharmony_ci }, 28062306a36Sopenharmony_ci .probe = lm73_probe, 28162306a36Sopenharmony_ci .id_table = lm73_ids, 28262306a36Sopenharmony_ci .detect = lm73_detect, 28362306a36Sopenharmony_ci .address_list = normal_i2c, 28462306a36Sopenharmony_ci}; 28562306a36Sopenharmony_ci 28662306a36Sopenharmony_cimodule_i2c_driver(lm73_driver); 28762306a36Sopenharmony_ci 28862306a36Sopenharmony_ciMODULE_AUTHOR("Guillaume Ligneul <guillaume.ligneul@gmail.com>"); 28962306a36Sopenharmony_ciMODULE_DESCRIPTION("LM73 driver"); 29062306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 291