18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * LM73 Sensor driver 48c2ecf20Sopenharmony_ci * Based on LM75 58c2ecf20Sopenharmony_ci * 68c2ecf20Sopenharmony_ci * Copyright (C) 2007, CenoSYS (www.cenosys.com). 78c2ecf20Sopenharmony_ci * Copyright (C) 2009, Bollore telecom (www.bolloretelecom.eu). 88c2ecf20Sopenharmony_ci * 98c2ecf20Sopenharmony_ci * Guillaume Ligneul <guillaume.ligneul@gmail.com> 108c2ecf20Sopenharmony_ci * Adrien Demarez <adrien.demarez@bolloretelecom.eu> 118c2ecf20Sopenharmony_ci * Jeremy Laine <jeremy.laine@bolloretelecom.eu> 128c2ecf20Sopenharmony_ci * Chris Verges <kg4ysn@gmail.com> 138c2ecf20Sopenharmony_ci */ 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_ci#include <linux/module.h> 168c2ecf20Sopenharmony_ci#include <linux/init.h> 178c2ecf20Sopenharmony_ci#include <linux/i2c.h> 188c2ecf20Sopenharmony_ci#include <linux/hwmon.h> 198c2ecf20Sopenharmony_ci#include <linux/hwmon-sysfs.h> 208c2ecf20Sopenharmony_ci#include <linux/err.h> 218c2ecf20Sopenharmony_ci 228c2ecf20Sopenharmony_ci 238c2ecf20Sopenharmony_ci/* Addresses scanned */ 248c2ecf20Sopenharmony_cistatic const unsigned short normal_i2c[] = { 0x48, 0x49, 0x4a, 0x4c, 258c2ecf20Sopenharmony_ci 0x4d, 0x4e, I2C_CLIENT_END }; 268c2ecf20Sopenharmony_ci 278c2ecf20Sopenharmony_ci/* LM73 registers */ 288c2ecf20Sopenharmony_ci#define LM73_REG_INPUT 0x00 298c2ecf20Sopenharmony_ci#define LM73_REG_CONF 0x01 308c2ecf20Sopenharmony_ci#define LM73_REG_MAX 0x02 318c2ecf20Sopenharmony_ci#define LM73_REG_MIN 0x03 328c2ecf20Sopenharmony_ci#define LM73_REG_CTRL 0x04 338c2ecf20Sopenharmony_ci#define LM73_REG_ID 0x07 348c2ecf20Sopenharmony_ci 358c2ecf20Sopenharmony_ci#define LM73_ID 0x9001 /* 0x0190, byte-swapped */ 368c2ecf20Sopenharmony_ci#define DRVNAME "lm73" 378c2ecf20Sopenharmony_ci#define LM73_TEMP_MIN (-256000 / 250) 388c2ecf20Sopenharmony_ci#define LM73_TEMP_MAX (255750 / 250) 398c2ecf20Sopenharmony_ci 408c2ecf20Sopenharmony_ci#define LM73_CTRL_RES_SHIFT 5 418c2ecf20Sopenharmony_ci#define LM73_CTRL_RES_MASK (BIT(5) | BIT(6)) 428c2ecf20Sopenharmony_ci#define LM73_CTRL_TO_MASK BIT(7) 438c2ecf20Sopenharmony_ci 448c2ecf20Sopenharmony_ci#define LM73_CTRL_HI_SHIFT 2 458c2ecf20Sopenharmony_ci#define LM73_CTRL_LO_SHIFT 1 468c2ecf20Sopenharmony_ci 478c2ecf20Sopenharmony_cistatic const unsigned short lm73_convrates[] = { 488c2ecf20Sopenharmony_ci 14, /* 11-bits (0.25000 C/LSB): RES1 Bit = 0, RES0 Bit = 0 */ 498c2ecf20Sopenharmony_ci 28, /* 12-bits (0.12500 C/LSB): RES1 Bit = 0, RES0 Bit = 1 */ 508c2ecf20Sopenharmony_ci 56, /* 13-bits (0.06250 C/LSB): RES1 Bit = 1, RES0 Bit = 0 */ 518c2ecf20Sopenharmony_ci 112, /* 14-bits (0.03125 C/LSB): RES1 Bit = 1, RES0 Bit = 1 */ 528c2ecf20Sopenharmony_ci}; 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_cistruct lm73_data { 558c2ecf20Sopenharmony_ci struct i2c_client *client; 568c2ecf20Sopenharmony_ci struct mutex lock; 578c2ecf20Sopenharmony_ci u8 ctrl; /* control register value */ 588c2ecf20Sopenharmony_ci}; 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_ci/*-----------------------------------------------------------------------*/ 618c2ecf20Sopenharmony_ci 628c2ecf20Sopenharmony_cistatic ssize_t temp_store(struct device *dev, struct device_attribute *da, 638c2ecf20Sopenharmony_ci const char *buf, size_t count) 648c2ecf20Sopenharmony_ci{ 658c2ecf20Sopenharmony_ci struct sensor_device_attribute *attr = to_sensor_dev_attr(da); 668c2ecf20Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 678c2ecf20Sopenharmony_ci long temp; 688c2ecf20Sopenharmony_ci short value; 698c2ecf20Sopenharmony_ci s32 err; 708c2ecf20Sopenharmony_ci 718c2ecf20Sopenharmony_ci int status = kstrtol(buf, 10, &temp); 728c2ecf20Sopenharmony_ci if (status < 0) 738c2ecf20Sopenharmony_ci return status; 748c2ecf20Sopenharmony_ci 758c2ecf20Sopenharmony_ci /* Write value */ 768c2ecf20Sopenharmony_ci value = clamp_val(temp / 250, LM73_TEMP_MIN, LM73_TEMP_MAX) << 5; 778c2ecf20Sopenharmony_ci err = i2c_smbus_write_word_swapped(data->client, attr->index, value); 788c2ecf20Sopenharmony_ci return (err < 0) ? err : count; 798c2ecf20Sopenharmony_ci} 808c2ecf20Sopenharmony_ci 818c2ecf20Sopenharmony_cistatic ssize_t temp_show(struct device *dev, struct device_attribute *da, 828c2ecf20Sopenharmony_ci char *buf) 838c2ecf20Sopenharmony_ci{ 848c2ecf20Sopenharmony_ci struct sensor_device_attribute *attr = to_sensor_dev_attr(da); 858c2ecf20Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 868c2ecf20Sopenharmony_ci int temp; 878c2ecf20Sopenharmony_ci 888c2ecf20Sopenharmony_ci s32 err = i2c_smbus_read_word_swapped(data->client, attr->index); 898c2ecf20Sopenharmony_ci if (err < 0) 908c2ecf20Sopenharmony_ci return err; 918c2ecf20Sopenharmony_ci 928c2ecf20Sopenharmony_ci /* use integer division instead of equivalent right shift to 938c2ecf20Sopenharmony_ci guarantee arithmetic shift and preserve the sign */ 948c2ecf20Sopenharmony_ci temp = (((s16) err) * 250) / 32; 958c2ecf20Sopenharmony_ci return scnprintf(buf, PAGE_SIZE, "%d\n", temp); 968c2ecf20Sopenharmony_ci} 978c2ecf20Sopenharmony_ci 988c2ecf20Sopenharmony_cistatic ssize_t convrate_store(struct device *dev, struct device_attribute *da, 998c2ecf20Sopenharmony_ci const char *buf, size_t count) 1008c2ecf20Sopenharmony_ci{ 1018c2ecf20Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 1028c2ecf20Sopenharmony_ci unsigned long convrate; 1038c2ecf20Sopenharmony_ci s32 err; 1048c2ecf20Sopenharmony_ci int res = 0; 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_ci err = kstrtoul(buf, 10, &convrate); 1078c2ecf20Sopenharmony_ci if (err < 0) 1088c2ecf20Sopenharmony_ci return err; 1098c2ecf20Sopenharmony_ci 1108c2ecf20Sopenharmony_ci /* 1118c2ecf20Sopenharmony_ci * Convert the desired conversion rate into register bits. 1128c2ecf20Sopenharmony_ci * res is already initialized, and everything past the second-to-last 1138c2ecf20Sopenharmony_ci * value in the array is treated as belonging to the last value 1148c2ecf20Sopenharmony_ci * in the array. 1158c2ecf20Sopenharmony_ci */ 1168c2ecf20Sopenharmony_ci while (res < (ARRAY_SIZE(lm73_convrates) - 1) && 1178c2ecf20Sopenharmony_ci convrate > lm73_convrates[res]) 1188c2ecf20Sopenharmony_ci res++; 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_ci mutex_lock(&data->lock); 1218c2ecf20Sopenharmony_ci data->ctrl &= LM73_CTRL_TO_MASK; 1228c2ecf20Sopenharmony_ci data->ctrl |= res << LM73_CTRL_RES_SHIFT; 1238c2ecf20Sopenharmony_ci err = i2c_smbus_write_byte_data(data->client, LM73_REG_CTRL, 1248c2ecf20Sopenharmony_ci data->ctrl); 1258c2ecf20Sopenharmony_ci mutex_unlock(&data->lock); 1268c2ecf20Sopenharmony_ci 1278c2ecf20Sopenharmony_ci if (err < 0) 1288c2ecf20Sopenharmony_ci return err; 1298c2ecf20Sopenharmony_ci 1308c2ecf20Sopenharmony_ci return count; 1318c2ecf20Sopenharmony_ci} 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_cistatic ssize_t convrate_show(struct device *dev, struct device_attribute *da, 1348c2ecf20Sopenharmony_ci char *buf) 1358c2ecf20Sopenharmony_ci{ 1368c2ecf20Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 1378c2ecf20Sopenharmony_ci int res; 1388c2ecf20Sopenharmony_ci 1398c2ecf20Sopenharmony_ci res = (data->ctrl & LM73_CTRL_RES_MASK) >> LM73_CTRL_RES_SHIFT; 1408c2ecf20Sopenharmony_ci return scnprintf(buf, PAGE_SIZE, "%hu\n", lm73_convrates[res]); 1418c2ecf20Sopenharmony_ci} 1428c2ecf20Sopenharmony_ci 1438c2ecf20Sopenharmony_cistatic ssize_t maxmin_alarm_show(struct device *dev, 1448c2ecf20Sopenharmony_ci struct device_attribute *da, char *buf) 1458c2ecf20Sopenharmony_ci{ 1468c2ecf20Sopenharmony_ci struct sensor_device_attribute *attr = to_sensor_dev_attr(da); 1478c2ecf20Sopenharmony_ci struct lm73_data *data = dev_get_drvdata(dev); 1488c2ecf20Sopenharmony_ci s32 ctrl; 1498c2ecf20Sopenharmony_ci 1508c2ecf20Sopenharmony_ci mutex_lock(&data->lock); 1518c2ecf20Sopenharmony_ci ctrl = i2c_smbus_read_byte_data(data->client, LM73_REG_CTRL); 1528c2ecf20Sopenharmony_ci if (ctrl < 0) 1538c2ecf20Sopenharmony_ci goto abort; 1548c2ecf20Sopenharmony_ci data->ctrl = ctrl; 1558c2ecf20Sopenharmony_ci mutex_unlock(&data->lock); 1568c2ecf20Sopenharmony_ci 1578c2ecf20Sopenharmony_ci return scnprintf(buf, PAGE_SIZE, "%d\n", (ctrl >> attr->index) & 1); 1588c2ecf20Sopenharmony_ci 1598c2ecf20Sopenharmony_ciabort: 1608c2ecf20Sopenharmony_ci mutex_unlock(&data->lock); 1618c2ecf20Sopenharmony_ci return ctrl; 1628c2ecf20Sopenharmony_ci} 1638c2ecf20Sopenharmony_ci 1648c2ecf20Sopenharmony_ci/*-----------------------------------------------------------------------*/ 1658c2ecf20Sopenharmony_ci 1668c2ecf20Sopenharmony_ci/* sysfs attributes for hwmon */ 1678c2ecf20Sopenharmony_ci 1688c2ecf20Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RW(temp1_max, temp, LM73_REG_MAX); 1698c2ecf20Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RW(temp1_min, temp, LM73_REG_MIN); 1708c2ecf20Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RO(temp1_input, temp, LM73_REG_INPUT); 1718c2ecf20Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RW(update_interval, convrate, 0); 1728c2ecf20Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RO(temp1_max_alarm, maxmin_alarm, 1738c2ecf20Sopenharmony_ci LM73_CTRL_HI_SHIFT); 1748c2ecf20Sopenharmony_cistatic SENSOR_DEVICE_ATTR_RO(temp1_min_alarm, maxmin_alarm, 1758c2ecf20Sopenharmony_ci LM73_CTRL_LO_SHIFT); 1768c2ecf20Sopenharmony_ci 1778c2ecf20Sopenharmony_cistatic struct attribute *lm73_attrs[] = { 1788c2ecf20Sopenharmony_ci &sensor_dev_attr_temp1_input.dev_attr.attr, 1798c2ecf20Sopenharmony_ci &sensor_dev_attr_temp1_max.dev_attr.attr, 1808c2ecf20Sopenharmony_ci &sensor_dev_attr_temp1_min.dev_attr.attr, 1818c2ecf20Sopenharmony_ci &sensor_dev_attr_update_interval.dev_attr.attr, 1828c2ecf20Sopenharmony_ci &sensor_dev_attr_temp1_max_alarm.dev_attr.attr, 1838c2ecf20Sopenharmony_ci &sensor_dev_attr_temp1_min_alarm.dev_attr.attr, 1848c2ecf20Sopenharmony_ci NULL 1858c2ecf20Sopenharmony_ci}; 1868c2ecf20Sopenharmony_ciATTRIBUTE_GROUPS(lm73); 1878c2ecf20Sopenharmony_ci 1888c2ecf20Sopenharmony_ci/*-----------------------------------------------------------------------*/ 1898c2ecf20Sopenharmony_ci 1908c2ecf20Sopenharmony_ci/* device probe and removal */ 1918c2ecf20Sopenharmony_ci 1928c2ecf20Sopenharmony_cistatic int 1938c2ecf20Sopenharmony_cilm73_probe(struct i2c_client *client) 1948c2ecf20Sopenharmony_ci{ 1958c2ecf20Sopenharmony_ci struct device *dev = &client->dev; 1968c2ecf20Sopenharmony_ci struct device *hwmon_dev; 1978c2ecf20Sopenharmony_ci struct lm73_data *data; 1988c2ecf20Sopenharmony_ci int ctrl; 1998c2ecf20Sopenharmony_ci 2008c2ecf20Sopenharmony_ci data = devm_kzalloc(dev, sizeof(struct lm73_data), GFP_KERNEL); 2018c2ecf20Sopenharmony_ci if (!data) 2028c2ecf20Sopenharmony_ci return -ENOMEM; 2038c2ecf20Sopenharmony_ci 2048c2ecf20Sopenharmony_ci data->client = client; 2058c2ecf20Sopenharmony_ci mutex_init(&data->lock); 2068c2ecf20Sopenharmony_ci 2078c2ecf20Sopenharmony_ci ctrl = i2c_smbus_read_byte_data(client, LM73_REG_CTRL); 2088c2ecf20Sopenharmony_ci if (ctrl < 0) 2098c2ecf20Sopenharmony_ci return ctrl; 2108c2ecf20Sopenharmony_ci data->ctrl = ctrl; 2118c2ecf20Sopenharmony_ci 2128c2ecf20Sopenharmony_ci hwmon_dev = devm_hwmon_device_register_with_groups(dev, client->name, 2138c2ecf20Sopenharmony_ci data, lm73_groups); 2148c2ecf20Sopenharmony_ci if (IS_ERR(hwmon_dev)) 2158c2ecf20Sopenharmony_ci return PTR_ERR(hwmon_dev); 2168c2ecf20Sopenharmony_ci 2178c2ecf20Sopenharmony_ci dev_info(dev, "sensor '%s'\n", client->name); 2188c2ecf20Sopenharmony_ci 2198c2ecf20Sopenharmony_ci return 0; 2208c2ecf20Sopenharmony_ci} 2218c2ecf20Sopenharmony_ci 2228c2ecf20Sopenharmony_cistatic const struct i2c_device_id lm73_ids[] = { 2238c2ecf20Sopenharmony_ci { "lm73", 0 }, 2248c2ecf20Sopenharmony_ci { /* LIST END */ } 2258c2ecf20Sopenharmony_ci}; 2268c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, lm73_ids); 2278c2ecf20Sopenharmony_ci 2288c2ecf20Sopenharmony_ci/* Return 0 if detection is successful, -ENODEV otherwise */ 2298c2ecf20Sopenharmony_cistatic int lm73_detect(struct i2c_client *new_client, 2308c2ecf20Sopenharmony_ci struct i2c_board_info *info) 2318c2ecf20Sopenharmony_ci{ 2328c2ecf20Sopenharmony_ci struct i2c_adapter *adapter = new_client->adapter; 2338c2ecf20Sopenharmony_ci int id, ctrl, conf; 2348c2ecf20Sopenharmony_ci 2358c2ecf20Sopenharmony_ci if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE_DATA | 2368c2ecf20Sopenharmony_ci I2C_FUNC_SMBUS_WORD_DATA)) 2378c2ecf20Sopenharmony_ci return -ENODEV; 2388c2ecf20Sopenharmony_ci 2398c2ecf20Sopenharmony_ci /* 2408c2ecf20Sopenharmony_ci * Do as much detection as possible with byte reads first, as word 2418c2ecf20Sopenharmony_ci * reads can confuse other devices. 2428c2ecf20Sopenharmony_ci */ 2438c2ecf20Sopenharmony_ci ctrl = i2c_smbus_read_byte_data(new_client, LM73_REG_CTRL); 2448c2ecf20Sopenharmony_ci if (ctrl < 0 || (ctrl & 0x10)) 2458c2ecf20Sopenharmony_ci return -ENODEV; 2468c2ecf20Sopenharmony_ci 2478c2ecf20Sopenharmony_ci conf = i2c_smbus_read_byte_data(new_client, LM73_REG_CONF); 2488c2ecf20Sopenharmony_ci if (conf < 0 || (conf & 0x0c)) 2498c2ecf20Sopenharmony_ci return -ENODEV; 2508c2ecf20Sopenharmony_ci 2518c2ecf20Sopenharmony_ci id = i2c_smbus_read_byte_data(new_client, LM73_REG_ID); 2528c2ecf20Sopenharmony_ci if (id < 0 || id != (LM73_ID & 0xff)) 2538c2ecf20Sopenharmony_ci return -ENODEV; 2548c2ecf20Sopenharmony_ci 2558c2ecf20Sopenharmony_ci /* Check device ID */ 2568c2ecf20Sopenharmony_ci id = i2c_smbus_read_word_data(new_client, LM73_REG_ID); 2578c2ecf20Sopenharmony_ci if (id < 0 || id != LM73_ID) 2588c2ecf20Sopenharmony_ci return -ENODEV; 2598c2ecf20Sopenharmony_ci 2608c2ecf20Sopenharmony_ci strlcpy(info->type, "lm73", I2C_NAME_SIZE); 2618c2ecf20Sopenharmony_ci 2628c2ecf20Sopenharmony_ci return 0; 2638c2ecf20Sopenharmony_ci} 2648c2ecf20Sopenharmony_ci 2658c2ecf20Sopenharmony_cistatic const struct of_device_id lm73_of_match[] = { 2668c2ecf20Sopenharmony_ci { 2678c2ecf20Sopenharmony_ci .compatible = "ti,lm73", 2688c2ecf20Sopenharmony_ci }, 2698c2ecf20Sopenharmony_ci { }, 2708c2ecf20Sopenharmony_ci}; 2718c2ecf20Sopenharmony_ci 2728c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(of, lm73_of_match); 2738c2ecf20Sopenharmony_ci 2748c2ecf20Sopenharmony_cistatic struct i2c_driver lm73_driver = { 2758c2ecf20Sopenharmony_ci .class = I2C_CLASS_HWMON, 2768c2ecf20Sopenharmony_ci .driver = { 2778c2ecf20Sopenharmony_ci .name = "lm73", 2788c2ecf20Sopenharmony_ci .of_match_table = lm73_of_match, 2798c2ecf20Sopenharmony_ci }, 2808c2ecf20Sopenharmony_ci .probe_new = lm73_probe, 2818c2ecf20Sopenharmony_ci .id_table = lm73_ids, 2828c2ecf20Sopenharmony_ci .detect = lm73_detect, 2838c2ecf20Sopenharmony_ci .address_list = normal_i2c, 2848c2ecf20Sopenharmony_ci}; 2858c2ecf20Sopenharmony_ci 2868c2ecf20Sopenharmony_cimodule_i2c_driver(lm73_driver); 2878c2ecf20Sopenharmony_ci 2888c2ecf20Sopenharmony_ciMODULE_AUTHOR("Guillaume Ligneul <guillaume.ligneul@gmail.com>"); 2898c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("LM73 driver"); 2908c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 291