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