18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * nct7904.c - driver for Nuvoton NCT7904D. 48c2ecf20Sopenharmony_ci * 58c2ecf20Sopenharmony_ci * Copyright (c) 2015 Kontron 68c2ecf20Sopenharmony_ci * Author: Vadim V. Vlasov <vvlasov@dev.rtsoft.ru> 78c2ecf20Sopenharmony_ci * 88c2ecf20Sopenharmony_ci * Copyright (c) 2019 Advantech 98c2ecf20Sopenharmony_ci * Author: Amy.Shih <amy.shih@advantech.com.tw> 108c2ecf20Sopenharmony_ci * 118c2ecf20Sopenharmony_ci * Copyright (c) 2020 Advantech 128c2ecf20Sopenharmony_ci * Author: Yuechao Zhao <yuechao.zhao@advantech.com.cn> 138c2ecf20Sopenharmony_ci * 148c2ecf20Sopenharmony_ci * Supports the following chips: 158c2ecf20Sopenharmony_ci * 168c2ecf20Sopenharmony_ci * Chip #vin #fan #pwm #temp #dts chip ID 178c2ecf20Sopenharmony_ci * nct7904d 20 12 4 5 8 0xc5 188c2ecf20Sopenharmony_ci */ 198c2ecf20Sopenharmony_ci 208c2ecf20Sopenharmony_ci#include <linux/module.h> 218c2ecf20Sopenharmony_ci#include <linux/device.h> 228c2ecf20Sopenharmony_ci#include <linux/init.h> 238c2ecf20Sopenharmony_ci#include <linux/i2c.h> 248c2ecf20Sopenharmony_ci#include <linux/mutex.h> 258c2ecf20Sopenharmony_ci#include <linux/hwmon.h> 268c2ecf20Sopenharmony_ci#include <linux/watchdog.h> 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci#define VENDOR_ID_REG 0x7A /* Any bank */ 298c2ecf20Sopenharmony_ci#define NUVOTON_ID 0x50 308c2ecf20Sopenharmony_ci#define CHIP_ID_REG 0x7B /* Any bank */ 318c2ecf20Sopenharmony_ci#define NCT7904_ID 0xC5 328c2ecf20Sopenharmony_ci#define DEVICE_ID_REG 0x7C /* Any bank */ 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci#define BANK_SEL_REG 0xFF 358c2ecf20Sopenharmony_ci#define BANK_0 0x00 368c2ecf20Sopenharmony_ci#define BANK_1 0x01 378c2ecf20Sopenharmony_ci#define BANK_2 0x02 388c2ecf20Sopenharmony_ci#define BANK_3 0x03 398c2ecf20Sopenharmony_ci#define BANK_4 0x04 408c2ecf20Sopenharmony_ci#define BANK_MAX 0x04 418c2ecf20Sopenharmony_ci 428c2ecf20Sopenharmony_ci#define FANIN_MAX 12 /* Counted from 1 */ 438c2ecf20Sopenharmony_ci#define VSEN_MAX 21 /* VSEN1..14, 3VDD, VBAT, V3VSB, 448c2ecf20Sopenharmony_ci LTD (not a voltage), VSEN17..19 */ 458c2ecf20Sopenharmony_ci#define FANCTL_MAX 4 /* Counted from 1 */ 468c2ecf20Sopenharmony_ci#define TCPU_MAX 8 /* Counted from 1 */ 478c2ecf20Sopenharmony_ci#define TEMP_MAX 4 /* Counted from 1 */ 488c2ecf20Sopenharmony_ci#define SMI_STS_MAX 10 /* Counted from 1 */ 498c2ecf20Sopenharmony_ci 508c2ecf20Sopenharmony_ci#define VT_ADC_CTRL0_REG 0x20 /* Bank 0 */ 518c2ecf20Sopenharmony_ci#define VT_ADC_CTRL1_REG 0x21 /* Bank 0 */ 528c2ecf20Sopenharmony_ci#define VT_ADC_CTRL2_REG 0x22 /* Bank 0 */ 538c2ecf20Sopenharmony_ci#define FANIN_CTRL0_REG 0x24 548c2ecf20Sopenharmony_ci#define FANIN_CTRL1_REG 0x25 558c2ecf20Sopenharmony_ci#define DTS_T_CTRL0_REG 0x26 568c2ecf20Sopenharmony_ci#define DTS_T_CTRL1_REG 0x27 578c2ecf20Sopenharmony_ci#define VT_ADC_MD_REG 0x2E 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_ci#define VSEN1_HV_LL_REG 0x02 /* Bank 1; 2 regs (HV/LV) per sensor */ 608c2ecf20Sopenharmony_ci#define VSEN1_LV_LL_REG 0x03 /* Bank 1; 2 regs (HV/LV) per sensor */ 618c2ecf20Sopenharmony_ci#define VSEN1_HV_HL_REG 0x00 /* Bank 1; 2 regs (HV/LV) per sensor */ 628c2ecf20Sopenharmony_ci#define VSEN1_LV_HL_REG 0x01 /* Bank 1; 2 regs (HV/LV) per sensor */ 638c2ecf20Sopenharmony_ci#define SMI_STS1_REG 0xC1 /* Bank 0; SMI Status Register */ 648c2ecf20Sopenharmony_ci#define SMI_STS3_REG 0xC3 /* Bank 0; SMI Status Register */ 658c2ecf20Sopenharmony_ci#define SMI_STS5_REG 0xC5 /* Bank 0; SMI Status Register */ 668c2ecf20Sopenharmony_ci#define SMI_STS7_REG 0xC7 /* Bank 0; SMI Status Register */ 678c2ecf20Sopenharmony_ci#define SMI_STS8_REG 0xC8 /* Bank 0; SMI Status Register */ 688c2ecf20Sopenharmony_ci 698c2ecf20Sopenharmony_ci#define VSEN1_HV_REG 0x40 /* Bank 0; 2 regs (HV/LV) per sensor */ 708c2ecf20Sopenharmony_ci#define TEMP_CH1_HV_REG 0x42 /* Bank 0; same as VSEN2_HV */ 718c2ecf20Sopenharmony_ci#define LTD_HV_REG 0x62 /* Bank 0; 2 regs in VSEN range */ 728c2ecf20Sopenharmony_ci#define LTD_HV_HL_REG 0x44 /* Bank 1; 1 reg for LTD */ 738c2ecf20Sopenharmony_ci#define LTD_LV_HL_REG 0x45 /* Bank 1; 1 reg for LTD */ 748c2ecf20Sopenharmony_ci#define LTD_HV_LL_REG 0x46 /* Bank 1; 1 reg for LTD */ 758c2ecf20Sopenharmony_ci#define LTD_LV_LL_REG 0x47 /* Bank 1; 1 reg for LTD */ 768c2ecf20Sopenharmony_ci#define TEMP_CH1_CH_REG 0x05 /* Bank 1; 1 reg for LTD */ 778c2ecf20Sopenharmony_ci#define TEMP_CH1_W_REG 0x06 /* Bank 1; 1 reg for LTD */ 788c2ecf20Sopenharmony_ci#define TEMP_CH1_WH_REG 0x07 /* Bank 1; 1 reg for LTD */ 798c2ecf20Sopenharmony_ci#define TEMP_CH1_C_REG 0x04 /* Bank 1; 1 reg per sensor */ 808c2ecf20Sopenharmony_ci#define DTS_T_CPU1_C_REG 0x90 /* Bank 1; 1 reg per sensor */ 818c2ecf20Sopenharmony_ci#define DTS_T_CPU1_CH_REG 0x91 /* Bank 1; 1 reg per sensor */ 828c2ecf20Sopenharmony_ci#define DTS_T_CPU1_W_REG 0x92 /* Bank 1; 1 reg per sensor */ 838c2ecf20Sopenharmony_ci#define DTS_T_CPU1_WH_REG 0x93 /* Bank 1; 1 reg per sensor */ 848c2ecf20Sopenharmony_ci#define FANIN1_HV_REG 0x80 /* Bank 0; 2 regs (HV/LV) per sensor */ 858c2ecf20Sopenharmony_ci#define FANIN1_HV_HL_REG 0x60 /* Bank 1; 2 regs (HV/LV) per sensor */ 868c2ecf20Sopenharmony_ci#define FANIN1_LV_HL_REG 0x61 /* Bank 1; 2 regs (HV/LV) per sensor */ 878c2ecf20Sopenharmony_ci#define T_CPU1_HV_REG 0xA0 /* Bank 0; 2 regs (HV/LV) per sensor */ 888c2ecf20Sopenharmony_ci 898c2ecf20Sopenharmony_ci#define PRTS_REG 0x03 /* Bank 2 */ 908c2ecf20Sopenharmony_ci#define PFE_REG 0x00 /* Bank 2; PECI Function Enable */ 918c2ecf20Sopenharmony_ci#define TSI_CTRL_REG 0x50 /* Bank 2; TSI Control Register */ 928c2ecf20Sopenharmony_ci#define FANCTL1_FMR_REG 0x00 /* Bank 3; 1 reg per channel */ 938c2ecf20Sopenharmony_ci#define FANCTL1_OUT_REG 0x10 /* Bank 3; 1 reg per channel */ 948c2ecf20Sopenharmony_ci 958c2ecf20Sopenharmony_ci#define WDT_LOCK_REG 0xE0 /* W/O Lock Watchdog Register */ 968c2ecf20Sopenharmony_ci#define WDT_EN_REG 0xE1 /* R/O Watchdog Enable Register */ 978c2ecf20Sopenharmony_ci#define WDT_STS_REG 0xE2 /* R/O Watchdog Status Register */ 988c2ecf20Sopenharmony_ci#define WDT_TIMER_REG 0xE3 /* R/W Watchdog Timer Register */ 998c2ecf20Sopenharmony_ci#define WDT_SOFT_EN 0x55 /* Enable soft watchdog timer */ 1008c2ecf20Sopenharmony_ci#define WDT_SOFT_DIS 0xAA /* Disable soft watchdog timer */ 1018c2ecf20Sopenharmony_ci 1028c2ecf20Sopenharmony_ci#define VOLT_MONITOR_MODE 0x0 1038c2ecf20Sopenharmony_ci#define THERMAL_DIODE_MODE 0x1 1048c2ecf20Sopenharmony_ci#define THERMISTOR_MODE 0x3 1058c2ecf20Sopenharmony_ci 1068c2ecf20Sopenharmony_ci#define ENABLE_TSI BIT(1) 1078c2ecf20Sopenharmony_ci 1088c2ecf20Sopenharmony_ci#define WATCHDOG_TIMEOUT 1 /* 1 minute default timeout */ 1098c2ecf20Sopenharmony_ci 1108c2ecf20Sopenharmony_ci/*The timeout range is 1-255 minutes*/ 1118c2ecf20Sopenharmony_ci#define MIN_TIMEOUT (1 * 60) 1128c2ecf20Sopenharmony_ci#define MAX_TIMEOUT (255 * 60) 1138c2ecf20Sopenharmony_ci 1148c2ecf20Sopenharmony_cistatic int timeout; 1158c2ecf20Sopenharmony_cimodule_param(timeout, int, 0); 1168c2ecf20Sopenharmony_ciMODULE_PARM_DESC(timeout, "Watchdog timeout in minutes. 1 <= timeout <= 255, default=" 1178c2ecf20Sopenharmony_ci __MODULE_STRING(WATCHDOG_TIMEOUT) "."); 1188c2ecf20Sopenharmony_ci 1198c2ecf20Sopenharmony_cistatic bool nowayout = WATCHDOG_NOWAYOUT; 1208c2ecf20Sopenharmony_cimodule_param(nowayout, bool, 0); 1218c2ecf20Sopenharmony_ciMODULE_PARM_DESC(nowayout, "Watchdog cannot be stopped once started (default=" 1228c2ecf20Sopenharmony_ci __MODULE_STRING(WATCHDOG_NOWAYOUT) ")"); 1238c2ecf20Sopenharmony_ci 1248c2ecf20Sopenharmony_cistatic const unsigned short normal_i2c[] = { 1258c2ecf20Sopenharmony_ci 0x2d, 0x2e, I2C_CLIENT_END 1268c2ecf20Sopenharmony_ci}; 1278c2ecf20Sopenharmony_ci 1288c2ecf20Sopenharmony_cistruct nct7904_data { 1298c2ecf20Sopenharmony_ci struct i2c_client *client; 1308c2ecf20Sopenharmony_ci struct watchdog_device wdt; 1318c2ecf20Sopenharmony_ci struct mutex bank_lock; 1328c2ecf20Sopenharmony_ci int bank_sel; 1338c2ecf20Sopenharmony_ci u32 fanin_mask; 1348c2ecf20Sopenharmony_ci u32 vsen_mask; 1358c2ecf20Sopenharmony_ci u32 tcpu_mask; 1368c2ecf20Sopenharmony_ci u8 fan_mode[FANCTL_MAX]; 1378c2ecf20Sopenharmony_ci u8 enable_dts; 1388c2ecf20Sopenharmony_ci u8 has_dts; 1398c2ecf20Sopenharmony_ci u8 temp_mode; /* 0: TR mode, 1: TD mode */ 1408c2ecf20Sopenharmony_ci u8 fan_alarm[2]; 1418c2ecf20Sopenharmony_ci u8 vsen_alarm[3]; 1428c2ecf20Sopenharmony_ci}; 1438c2ecf20Sopenharmony_ci 1448c2ecf20Sopenharmony_ci/* Access functions */ 1458c2ecf20Sopenharmony_cistatic int nct7904_bank_lock(struct nct7904_data *data, unsigned int bank) 1468c2ecf20Sopenharmony_ci{ 1478c2ecf20Sopenharmony_ci int ret; 1488c2ecf20Sopenharmony_ci 1498c2ecf20Sopenharmony_ci mutex_lock(&data->bank_lock); 1508c2ecf20Sopenharmony_ci if (data->bank_sel == bank) 1518c2ecf20Sopenharmony_ci return 0; 1528c2ecf20Sopenharmony_ci ret = i2c_smbus_write_byte_data(data->client, BANK_SEL_REG, bank); 1538c2ecf20Sopenharmony_ci if (ret == 0) 1548c2ecf20Sopenharmony_ci data->bank_sel = bank; 1558c2ecf20Sopenharmony_ci else 1568c2ecf20Sopenharmony_ci data->bank_sel = -1; 1578c2ecf20Sopenharmony_ci return ret; 1588c2ecf20Sopenharmony_ci} 1598c2ecf20Sopenharmony_ci 1608c2ecf20Sopenharmony_cistatic inline void nct7904_bank_release(struct nct7904_data *data) 1618c2ecf20Sopenharmony_ci{ 1628c2ecf20Sopenharmony_ci mutex_unlock(&data->bank_lock); 1638c2ecf20Sopenharmony_ci} 1648c2ecf20Sopenharmony_ci 1658c2ecf20Sopenharmony_ci/* Read 1-byte register. Returns unsigned reg or -ERRNO on error. */ 1668c2ecf20Sopenharmony_cistatic int nct7904_read_reg(struct nct7904_data *data, 1678c2ecf20Sopenharmony_ci unsigned int bank, unsigned int reg) 1688c2ecf20Sopenharmony_ci{ 1698c2ecf20Sopenharmony_ci struct i2c_client *client = data->client; 1708c2ecf20Sopenharmony_ci int ret; 1718c2ecf20Sopenharmony_ci 1728c2ecf20Sopenharmony_ci ret = nct7904_bank_lock(data, bank); 1738c2ecf20Sopenharmony_ci if (ret == 0) 1748c2ecf20Sopenharmony_ci ret = i2c_smbus_read_byte_data(client, reg); 1758c2ecf20Sopenharmony_ci 1768c2ecf20Sopenharmony_ci nct7904_bank_release(data); 1778c2ecf20Sopenharmony_ci return ret; 1788c2ecf20Sopenharmony_ci} 1798c2ecf20Sopenharmony_ci 1808c2ecf20Sopenharmony_ci/* 1818c2ecf20Sopenharmony_ci * Read 2-byte register. Returns register in big-endian format or 1828c2ecf20Sopenharmony_ci * -ERRNO on error. 1838c2ecf20Sopenharmony_ci */ 1848c2ecf20Sopenharmony_cistatic int nct7904_read_reg16(struct nct7904_data *data, 1858c2ecf20Sopenharmony_ci unsigned int bank, unsigned int reg) 1868c2ecf20Sopenharmony_ci{ 1878c2ecf20Sopenharmony_ci struct i2c_client *client = data->client; 1888c2ecf20Sopenharmony_ci int ret, hi; 1898c2ecf20Sopenharmony_ci 1908c2ecf20Sopenharmony_ci ret = nct7904_bank_lock(data, bank); 1918c2ecf20Sopenharmony_ci if (ret == 0) { 1928c2ecf20Sopenharmony_ci ret = i2c_smbus_read_byte_data(client, reg); 1938c2ecf20Sopenharmony_ci if (ret >= 0) { 1948c2ecf20Sopenharmony_ci hi = ret; 1958c2ecf20Sopenharmony_ci ret = i2c_smbus_read_byte_data(client, reg + 1); 1968c2ecf20Sopenharmony_ci if (ret >= 0) 1978c2ecf20Sopenharmony_ci ret |= hi << 8; 1988c2ecf20Sopenharmony_ci } 1998c2ecf20Sopenharmony_ci } 2008c2ecf20Sopenharmony_ci 2018c2ecf20Sopenharmony_ci nct7904_bank_release(data); 2028c2ecf20Sopenharmony_ci return ret; 2038c2ecf20Sopenharmony_ci} 2048c2ecf20Sopenharmony_ci 2058c2ecf20Sopenharmony_ci/* Write 1-byte register. Returns 0 or -ERRNO on error. */ 2068c2ecf20Sopenharmony_cistatic int nct7904_write_reg(struct nct7904_data *data, 2078c2ecf20Sopenharmony_ci unsigned int bank, unsigned int reg, u8 val) 2088c2ecf20Sopenharmony_ci{ 2098c2ecf20Sopenharmony_ci struct i2c_client *client = data->client; 2108c2ecf20Sopenharmony_ci int ret; 2118c2ecf20Sopenharmony_ci 2128c2ecf20Sopenharmony_ci ret = nct7904_bank_lock(data, bank); 2138c2ecf20Sopenharmony_ci if (ret == 0) 2148c2ecf20Sopenharmony_ci ret = i2c_smbus_write_byte_data(client, reg, val); 2158c2ecf20Sopenharmony_ci 2168c2ecf20Sopenharmony_ci nct7904_bank_release(data); 2178c2ecf20Sopenharmony_ci return ret; 2188c2ecf20Sopenharmony_ci} 2198c2ecf20Sopenharmony_ci 2208c2ecf20Sopenharmony_cistatic int nct7904_read_fan(struct device *dev, u32 attr, int channel, 2218c2ecf20Sopenharmony_ci long *val) 2228c2ecf20Sopenharmony_ci{ 2238c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 2248c2ecf20Sopenharmony_ci unsigned int cnt, rpm; 2258c2ecf20Sopenharmony_ci int ret; 2268c2ecf20Sopenharmony_ci 2278c2ecf20Sopenharmony_ci switch (attr) { 2288c2ecf20Sopenharmony_ci case hwmon_fan_input: 2298c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_0, 2308c2ecf20Sopenharmony_ci FANIN1_HV_REG + channel * 2); 2318c2ecf20Sopenharmony_ci if (ret < 0) 2328c2ecf20Sopenharmony_ci return ret; 2338c2ecf20Sopenharmony_ci cnt = ((ret & 0xff00) >> 3) | (ret & 0x1f); 2348c2ecf20Sopenharmony_ci if (cnt == 0 || cnt == 0x1fff) 2358c2ecf20Sopenharmony_ci rpm = 0; 2368c2ecf20Sopenharmony_ci else 2378c2ecf20Sopenharmony_ci rpm = 1350000 / cnt; 2388c2ecf20Sopenharmony_ci *val = rpm; 2398c2ecf20Sopenharmony_ci return 0; 2408c2ecf20Sopenharmony_ci case hwmon_fan_min: 2418c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_1, 2428c2ecf20Sopenharmony_ci FANIN1_HV_HL_REG + channel * 2); 2438c2ecf20Sopenharmony_ci if (ret < 0) 2448c2ecf20Sopenharmony_ci return ret; 2458c2ecf20Sopenharmony_ci cnt = ((ret & 0xff00) >> 3) | (ret & 0x1f); 2468c2ecf20Sopenharmony_ci if (cnt == 0 || cnt == 0x1fff) 2478c2ecf20Sopenharmony_ci rpm = 0; 2488c2ecf20Sopenharmony_ci else 2498c2ecf20Sopenharmony_ci rpm = 1350000 / cnt; 2508c2ecf20Sopenharmony_ci *val = rpm; 2518c2ecf20Sopenharmony_ci return 0; 2528c2ecf20Sopenharmony_ci case hwmon_fan_alarm: 2538c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, 2548c2ecf20Sopenharmony_ci SMI_STS5_REG + (channel >> 3)); 2558c2ecf20Sopenharmony_ci if (ret < 0) 2568c2ecf20Sopenharmony_ci return ret; 2578c2ecf20Sopenharmony_ci if (!data->fan_alarm[channel >> 3]) 2588c2ecf20Sopenharmony_ci data->fan_alarm[channel >> 3] = ret & 0xff; 2598c2ecf20Sopenharmony_ci else 2608c2ecf20Sopenharmony_ci /* If there is new alarm showing up */ 2618c2ecf20Sopenharmony_ci data->fan_alarm[channel >> 3] |= (ret & 0xff); 2628c2ecf20Sopenharmony_ci *val = (data->fan_alarm[channel >> 3] >> (channel & 0x07)) & 1; 2638c2ecf20Sopenharmony_ci /* Needs to clean the alarm if alarm existing */ 2648c2ecf20Sopenharmony_ci if (*val) 2658c2ecf20Sopenharmony_ci data->fan_alarm[channel >> 3] ^= 1 << (channel & 0x07); 2668c2ecf20Sopenharmony_ci return 0; 2678c2ecf20Sopenharmony_ci default: 2688c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 2698c2ecf20Sopenharmony_ci } 2708c2ecf20Sopenharmony_ci} 2718c2ecf20Sopenharmony_ci 2728c2ecf20Sopenharmony_cistatic umode_t nct7904_fan_is_visible(const void *_data, u32 attr, int channel) 2738c2ecf20Sopenharmony_ci{ 2748c2ecf20Sopenharmony_ci const struct nct7904_data *data = _data; 2758c2ecf20Sopenharmony_ci 2768c2ecf20Sopenharmony_ci switch (attr) { 2778c2ecf20Sopenharmony_ci case hwmon_fan_input: 2788c2ecf20Sopenharmony_ci case hwmon_fan_alarm: 2798c2ecf20Sopenharmony_ci if (data->fanin_mask & (1 << channel)) 2808c2ecf20Sopenharmony_ci return 0444; 2818c2ecf20Sopenharmony_ci break; 2828c2ecf20Sopenharmony_ci case hwmon_fan_min: 2838c2ecf20Sopenharmony_ci if (data->fanin_mask & (1 << channel)) 2848c2ecf20Sopenharmony_ci return 0644; 2858c2ecf20Sopenharmony_ci break; 2868c2ecf20Sopenharmony_ci default: 2878c2ecf20Sopenharmony_ci break; 2888c2ecf20Sopenharmony_ci } 2898c2ecf20Sopenharmony_ci 2908c2ecf20Sopenharmony_ci return 0; 2918c2ecf20Sopenharmony_ci} 2928c2ecf20Sopenharmony_ci 2938c2ecf20Sopenharmony_cistatic u8 nct7904_chan_to_index[] = { 2948c2ecf20Sopenharmony_ci 0, /* Not used */ 2958c2ecf20Sopenharmony_ci 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 2968c2ecf20Sopenharmony_ci 18, 19, 20, 16 2978c2ecf20Sopenharmony_ci}; 2988c2ecf20Sopenharmony_ci 2998c2ecf20Sopenharmony_cistatic int nct7904_read_in(struct device *dev, u32 attr, int channel, 3008c2ecf20Sopenharmony_ci long *val) 3018c2ecf20Sopenharmony_ci{ 3028c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 3038c2ecf20Sopenharmony_ci int ret, volt, index; 3048c2ecf20Sopenharmony_ci 3058c2ecf20Sopenharmony_ci index = nct7904_chan_to_index[channel]; 3068c2ecf20Sopenharmony_ci 3078c2ecf20Sopenharmony_ci switch (attr) { 3088c2ecf20Sopenharmony_ci case hwmon_in_input: 3098c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_0, 3108c2ecf20Sopenharmony_ci VSEN1_HV_REG + index * 2); 3118c2ecf20Sopenharmony_ci if (ret < 0) 3128c2ecf20Sopenharmony_ci return ret; 3138c2ecf20Sopenharmony_ci volt = ((ret & 0xff00) >> 5) | (ret & 0x7); 3148c2ecf20Sopenharmony_ci if (index < 14) 3158c2ecf20Sopenharmony_ci volt *= 2; /* 0.002V scale */ 3168c2ecf20Sopenharmony_ci else 3178c2ecf20Sopenharmony_ci volt *= 6; /* 0.006V scale */ 3188c2ecf20Sopenharmony_ci *val = volt; 3198c2ecf20Sopenharmony_ci return 0; 3208c2ecf20Sopenharmony_ci case hwmon_in_min: 3218c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_1, 3228c2ecf20Sopenharmony_ci VSEN1_HV_LL_REG + index * 4); 3238c2ecf20Sopenharmony_ci if (ret < 0) 3248c2ecf20Sopenharmony_ci return ret; 3258c2ecf20Sopenharmony_ci volt = ((ret & 0xff00) >> 5) | (ret & 0x7); 3268c2ecf20Sopenharmony_ci if (index < 14) 3278c2ecf20Sopenharmony_ci volt *= 2; /* 0.002V scale */ 3288c2ecf20Sopenharmony_ci else 3298c2ecf20Sopenharmony_ci volt *= 6; /* 0.006V scale */ 3308c2ecf20Sopenharmony_ci *val = volt; 3318c2ecf20Sopenharmony_ci return 0; 3328c2ecf20Sopenharmony_ci case hwmon_in_max: 3338c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_1, 3348c2ecf20Sopenharmony_ci VSEN1_HV_HL_REG + index * 4); 3358c2ecf20Sopenharmony_ci if (ret < 0) 3368c2ecf20Sopenharmony_ci return ret; 3378c2ecf20Sopenharmony_ci volt = ((ret & 0xff00) >> 5) | (ret & 0x7); 3388c2ecf20Sopenharmony_ci if (index < 14) 3398c2ecf20Sopenharmony_ci volt *= 2; /* 0.002V scale */ 3408c2ecf20Sopenharmony_ci else 3418c2ecf20Sopenharmony_ci volt *= 6; /* 0.006V scale */ 3428c2ecf20Sopenharmony_ci *val = volt; 3438c2ecf20Sopenharmony_ci return 0; 3448c2ecf20Sopenharmony_ci case hwmon_in_alarm: 3458c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, 3468c2ecf20Sopenharmony_ci SMI_STS1_REG + (index >> 3)); 3478c2ecf20Sopenharmony_ci if (ret < 0) 3488c2ecf20Sopenharmony_ci return ret; 3498c2ecf20Sopenharmony_ci if (!data->vsen_alarm[index >> 3]) 3508c2ecf20Sopenharmony_ci data->vsen_alarm[index >> 3] = ret & 0xff; 3518c2ecf20Sopenharmony_ci else 3528c2ecf20Sopenharmony_ci /* If there is new alarm showing up */ 3538c2ecf20Sopenharmony_ci data->vsen_alarm[index >> 3] |= (ret & 0xff); 3548c2ecf20Sopenharmony_ci *val = (data->vsen_alarm[index >> 3] >> (index & 0x07)) & 1; 3558c2ecf20Sopenharmony_ci /* Needs to clean the alarm if alarm existing */ 3568c2ecf20Sopenharmony_ci if (*val) 3578c2ecf20Sopenharmony_ci data->vsen_alarm[index >> 3] ^= 1 << (index & 0x07); 3588c2ecf20Sopenharmony_ci return 0; 3598c2ecf20Sopenharmony_ci default: 3608c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 3618c2ecf20Sopenharmony_ci } 3628c2ecf20Sopenharmony_ci} 3638c2ecf20Sopenharmony_ci 3648c2ecf20Sopenharmony_cistatic umode_t nct7904_in_is_visible(const void *_data, u32 attr, int channel) 3658c2ecf20Sopenharmony_ci{ 3668c2ecf20Sopenharmony_ci const struct nct7904_data *data = _data; 3678c2ecf20Sopenharmony_ci int index = nct7904_chan_to_index[channel]; 3688c2ecf20Sopenharmony_ci 3698c2ecf20Sopenharmony_ci switch (attr) { 3708c2ecf20Sopenharmony_ci case hwmon_in_input: 3718c2ecf20Sopenharmony_ci case hwmon_in_alarm: 3728c2ecf20Sopenharmony_ci if (channel > 0 && (data->vsen_mask & BIT(index))) 3738c2ecf20Sopenharmony_ci return 0444; 3748c2ecf20Sopenharmony_ci break; 3758c2ecf20Sopenharmony_ci case hwmon_in_min: 3768c2ecf20Sopenharmony_ci case hwmon_in_max: 3778c2ecf20Sopenharmony_ci if (channel > 0 && (data->vsen_mask & BIT(index))) 3788c2ecf20Sopenharmony_ci return 0644; 3798c2ecf20Sopenharmony_ci break; 3808c2ecf20Sopenharmony_ci default: 3818c2ecf20Sopenharmony_ci break; 3828c2ecf20Sopenharmony_ci } 3838c2ecf20Sopenharmony_ci 3848c2ecf20Sopenharmony_ci return 0; 3858c2ecf20Sopenharmony_ci} 3868c2ecf20Sopenharmony_ci 3878c2ecf20Sopenharmony_cistatic int nct7904_read_temp(struct device *dev, u32 attr, int channel, 3888c2ecf20Sopenharmony_ci long *val) 3898c2ecf20Sopenharmony_ci{ 3908c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 3918c2ecf20Sopenharmony_ci int ret, temp; 3928c2ecf20Sopenharmony_ci unsigned int reg1, reg2, reg3; 3938c2ecf20Sopenharmony_ci s8 temps; 3948c2ecf20Sopenharmony_ci 3958c2ecf20Sopenharmony_ci switch (attr) { 3968c2ecf20Sopenharmony_ci case hwmon_temp_input: 3978c2ecf20Sopenharmony_ci if (channel == 4) 3988c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_0, LTD_HV_REG); 3998c2ecf20Sopenharmony_ci else if (channel < 5) 4008c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_0, 4018c2ecf20Sopenharmony_ci TEMP_CH1_HV_REG + channel * 4); 4028c2ecf20Sopenharmony_ci else 4038c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_0, 4048c2ecf20Sopenharmony_ci T_CPU1_HV_REG + (channel - 5) 4058c2ecf20Sopenharmony_ci * 2); 4068c2ecf20Sopenharmony_ci if (ret < 0) 4078c2ecf20Sopenharmony_ci return ret; 4088c2ecf20Sopenharmony_ci temp = ((ret & 0xff00) >> 5) | (ret & 0x7); 4098c2ecf20Sopenharmony_ci *val = sign_extend32(temp, 10) * 125; 4108c2ecf20Sopenharmony_ci return 0; 4118c2ecf20Sopenharmony_ci case hwmon_temp_alarm: 4128c2ecf20Sopenharmony_ci if (channel == 4) { 4138c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, 4148c2ecf20Sopenharmony_ci SMI_STS3_REG); 4158c2ecf20Sopenharmony_ci if (ret < 0) 4168c2ecf20Sopenharmony_ci return ret; 4178c2ecf20Sopenharmony_ci *val = (ret >> 1) & 1; 4188c2ecf20Sopenharmony_ci } else if (channel < 4) { 4198c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, 4208c2ecf20Sopenharmony_ci SMI_STS1_REG); 4218c2ecf20Sopenharmony_ci if (ret < 0) 4228c2ecf20Sopenharmony_ci return ret; 4238c2ecf20Sopenharmony_ci *val = (ret >> (((channel * 2) + 1) & 0x07)) & 1; 4248c2ecf20Sopenharmony_ci } else { 4258c2ecf20Sopenharmony_ci if ((channel - 5) < 4) { 4268c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, 4278c2ecf20Sopenharmony_ci SMI_STS7_REG + 4288c2ecf20Sopenharmony_ci ((channel - 5) >> 3)); 4298c2ecf20Sopenharmony_ci if (ret < 0) 4308c2ecf20Sopenharmony_ci return ret; 4318c2ecf20Sopenharmony_ci *val = (ret >> ((channel - 5) & 0x07)) & 1; 4328c2ecf20Sopenharmony_ci } else { 4338c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, 4348c2ecf20Sopenharmony_ci SMI_STS8_REG + 4358c2ecf20Sopenharmony_ci ((channel - 5) >> 3)); 4368c2ecf20Sopenharmony_ci if (ret < 0) 4378c2ecf20Sopenharmony_ci return ret; 4388c2ecf20Sopenharmony_ci *val = (ret >> (((channel - 5) & 0x07) - 4)) 4398c2ecf20Sopenharmony_ci & 1; 4408c2ecf20Sopenharmony_ci } 4418c2ecf20Sopenharmony_ci } 4428c2ecf20Sopenharmony_ci return 0; 4438c2ecf20Sopenharmony_ci case hwmon_temp_type: 4448c2ecf20Sopenharmony_ci if (channel < 5) { 4458c2ecf20Sopenharmony_ci if ((data->tcpu_mask >> channel) & 0x01) { 4468c2ecf20Sopenharmony_ci if ((data->temp_mode >> channel) & 0x01) 4478c2ecf20Sopenharmony_ci *val = 3; /* TD */ 4488c2ecf20Sopenharmony_ci else 4498c2ecf20Sopenharmony_ci *val = 4; /* TR */ 4508c2ecf20Sopenharmony_ci } else { 4518c2ecf20Sopenharmony_ci *val = 0; 4528c2ecf20Sopenharmony_ci } 4538c2ecf20Sopenharmony_ci } else { 4548c2ecf20Sopenharmony_ci if ((data->has_dts >> (channel - 5)) & 0x01) { 4558c2ecf20Sopenharmony_ci if (data->enable_dts & ENABLE_TSI) 4568c2ecf20Sopenharmony_ci *val = 5; /* TSI */ 4578c2ecf20Sopenharmony_ci else 4588c2ecf20Sopenharmony_ci *val = 6; /* PECI */ 4598c2ecf20Sopenharmony_ci } else { 4608c2ecf20Sopenharmony_ci *val = 0; 4618c2ecf20Sopenharmony_ci } 4628c2ecf20Sopenharmony_ci } 4638c2ecf20Sopenharmony_ci return 0; 4648c2ecf20Sopenharmony_ci case hwmon_temp_max: 4658c2ecf20Sopenharmony_ci reg1 = LTD_HV_LL_REG; 4668c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_W_REG; 4678c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_W_REG; 4688c2ecf20Sopenharmony_ci break; 4698c2ecf20Sopenharmony_ci case hwmon_temp_max_hyst: 4708c2ecf20Sopenharmony_ci reg1 = LTD_LV_LL_REG; 4718c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_WH_REG; 4728c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_WH_REG; 4738c2ecf20Sopenharmony_ci break; 4748c2ecf20Sopenharmony_ci case hwmon_temp_crit: 4758c2ecf20Sopenharmony_ci reg1 = LTD_HV_HL_REG; 4768c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_C_REG; 4778c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_C_REG; 4788c2ecf20Sopenharmony_ci break; 4798c2ecf20Sopenharmony_ci case hwmon_temp_crit_hyst: 4808c2ecf20Sopenharmony_ci reg1 = LTD_LV_HL_REG; 4818c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_CH_REG; 4828c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_CH_REG; 4838c2ecf20Sopenharmony_ci break; 4848c2ecf20Sopenharmony_ci default: 4858c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 4868c2ecf20Sopenharmony_ci } 4878c2ecf20Sopenharmony_ci 4888c2ecf20Sopenharmony_ci if (channel == 4) 4898c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_1, reg1); 4908c2ecf20Sopenharmony_ci else if (channel < 5) 4918c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_1, 4928c2ecf20Sopenharmony_ci reg2 + channel * 8); 4938c2ecf20Sopenharmony_ci else 4948c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_1, 4958c2ecf20Sopenharmony_ci reg3 + (channel - 5) * 4); 4968c2ecf20Sopenharmony_ci 4978c2ecf20Sopenharmony_ci if (ret < 0) 4988c2ecf20Sopenharmony_ci return ret; 4998c2ecf20Sopenharmony_ci temps = ret; 5008c2ecf20Sopenharmony_ci *val = temps * 1000; 5018c2ecf20Sopenharmony_ci return 0; 5028c2ecf20Sopenharmony_ci} 5038c2ecf20Sopenharmony_ci 5048c2ecf20Sopenharmony_cistatic umode_t nct7904_temp_is_visible(const void *_data, u32 attr, int channel) 5058c2ecf20Sopenharmony_ci{ 5068c2ecf20Sopenharmony_ci const struct nct7904_data *data = _data; 5078c2ecf20Sopenharmony_ci 5088c2ecf20Sopenharmony_ci switch (attr) { 5098c2ecf20Sopenharmony_ci case hwmon_temp_input: 5108c2ecf20Sopenharmony_ci case hwmon_temp_alarm: 5118c2ecf20Sopenharmony_ci case hwmon_temp_type: 5128c2ecf20Sopenharmony_ci if (channel < 5) { 5138c2ecf20Sopenharmony_ci if (data->tcpu_mask & BIT(channel)) 5148c2ecf20Sopenharmony_ci return 0444; 5158c2ecf20Sopenharmony_ci } else { 5168c2ecf20Sopenharmony_ci if (data->has_dts & BIT(channel - 5)) 5178c2ecf20Sopenharmony_ci return 0444; 5188c2ecf20Sopenharmony_ci } 5198c2ecf20Sopenharmony_ci break; 5208c2ecf20Sopenharmony_ci case hwmon_temp_max: 5218c2ecf20Sopenharmony_ci case hwmon_temp_max_hyst: 5228c2ecf20Sopenharmony_ci case hwmon_temp_crit: 5238c2ecf20Sopenharmony_ci case hwmon_temp_crit_hyst: 5248c2ecf20Sopenharmony_ci if (channel < 5) { 5258c2ecf20Sopenharmony_ci if (data->tcpu_mask & BIT(channel)) 5268c2ecf20Sopenharmony_ci return 0644; 5278c2ecf20Sopenharmony_ci } else { 5288c2ecf20Sopenharmony_ci if (data->has_dts & BIT(channel - 5)) 5298c2ecf20Sopenharmony_ci return 0644; 5308c2ecf20Sopenharmony_ci } 5318c2ecf20Sopenharmony_ci break; 5328c2ecf20Sopenharmony_ci default: 5338c2ecf20Sopenharmony_ci break; 5348c2ecf20Sopenharmony_ci } 5358c2ecf20Sopenharmony_ci 5368c2ecf20Sopenharmony_ci return 0; 5378c2ecf20Sopenharmony_ci} 5388c2ecf20Sopenharmony_ci 5398c2ecf20Sopenharmony_cistatic int nct7904_read_pwm(struct device *dev, u32 attr, int channel, 5408c2ecf20Sopenharmony_ci long *val) 5418c2ecf20Sopenharmony_ci{ 5428c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 5438c2ecf20Sopenharmony_ci int ret; 5448c2ecf20Sopenharmony_ci 5458c2ecf20Sopenharmony_ci switch (attr) { 5468c2ecf20Sopenharmony_ci case hwmon_pwm_input: 5478c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_3, FANCTL1_OUT_REG + channel); 5488c2ecf20Sopenharmony_ci if (ret < 0) 5498c2ecf20Sopenharmony_ci return ret; 5508c2ecf20Sopenharmony_ci *val = ret; 5518c2ecf20Sopenharmony_ci return 0; 5528c2ecf20Sopenharmony_ci case hwmon_pwm_enable: 5538c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_3, FANCTL1_FMR_REG + channel); 5548c2ecf20Sopenharmony_ci if (ret < 0) 5558c2ecf20Sopenharmony_ci return ret; 5568c2ecf20Sopenharmony_ci 5578c2ecf20Sopenharmony_ci *val = ret ? 2 : 1; 5588c2ecf20Sopenharmony_ci return 0; 5598c2ecf20Sopenharmony_ci default: 5608c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 5618c2ecf20Sopenharmony_ci } 5628c2ecf20Sopenharmony_ci} 5638c2ecf20Sopenharmony_ci 5648c2ecf20Sopenharmony_cistatic int nct7904_write_temp(struct device *dev, u32 attr, int channel, 5658c2ecf20Sopenharmony_ci long val) 5668c2ecf20Sopenharmony_ci{ 5678c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 5688c2ecf20Sopenharmony_ci int ret; 5698c2ecf20Sopenharmony_ci unsigned int reg1, reg2, reg3; 5708c2ecf20Sopenharmony_ci 5718c2ecf20Sopenharmony_ci val = clamp_val(val / 1000, -128, 127); 5728c2ecf20Sopenharmony_ci 5738c2ecf20Sopenharmony_ci switch (attr) { 5748c2ecf20Sopenharmony_ci case hwmon_temp_max: 5758c2ecf20Sopenharmony_ci reg1 = LTD_HV_LL_REG; 5768c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_W_REG; 5778c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_W_REG; 5788c2ecf20Sopenharmony_ci break; 5798c2ecf20Sopenharmony_ci case hwmon_temp_max_hyst: 5808c2ecf20Sopenharmony_ci reg1 = LTD_LV_LL_REG; 5818c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_WH_REG; 5828c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_WH_REG; 5838c2ecf20Sopenharmony_ci break; 5848c2ecf20Sopenharmony_ci case hwmon_temp_crit: 5858c2ecf20Sopenharmony_ci reg1 = LTD_HV_HL_REG; 5868c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_C_REG; 5878c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_C_REG; 5888c2ecf20Sopenharmony_ci break; 5898c2ecf20Sopenharmony_ci case hwmon_temp_crit_hyst: 5908c2ecf20Sopenharmony_ci reg1 = LTD_LV_HL_REG; 5918c2ecf20Sopenharmony_ci reg2 = TEMP_CH1_CH_REG; 5928c2ecf20Sopenharmony_ci reg3 = DTS_T_CPU1_CH_REG; 5938c2ecf20Sopenharmony_ci break; 5948c2ecf20Sopenharmony_ci default: 5958c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 5968c2ecf20Sopenharmony_ci } 5978c2ecf20Sopenharmony_ci if (channel == 4) 5988c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, reg1, val); 5998c2ecf20Sopenharmony_ci else if (channel < 5) 6008c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6018c2ecf20Sopenharmony_ci reg2 + channel * 8, val); 6028c2ecf20Sopenharmony_ci else 6038c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6048c2ecf20Sopenharmony_ci reg3 + (channel - 5) * 4, val); 6058c2ecf20Sopenharmony_ci 6068c2ecf20Sopenharmony_ci return ret; 6078c2ecf20Sopenharmony_ci} 6088c2ecf20Sopenharmony_ci 6098c2ecf20Sopenharmony_cistatic int nct7904_write_fan(struct device *dev, u32 attr, int channel, 6108c2ecf20Sopenharmony_ci long val) 6118c2ecf20Sopenharmony_ci{ 6128c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 6138c2ecf20Sopenharmony_ci int ret; 6148c2ecf20Sopenharmony_ci u8 tmp; 6158c2ecf20Sopenharmony_ci 6168c2ecf20Sopenharmony_ci switch (attr) { 6178c2ecf20Sopenharmony_ci case hwmon_fan_min: 6188c2ecf20Sopenharmony_ci if (val <= 0) 6198c2ecf20Sopenharmony_ci return -EINVAL; 6208c2ecf20Sopenharmony_ci 6218c2ecf20Sopenharmony_ci val = clamp_val(DIV_ROUND_CLOSEST(1350000, val), 1, 0x1fff); 6228c2ecf20Sopenharmony_ci tmp = (val >> 5) & 0xff; 6238c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6248c2ecf20Sopenharmony_ci FANIN1_HV_HL_REG + channel * 2, tmp); 6258c2ecf20Sopenharmony_ci if (ret < 0) 6268c2ecf20Sopenharmony_ci return ret; 6278c2ecf20Sopenharmony_ci tmp = val & 0x1f; 6288c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6298c2ecf20Sopenharmony_ci FANIN1_LV_HL_REG + channel * 2, tmp); 6308c2ecf20Sopenharmony_ci return ret; 6318c2ecf20Sopenharmony_ci default: 6328c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 6338c2ecf20Sopenharmony_ci } 6348c2ecf20Sopenharmony_ci} 6358c2ecf20Sopenharmony_ci 6368c2ecf20Sopenharmony_cistatic int nct7904_write_in(struct device *dev, u32 attr, int channel, 6378c2ecf20Sopenharmony_ci long val) 6388c2ecf20Sopenharmony_ci{ 6398c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 6408c2ecf20Sopenharmony_ci int ret, index, tmp; 6418c2ecf20Sopenharmony_ci 6428c2ecf20Sopenharmony_ci index = nct7904_chan_to_index[channel]; 6438c2ecf20Sopenharmony_ci 6448c2ecf20Sopenharmony_ci if (index < 14) 6458c2ecf20Sopenharmony_ci val = val / 2; /* 0.002V scale */ 6468c2ecf20Sopenharmony_ci else 6478c2ecf20Sopenharmony_ci val = val / 6; /* 0.006V scale */ 6488c2ecf20Sopenharmony_ci 6498c2ecf20Sopenharmony_ci val = clamp_val(val, 0, 0x7ff); 6508c2ecf20Sopenharmony_ci 6518c2ecf20Sopenharmony_ci switch (attr) { 6528c2ecf20Sopenharmony_ci case hwmon_in_min: 6538c2ecf20Sopenharmony_ci tmp = nct7904_read_reg(data, BANK_1, 6548c2ecf20Sopenharmony_ci VSEN1_LV_LL_REG + index * 4); 6558c2ecf20Sopenharmony_ci if (tmp < 0) 6568c2ecf20Sopenharmony_ci return tmp; 6578c2ecf20Sopenharmony_ci tmp &= ~0x7; 6588c2ecf20Sopenharmony_ci tmp |= val & 0x7; 6598c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6608c2ecf20Sopenharmony_ci VSEN1_LV_LL_REG + index * 4, tmp); 6618c2ecf20Sopenharmony_ci if (ret < 0) 6628c2ecf20Sopenharmony_ci return ret; 6638c2ecf20Sopenharmony_ci tmp = nct7904_read_reg(data, BANK_1, 6648c2ecf20Sopenharmony_ci VSEN1_HV_LL_REG + index * 4); 6658c2ecf20Sopenharmony_ci if (tmp < 0) 6668c2ecf20Sopenharmony_ci return tmp; 6678c2ecf20Sopenharmony_ci tmp = (val >> 3) & 0xff; 6688c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6698c2ecf20Sopenharmony_ci VSEN1_HV_LL_REG + index * 4, tmp); 6708c2ecf20Sopenharmony_ci return ret; 6718c2ecf20Sopenharmony_ci case hwmon_in_max: 6728c2ecf20Sopenharmony_ci tmp = nct7904_read_reg(data, BANK_1, 6738c2ecf20Sopenharmony_ci VSEN1_LV_HL_REG + index * 4); 6748c2ecf20Sopenharmony_ci if (tmp < 0) 6758c2ecf20Sopenharmony_ci return tmp; 6768c2ecf20Sopenharmony_ci tmp &= ~0x7; 6778c2ecf20Sopenharmony_ci tmp |= val & 0x7; 6788c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6798c2ecf20Sopenharmony_ci VSEN1_LV_HL_REG + index * 4, tmp); 6808c2ecf20Sopenharmony_ci if (ret < 0) 6818c2ecf20Sopenharmony_ci return ret; 6828c2ecf20Sopenharmony_ci tmp = nct7904_read_reg(data, BANK_1, 6838c2ecf20Sopenharmony_ci VSEN1_HV_HL_REG + index * 4); 6848c2ecf20Sopenharmony_ci if (tmp < 0) 6858c2ecf20Sopenharmony_ci return tmp; 6868c2ecf20Sopenharmony_ci tmp = (val >> 3) & 0xff; 6878c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_1, 6888c2ecf20Sopenharmony_ci VSEN1_HV_HL_REG + index * 4, tmp); 6898c2ecf20Sopenharmony_ci return ret; 6908c2ecf20Sopenharmony_ci default: 6918c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 6928c2ecf20Sopenharmony_ci } 6938c2ecf20Sopenharmony_ci} 6948c2ecf20Sopenharmony_ci 6958c2ecf20Sopenharmony_cistatic int nct7904_write_pwm(struct device *dev, u32 attr, int channel, 6968c2ecf20Sopenharmony_ci long val) 6978c2ecf20Sopenharmony_ci{ 6988c2ecf20Sopenharmony_ci struct nct7904_data *data = dev_get_drvdata(dev); 6998c2ecf20Sopenharmony_ci int ret; 7008c2ecf20Sopenharmony_ci 7018c2ecf20Sopenharmony_ci switch (attr) { 7028c2ecf20Sopenharmony_ci case hwmon_pwm_input: 7038c2ecf20Sopenharmony_ci if (val < 0 || val > 255) 7048c2ecf20Sopenharmony_ci return -EINVAL; 7058c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_3, FANCTL1_OUT_REG + channel, 7068c2ecf20Sopenharmony_ci val); 7078c2ecf20Sopenharmony_ci return ret; 7088c2ecf20Sopenharmony_ci case hwmon_pwm_enable: 7098c2ecf20Sopenharmony_ci if (val < 1 || val > 2 || 7108c2ecf20Sopenharmony_ci (val == 2 && !data->fan_mode[channel])) 7118c2ecf20Sopenharmony_ci return -EINVAL; 7128c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_3, FANCTL1_FMR_REG + channel, 7138c2ecf20Sopenharmony_ci val == 2 ? data->fan_mode[channel] : 0); 7148c2ecf20Sopenharmony_ci return ret; 7158c2ecf20Sopenharmony_ci default: 7168c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 7178c2ecf20Sopenharmony_ci } 7188c2ecf20Sopenharmony_ci} 7198c2ecf20Sopenharmony_ci 7208c2ecf20Sopenharmony_cistatic umode_t nct7904_pwm_is_visible(const void *_data, u32 attr, int channel) 7218c2ecf20Sopenharmony_ci{ 7228c2ecf20Sopenharmony_ci switch (attr) { 7238c2ecf20Sopenharmony_ci case hwmon_pwm_input: 7248c2ecf20Sopenharmony_ci case hwmon_pwm_enable: 7258c2ecf20Sopenharmony_ci return 0644; 7268c2ecf20Sopenharmony_ci default: 7278c2ecf20Sopenharmony_ci return 0; 7288c2ecf20Sopenharmony_ci } 7298c2ecf20Sopenharmony_ci} 7308c2ecf20Sopenharmony_ci 7318c2ecf20Sopenharmony_cistatic int nct7904_read(struct device *dev, enum hwmon_sensor_types type, 7328c2ecf20Sopenharmony_ci u32 attr, int channel, long *val) 7338c2ecf20Sopenharmony_ci{ 7348c2ecf20Sopenharmony_ci switch (type) { 7358c2ecf20Sopenharmony_ci case hwmon_in: 7368c2ecf20Sopenharmony_ci return nct7904_read_in(dev, attr, channel, val); 7378c2ecf20Sopenharmony_ci case hwmon_fan: 7388c2ecf20Sopenharmony_ci return nct7904_read_fan(dev, attr, channel, val); 7398c2ecf20Sopenharmony_ci case hwmon_pwm: 7408c2ecf20Sopenharmony_ci return nct7904_read_pwm(dev, attr, channel, val); 7418c2ecf20Sopenharmony_ci case hwmon_temp: 7428c2ecf20Sopenharmony_ci return nct7904_read_temp(dev, attr, channel, val); 7438c2ecf20Sopenharmony_ci default: 7448c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 7458c2ecf20Sopenharmony_ci } 7468c2ecf20Sopenharmony_ci} 7478c2ecf20Sopenharmony_ci 7488c2ecf20Sopenharmony_cistatic int nct7904_write(struct device *dev, enum hwmon_sensor_types type, 7498c2ecf20Sopenharmony_ci u32 attr, int channel, long val) 7508c2ecf20Sopenharmony_ci{ 7518c2ecf20Sopenharmony_ci switch (type) { 7528c2ecf20Sopenharmony_ci case hwmon_in: 7538c2ecf20Sopenharmony_ci return nct7904_write_in(dev, attr, channel, val); 7548c2ecf20Sopenharmony_ci case hwmon_fan: 7558c2ecf20Sopenharmony_ci return nct7904_write_fan(dev, attr, channel, val); 7568c2ecf20Sopenharmony_ci case hwmon_pwm: 7578c2ecf20Sopenharmony_ci return nct7904_write_pwm(dev, attr, channel, val); 7588c2ecf20Sopenharmony_ci case hwmon_temp: 7598c2ecf20Sopenharmony_ci return nct7904_write_temp(dev, attr, channel, val); 7608c2ecf20Sopenharmony_ci default: 7618c2ecf20Sopenharmony_ci return -EOPNOTSUPP; 7628c2ecf20Sopenharmony_ci } 7638c2ecf20Sopenharmony_ci} 7648c2ecf20Sopenharmony_ci 7658c2ecf20Sopenharmony_cistatic umode_t nct7904_is_visible(const void *data, 7668c2ecf20Sopenharmony_ci enum hwmon_sensor_types type, 7678c2ecf20Sopenharmony_ci u32 attr, int channel) 7688c2ecf20Sopenharmony_ci{ 7698c2ecf20Sopenharmony_ci switch (type) { 7708c2ecf20Sopenharmony_ci case hwmon_in: 7718c2ecf20Sopenharmony_ci return nct7904_in_is_visible(data, attr, channel); 7728c2ecf20Sopenharmony_ci case hwmon_fan: 7738c2ecf20Sopenharmony_ci return nct7904_fan_is_visible(data, attr, channel); 7748c2ecf20Sopenharmony_ci case hwmon_pwm: 7758c2ecf20Sopenharmony_ci return nct7904_pwm_is_visible(data, attr, channel); 7768c2ecf20Sopenharmony_ci case hwmon_temp: 7778c2ecf20Sopenharmony_ci return nct7904_temp_is_visible(data, attr, channel); 7788c2ecf20Sopenharmony_ci default: 7798c2ecf20Sopenharmony_ci return 0; 7808c2ecf20Sopenharmony_ci } 7818c2ecf20Sopenharmony_ci} 7828c2ecf20Sopenharmony_ci 7838c2ecf20Sopenharmony_ci/* Return 0 if detection is successful, -ENODEV otherwise */ 7848c2ecf20Sopenharmony_cistatic int nct7904_detect(struct i2c_client *client, 7858c2ecf20Sopenharmony_ci struct i2c_board_info *info) 7868c2ecf20Sopenharmony_ci{ 7878c2ecf20Sopenharmony_ci struct i2c_adapter *adapter = client->adapter; 7888c2ecf20Sopenharmony_ci 7898c2ecf20Sopenharmony_ci if (!i2c_check_functionality(adapter, 7908c2ecf20Sopenharmony_ci I2C_FUNC_SMBUS_READ_BYTE | 7918c2ecf20Sopenharmony_ci I2C_FUNC_SMBUS_WRITE_BYTE_DATA)) 7928c2ecf20Sopenharmony_ci return -ENODEV; 7938c2ecf20Sopenharmony_ci 7948c2ecf20Sopenharmony_ci /* Determine the chip type. */ 7958c2ecf20Sopenharmony_ci if (i2c_smbus_read_byte_data(client, VENDOR_ID_REG) != NUVOTON_ID || 7968c2ecf20Sopenharmony_ci i2c_smbus_read_byte_data(client, CHIP_ID_REG) != NCT7904_ID || 7978c2ecf20Sopenharmony_ci (i2c_smbus_read_byte_data(client, DEVICE_ID_REG) & 0xf0) != 0x50 || 7988c2ecf20Sopenharmony_ci (i2c_smbus_read_byte_data(client, BANK_SEL_REG) & 0xf8) != 0x00) 7998c2ecf20Sopenharmony_ci return -ENODEV; 8008c2ecf20Sopenharmony_ci 8018c2ecf20Sopenharmony_ci strlcpy(info->type, "nct7904", I2C_NAME_SIZE); 8028c2ecf20Sopenharmony_ci 8038c2ecf20Sopenharmony_ci return 0; 8048c2ecf20Sopenharmony_ci} 8058c2ecf20Sopenharmony_ci 8068c2ecf20Sopenharmony_cistatic const struct hwmon_channel_info *nct7904_info[] = { 8078c2ecf20Sopenharmony_ci HWMON_CHANNEL_INFO(in, 8088c2ecf20Sopenharmony_ci /* dummy, skipped in is_visible */ 8098c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8108c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8118c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8128c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8138c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8148c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8158c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8168c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8178c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8188c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8198c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8208c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8218c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8228c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8238c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8248c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8258c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8268c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8278c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8288c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8298c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8308c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8318c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8328c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8338c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8348c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8358c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8368c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8378c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8388c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8398c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8408c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8418c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8428c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8438c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8448c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8458c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8468c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8478c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8488c2ecf20Sopenharmony_ci HWMON_I_ALARM, 8498c2ecf20Sopenharmony_ci HWMON_I_INPUT | HWMON_I_MIN | HWMON_I_MAX | 8508c2ecf20Sopenharmony_ci HWMON_I_ALARM), 8518c2ecf20Sopenharmony_ci HWMON_CHANNEL_INFO(fan, 8528c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8538c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8548c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8558c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8568c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8578c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8588c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8598c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8608c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8618c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8628c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM, 8638c2ecf20Sopenharmony_ci HWMON_F_INPUT | HWMON_F_MIN | HWMON_F_ALARM), 8648c2ecf20Sopenharmony_ci HWMON_CHANNEL_INFO(pwm, 8658c2ecf20Sopenharmony_ci HWMON_PWM_INPUT | HWMON_PWM_ENABLE, 8668c2ecf20Sopenharmony_ci HWMON_PWM_INPUT | HWMON_PWM_ENABLE, 8678c2ecf20Sopenharmony_ci HWMON_PWM_INPUT | HWMON_PWM_ENABLE, 8688c2ecf20Sopenharmony_ci HWMON_PWM_INPUT | HWMON_PWM_ENABLE), 8698c2ecf20Sopenharmony_ci HWMON_CHANNEL_INFO(temp, 8708c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8718c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8728c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8738c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8748c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8758c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8768c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8778c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8788c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8798c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8808c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8818c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8828c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8838c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8848c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8858c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8868c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8878c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8888c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8898c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8908c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8918c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8928c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8938c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8948c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8958c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8968c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 8978c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 8988c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 8998c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 9008c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 9018c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 9028c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 9038c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 9048c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 9058c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST, 9068c2ecf20Sopenharmony_ci HWMON_T_INPUT | HWMON_T_ALARM | HWMON_T_MAX | 9078c2ecf20Sopenharmony_ci HWMON_T_MAX_HYST | HWMON_T_TYPE | HWMON_T_CRIT | 9088c2ecf20Sopenharmony_ci HWMON_T_CRIT_HYST), 9098c2ecf20Sopenharmony_ci NULL 9108c2ecf20Sopenharmony_ci}; 9118c2ecf20Sopenharmony_ci 9128c2ecf20Sopenharmony_cistatic const struct hwmon_ops nct7904_hwmon_ops = { 9138c2ecf20Sopenharmony_ci .is_visible = nct7904_is_visible, 9148c2ecf20Sopenharmony_ci .read = nct7904_read, 9158c2ecf20Sopenharmony_ci .write = nct7904_write, 9168c2ecf20Sopenharmony_ci}; 9178c2ecf20Sopenharmony_ci 9188c2ecf20Sopenharmony_cistatic const struct hwmon_chip_info nct7904_chip_info = { 9198c2ecf20Sopenharmony_ci .ops = &nct7904_hwmon_ops, 9208c2ecf20Sopenharmony_ci .info = nct7904_info, 9218c2ecf20Sopenharmony_ci}; 9228c2ecf20Sopenharmony_ci 9238c2ecf20Sopenharmony_ci/* 9248c2ecf20Sopenharmony_ci * Watchdog Function 9258c2ecf20Sopenharmony_ci */ 9268c2ecf20Sopenharmony_cistatic int nct7904_wdt_start(struct watchdog_device *wdt) 9278c2ecf20Sopenharmony_ci{ 9288c2ecf20Sopenharmony_ci struct nct7904_data *data = watchdog_get_drvdata(wdt); 9298c2ecf20Sopenharmony_ci 9308c2ecf20Sopenharmony_ci /* Enable soft watchdog timer */ 9318c2ecf20Sopenharmony_ci return nct7904_write_reg(data, BANK_0, WDT_LOCK_REG, WDT_SOFT_EN); 9328c2ecf20Sopenharmony_ci} 9338c2ecf20Sopenharmony_ci 9348c2ecf20Sopenharmony_cistatic int nct7904_wdt_stop(struct watchdog_device *wdt) 9358c2ecf20Sopenharmony_ci{ 9368c2ecf20Sopenharmony_ci struct nct7904_data *data = watchdog_get_drvdata(wdt); 9378c2ecf20Sopenharmony_ci 9388c2ecf20Sopenharmony_ci return nct7904_write_reg(data, BANK_0, WDT_LOCK_REG, WDT_SOFT_DIS); 9398c2ecf20Sopenharmony_ci} 9408c2ecf20Sopenharmony_ci 9418c2ecf20Sopenharmony_cistatic int nct7904_wdt_set_timeout(struct watchdog_device *wdt, 9428c2ecf20Sopenharmony_ci unsigned int timeout) 9438c2ecf20Sopenharmony_ci{ 9448c2ecf20Sopenharmony_ci struct nct7904_data *data = watchdog_get_drvdata(wdt); 9458c2ecf20Sopenharmony_ci /* 9468c2ecf20Sopenharmony_ci * The NCT7904 is very special in watchdog function. 9478c2ecf20Sopenharmony_ci * Its minimum unit is minutes. And wdt->timeout needs 9488c2ecf20Sopenharmony_ci * to match the actual timeout selected. So, this needs 9498c2ecf20Sopenharmony_ci * to be: wdt->timeout = timeout / 60 * 60. 9508c2ecf20Sopenharmony_ci * For example, if the user configures a timeout of 9518c2ecf20Sopenharmony_ci * 119 seconds, the actual timeout will be 60 seconds. 9528c2ecf20Sopenharmony_ci * So, wdt->timeout must then be set to 60 seconds. 9538c2ecf20Sopenharmony_ci */ 9548c2ecf20Sopenharmony_ci wdt->timeout = timeout / 60 * 60; 9558c2ecf20Sopenharmony_ci 9568c2ecf20Sopenharmony_ci return nct7904_write_reg(data, BANK_0, WDT_TIMER_REG, 9578c2ecf20Sopenharmony_ci wdt->timeout / 60); 9588c2ecf20Sopenharmony_ci} 9598c2ecf20Sopenharmony_ci 9608c2ecf20Sopenharmony_cistatic int nct7904_wdt_ping(struct watchdog_device *wdt) 9618c2ecf20Sopenharmony_ci{ 9628c2ecf20Sopenharmony_ci /* 9638c2ecf20Sopenharmony_ci * Note: 9648c2ecf20Sopenharmony_ci * NCT7904 does not support refreshing WDT_TIMER_REG register when 9658c2ecf20Sopenharmony_ci * the watchdog is active. Please disable watchdog before feeding 9668c2ecf20Sopenharmony_ci * the watchdog and enable it again. 9678c2ecf20Sopenharmony_ci */ 9688c2ecf20Sopenharmony_ci struct nct7904_data *data = watchdog_get_drvdata(wdt); 9698c2ecf20Sopenharmony_ci int ret; 9708c2ecf20Sopenharmony_ci 9718c2ecf20Sopenharmony_ci /* Disable soft watchdog timer */ 9728c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_0, WDT_LOCK_REG, WDT_SOFT_DIS); 9738c2ecf20Sopenharmony_ci if (ret < 0) 9748c2ecf20Sopenharmony_ci return ret; 9758c2ecf20Sopenharmony_ci 9768c2ecf20Sopenharmony_ci /* feed watchdog */ 9778c2ecf20Sopenharmony_ci ret = nct7904_write_reg(data, BANK_0, WDT_TIMER_REG, wdt->timeout / 60); 9788c2ecf20Sopenharmony_ci if (ret < 0) 9798c2ecf20Sopenharmony_ci return ret; 9808c2ecf20Sopenharmony_ci 9818c2ecf20Sopenharmony_ci /* Enable soft watchdog timer */ 9828c2ecf20Sopenharmony_ci return nct7904_write_reg(data, BANK_0, WDT_LOCK_REG, WDT_SOFT_EN); 9838c2ecf20Sopenharmony_ci} 9848c2ecf20Sopenharmony_ci 9858c2ecf20Sopenharmony_cistatic unsigned int nct7904_wdt_get_timeleft(struct watchdog_device *wdt) 9868c2ecf20Sopenharmony_ci{ 9878c2ecf20Sopenharmony_ci struct nct7904_data *data = watchdog_get_drvdata(wdt); 9888c2ecf20Sopenharmony_ci int ret; 9898c2ecf20Sopenharmony_ci 9908c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, WDT_TIMER_REG); 9918c2ecf20Sopenharmony_ci if (ret < 0) 9928c2ecf20Sopenharmony_ci return 0; 9938c2ecf20Sopenharmony_ci 9948c2ecf20Sopenharmony_ci return ret * 60; 9958c2ecf20Sopenharmony_ci} 9968c2ecf20Sopenharmony_ci 9978c2ecf20Sopenharmony_cistatic const struct watchdog_info nct7904_wdt_info = { 9988c2ecf20Sopenharmony_ci .options = WDIOF_SETTIMEOUT | WDIOF_KEEPALIVEPING | 9998c2ecf20Sopenharmony_ci WDIOF_MAGICCLOSE, 10008c2ecf20Sopenharmony_ci .identity = "nct7904 watchdog", 10018c2ecf20Sopenharmony_ci}; 10028c2ecf20Sopenharmony_ci 10038c2ecf20Sopenharmony_cistatic const struct watchdog_ops nct7904_wdt_ops = { 10048c2ecf20Sopenharmony_ci .owner = THIS_MODULE, 10058c2ecf20Sopenharmony_ci .start = nct7904_wdt_start, 10068c2ecf20Sopenharmony_ci .stop = nct7904_wdt_stop, 10078c2ecf20Sopenharmony_ci .ping = nct7904_wdt_ping, 10088c2ecf20Sopenharmony_ci .set_timeout = nct7904_wdt_set_timeout, 10098c2ecf20Sopenharmony_ci .get_timeleft = nct7904_wdt_get_timeleft, 10108c2ecf20Sopenharmony_ci}; 10118c2ecf20Sopenharmony_ci 10128c2ecf20Sopenharmony_cistatic int nct7904_probe(struct i2c_client *client) 10138c2ecf20Sopenharmony_ci{ 10148c2ecf20Sopenharmony_ci struct nct7904_data *data; 10158c2ecf20Sopenharmony_ci struct device *hwmon_dev; 10168c2ecf20Sopenharmony_ci struct device *dev = &client->dev; 10178c2ecf20Sopenharmony_ci int ret, i; 10188c2ecf20Sopenharmony_ci u32 mask; 10198c2ecf20Sopenharmony_ci u8 val, bit; 10208c2ecf20Sopenharmony_ci 10218c2ecf20Sopenharmony_ci data = devm_kzalloc(dev, sizeof(struct nct7904_data), GFP_KERNEL); 10228c2ecf20Sopenharmony_ci if (!data) 10238c2ecf20Sopenharmony_ci return -ENOMEM; 10248c2ecf20Sopenharmony_ci 10258c2ecf20Sopenharmony_ci data->client = client; 10268c2ecf20Sopenharmony_ci mutex_init(&data->bank_lock); 10278c2ecf20Sopenharmony_ci data->bank_sel = -1; 10288c2ecf20Sopenharmony_ci 10298c2ecf20Sopenharmony_ci /* Setup sensor groups. */ 10308c2ecf20Sopenharmony_ci /* FANIN attributes */ 10318c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_0, FANIN_CTRL0_REG); 10328c2ecf20Sopenharmony_ci if (ret < 0) 10338c2ecf20Sopenharmony_ci return ret; 10348c2ecf20Sopenharmony_ci data->fanin_mask = (ret >> 8) | ((ret & 0xff) << 8); 10358c2ecf20Sopenharmony_ci 10368c2ecf20Sopenharmony_ci /* 10378c2ecf20Sopenharmony_ci * VSEN attributes 10388c2ecf20Sopenharmony_ci * 10398c2ecf20Sopenharmony_ci * Note: voltage sensors overlap with external temperature 10408c2ecf20Sopenharmony_ci * sensors. So, if we ever decide to support the latter 10418c2ecf20Sopenharmony_ci * we will have to adjust 'vsen_mask' accordingly. 10428c2ecf20Sopenharmony_ci */ 10438c2ecf20Sopenharmony_ci mask = 0; 10448c2ecf20Sopenharmony_ci ret = nct7904_read_reg16(data, BANK_0, VT_ADC_CTRL0_REG); 10458c2ecf20Sopenharmony_ci if (ret >= 0) 10468c2ecf20Sopenharmony_ci mask = (ret >> 8) | ((ret & 0xff) << 8); 10478c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, VT_ADC_CTRL2_REG); 10488c2ecf20Sopenharmony_ci if (ret >= 0) 10498c2ecf20Sopenharmony_ci mask |= (ret << 16); 10508c2ecf20Sopenharmony_ci data->vsen_mask = mask; 10518c2ecf20Sopenharmony_ci 10528c2ecf20Sopenharmony_ci /* CPU_TEMP attributes */ 10538c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, VT_ADC_CTRL0_REG); 10548c2ecf20Sopenharmony_ci if (ret < 0) 10558c2ecf20Sopenharmony_ci return ret; 10568c2ecf20Sopenharmony_ci 10578c2ecf20Sopenharmony_ci if ((ret & 0x6) == 0x6) 10588c2ecf20Sopenharmony_ci data->tcpu_mask |= 1; /* TR1 */ 10598c2ecf20Sopenharmony_ci if ((ret & 0x18) == 0x18) 10608c2ecf20Sopenharmony_ci data->tcpu_mask |= 2; /* TR2 */ 10618c2ecf20Sopenharmony_ci if ((ret & 0x20) == 0x20) 10628c2ecf20Sopenharmony_ci data->tcpu_mask |= 4; /* TR3 */ 10638c2ecf20Sopenharmony_ci if ((ret & 0x80) == 0x80) 10648c2ecf20Sopenharmony_ci data->tcpu_mask |= 8; /* TR4 */ 10658c2ecf20Sopenharmony_ci 10668c2ecf20Sopenharmony_ci /* LTD */ 10678c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, VT_ADC_CTRL2_REG); 10688c2ecf20Sopenharmony_ci if (ret < 0) 10698c2ecf20Sopenharmony_ci return ret; 10708c2ecf20Sopenharmony_ci if ((ret & 0x02) == 0x02) 10718c2ecf20Sopenharmony_ci data->tcpu_mask |= 0x10; 10728c2ecf20Sopenharmony_ci 10738c2ecf20Sopenharmony_ci /* Multi-Function detecting for Volt and TR/TD */ 10748c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, VT_ADC_MD_REG); 10758c2ecf20Sopenharmony_ci if (ret < 0) 10768c2ecf20Sopenharmony_ci return ret; 10778c2ecf20Sopenharmony_ci 10788c2ecf20Sopenharmony_ci data->temp_mode = 0; 10798c2ecf20Sopenharmony_ci for (i = 0; i < 4; i++) { 10808c2ecf20Sopenharmony_ci val = (ret >> (i * 2)) & 0x03; 10818c2ecf20Sopenharmony_ci bit = (1 << i); 10828c2ecf20Sopenharmony_ci if (val == VOLT_MONITOR_MODE) { 10838c2ecf20Sopenharmony_ci data->tcpu_mask &= ~bit; 10848c2ecf20Sopenharmony_ci } else if (val == THERMAL_DIODE_MODE && i < 2) { 10858c2ecf20Sopenharmony_ci data->temp_mode |= bit; 10868c2ecf20Sopenharmony_ci data->vsen_mask &= ~(0x06 << (i * 2)); 10878c2ecf20Sopenharmony_ci } else if (val == THERMISTOR_MODE) { 10888c2ecf20Sopenharmony_ci data->vsen_mask &= ~(0x02 << (i * 2)); 10898c2ecf20Sopenharmony_ci } else { 10908c2ecf20Sopenharmony_ci /* Reserved */ 10918c2ecf20Sopenharmony_ci data->tcpu_mask &= ~bit; 10928c2ecf20Sopenharmony_ci data->vsen_mask &= ~(0x06 << (i * 2)); 10938c2ecf20Sopenharmony_ci } 10948c2ecf20Sopenharmony_ci } 10958c2ecf20Sopenharmony_ci 10968c2ecf20Sopenharmony_ci /* PECI */ 10978c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_2, PFE_REG); 10988c2ecf20Sopenharmony_ci if (ret < 0) 10998c2ecf20Sopenharmony_ci return ret; 11008c2ecf20Sopenharmony_ci if (ret & 0x80) { 11018c2ecf20Sopenharmony_ci data->enable_dts = 1; /* Enable DTS & PECI */ 11028c2ecf20Sopenharmony_ci } else { 11038c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_2, TSI_CTRL_REG); 11048c2ecf20Sopenharmony_ci if (ret < 0) 11058c2ecf20Sopenharmony_ci return ret; 11068c2ecf20Sopenharmony_ci if (ret & 0x80) 11078c2ecf20Sopenharmony_ci data->enable_dts = 0x3; /* Enable DTS & TSI */ 11088c2ecf20Sopenharmony_ci } 11098c2ecf20Sopenharmony_ci 11108c2ecf20Sopenharmony_ci /* Check DTS enable status */ 11118c2ecf20Sopenharmony_ci if (data->enable_dts) { 11128c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, DTS_T_CTRL0_REG); 11138c2ecf20Sopenharmony_ci if (ret < 0) 11148c2ecf20Sopenharmony_ci return ret; 11158c2ecf20Sopenharmony_ci data->has_dts = ret & 0xF; 11168c2ecf20Sopenharmony_ci if (data->enable_dts & ENABLE_TSI) { 11178c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, DTS_T_CTRL1_REG); 11188c2ecf20Sopenharmony_ci if (ret < 0) 11198c2ecf20Sopenharmony_ci return ret; 11208c2ecf20Sopenharmony_ci data->has_dts |= (ret & 0xF) << 4; 11218c2ecf20Sopenharmony_ci } 11228c2ecf20Sopenharmony_ci } 11238c2ecf20Sopenharmony_ci 11248c2ecf20Sopenharmony_ci for (i = 0; i < FANCTL_MAX; i++) { 11258c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_3, FANCTL1_FMR_REG + i); 11268c2ecf20Sopenharmony_ci if (ret < 0) 11278c2ecf20Sopenharmony_ci return ret; 11288c2ecf20Sopenharmony_ci data->fan_mode[i] = ret; 11298c2ecf20Sopenharmony_ci } 11308c2ecf20Sopenharmony_ci 11318c2ecf20Sopenharmony_ci /* Read all of SMI status register to clear alarms */ 11328c2ecf20Sopenharmony_ci for (i = 0; i < SMI_STS_MAX; i++) { 11338c2ecf20Sopenharmony_ci ret = nct7904_read_reg(data, BANK_0, SMI_STS1_REG + i); 11348c2ecf20Sopenharmony_ci if (ret < 0) 11358c2ecf20Sopenharmony_ci return ret; 11368c2ecf20Sopenharmony_ci } 11378c2ecf20Sopenharmony_ci 11388c2ecf20Sopenharmony_ci hwmon_dev = 11398c2ecf20Sopenharmony_ci devm_hwmon_device_register_with_info(dev, client->name, data, 11408c2ecf20Sopenharmony_ci &nct7904_chip_info, NULL); 11418c2ecf20Sopenharmony_ci ret = PTR_ERR_OR_ZERO(hwmon_dev); 11428c2ecf20Sopenharmony_ci if (ret) 11438c2ecf20Sopenharmony_ci return ret; 11448c2ecf20Sopenharmony_ci 11458c2ecf20Sopenharmony_ci /* Watchdog initialization */ 11468c2ecf20Sopenharmony_ci data->wdt.ops = &nct7904_wdt_ops; 11478c2ecf20Sopenharmony_ci data->wdt.info = &nct7904_wdt_info; 11488c2ecf20Sopenharmony_ci 11498c2ecf20Sopenharmony_ci data->wdt.timeout = WATCHDOG_TIMEOUT * 60; /* Set default timeout */ 11508c2ecf20Sopenharmony_ci data->wdt.min_timeout = MIN_TIMEOUT; 11518c2ecf20Sopenharmony_ci data->wdt.max_timeout = MAX_TIMEOUT; 11528c2ecf20Sopenharmony_ci data->wdt.parent = &client->dev; 11538c2ecf20Sopenharmony_ci 11548c2ecf20Sopenharmony_ci watchdog_init_timeout(&data->wdt, timeout * 60, &client->dev); 11558c2ecf20Sopenharmony_ci watchdog_set_nowayout(&data->wdt, nowayout); 11568c2ecf20Sopenharmony_ci watchdog_set_drvdata(&data->wdt, data); 11578c2ecf20Sopenharmony_ci 11588c2ecf20Sopenharmony_ci watchdog_stop_on_unregister(&data->wdt); 11598c2ecf20Sopenharmony_ci 11608c2ecf20Sopenharmony_ci return devm_watchdog_register_device(dev, &data->wdt); 11618c2ecf20Sopenharmony_ci} 11628c2ecf20Sopenharmony_ci 11638c2ecf20Sopenharmony_cistatic const struct i2c_device_id nct7904_id[] = { 11648c2ecf20Sopenharmony_ci {"nct7904", 0}, 11658c2ecf20Sopenharmony_ci {} 11668c2ecf20Sopenharmony_ci}; 11678c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, nct7904_id); 11688c2ecf20Sopenharmony_ci 11698c2ecf20Sopenharmony_cistatic struct i2c_driver nct7904_driver = { 11708c2ecf20Sopenharmony_ci .class = I2C_CLASS_HWMON, 11718c2ecf20Sopenharmony_ci .driver = { 11728c2ecf20Sopenharmony_ci .name = "nct7904", 11738c2ecf20Sopenharmony_ci }, 11748c2ecf20Sopenharmony_ci .probe_new = nct7904_probe, 11758c2ecf20Sopenharmony_ci .id_table = nct7904_id, 11768c2ecf20Sopenharmony_ci .detect = nct7904_detect, 11778c2ecf20Sopenharmony_ci .address_list = normal_i2c, 11788c2ecf20Sopenharmony_ci}; 11798c2ecf20Sopenharmony_ci 11808c2ecf20Sopenharmony_cimodule_i2c_driver(nct7904_driver); 11818c2ecf20Sopenharmony_ci 11828c2ecf20Sopenharmony_ciMODULE_AUTHOR("Vadim V. Vlasov <vvlasov@dev.rtsoft.ru>"); 11838c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Hwmon driver for NUVOTON NCT7904"); 11848c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL"); 1185