xref: /openbmc/linux/drivers/hwmon/nct6775.h (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
1c3963bc0SZev Weiss /* SPDX-License-Identifier: GPL-2.0-or-later */
2c3963bc0SZev Weiss #ifndef __HWMON_NCT6775_H__
3c3963bc0SZev Weiss #define __HWMON_NCT6775_H__
4c3963bc0SZev Weiss 
5c3963bc0SZev Weiss #include <linux/types.h>
6c3963bc0SZev Weiss 
7c3963bc0SZev Weiss enum kinds { nct6106, nct6116, nct6775, nct6776, nct6779, nct6791, nct6792,
8aee395bbSGuenter Roeck 	     nct6793, nct6795, nct6796, nct6797, nct6798, nct6799 };
9c3963bc0SZev Weiss enum pwm_enable { off, manual, thermal_cruise, speed_cruise, sf3, sf4 };
10c3963bc0SZev Weiss 
11*b7f1f7b2SAhmad Khalifa #define NUM_TEMP	12	/* Max number of temp attribute sets w/ limits*/
12c3963bc0SZev Weiss #define NUM_TEMP_FIXED	6	/* Max number of fixed temp attribute sets */
13c3963bc0SZev Weiss #define NUM_TSI_TEMP	8	/* Max number of TSI temp register pairs */
14c3963bc0SZev Weiss 
15c3963bc0SZev Weiss #define NUM_REG_ALARM	7	/* Max number of alarm registers */
16c3963bc0SZev Weiss #define NUM_REG_BEEP	5	/* Max number of beep registers */
17c3963bc0SZev Weiss 
18c3963bc0SZev Weiss #define NUM_FAN		7
194f65c15cSAhmad Khalifa #define NUM_IN		18
20c3963bc0SZev Weiss 
21c3963bc0SZev Weiss struct nct6775_data {
22c3963bc0SZev Weiss 	int addr;	/* IO base of hw monitor block */
23c3963bc0SZev Weiss 	int sioreg;	/* SIO register address */
24c3963bc0SZev Weiss 	enum kinds kind;
25c3963bc0SZev Weiss 	const char *name;
26c3963bc0SZev Weiss 
27c3963bc0SZev Weiss 	const struct attribute_group *groups[7];
28c3963bc0SZev Weiss 	u8 num_groups;
29c3963bc0SZev Weiss 
30c3963bc0SZev Weiss 	u16 reg_temp[5][NUM_TEMP]; /* 0=temp, 1=temp_over, 2=temp_hyst,
31c3963bc0SZev Weiss 				    * 3=temp_crit, 4=temp_lcrit
32c3963bc0SZev Weiss 				    */
33c3963bc0SZev Weiss 	u8 temp_src[NUM_TEMP];
34c3963bc0SZev Weiss 	u16 reg_temp_config[NUM_TEMP];
35c3963bc0SZev Weiss 	const char * const *temp_label;
36c3963bc0SZev Weiss 	u32 temp_mask;
37c3963bc0SZev Weiss 	u32 virt_temp_mask;
38c3963bc0SZev Weiss 
39c3963bc0SZev Weiss 	u16 REG_CONFIG;
40c3963bc0SZev Weiss 	u16 REG_VBAT;
41c3963bc0SZev Weiss 	u16 REG_DIODE;
42c3963bc0SZev Weiss 	u8 DIODE_MASK;
43c3963bc0SZev Weiss 
44c3963bc0SZev Weiss 	const s8 *ALARM_BITS;
45c3963bc0SZev Weiss 	const s8 *BEEP_BITS;
46c3963bc0SZev Weiss 
47c3963bc0SZev Weiss 	const u16 *REG_VIN;
48c3963bc0SZev Weiss 	const u16 *REG_IN_MINMAX[2];
49c3963bc0SZev Weiss 
50c3963bc0SZev Weiss 	const u16 *REG_TARGET;
51c3963bc0SZev Weiss 	const u16 *REG_FAN;
52c3963bc0SZev Weiss 	const u16 *REG_FAN_MODE;
53c3963bc0SZev Weiss 	const u16 *REG_FAN_MIN;
54c3963bc0SZev Weiss 	const u16 *REG_FAN_PULSES;
55c3963bc0SZev Weiss 	const u16 *FAN_PULSE_SHIFT;
56c3963bc0SZev Weiss 	const u16 *REG_FAN_TIME[3];
57c3963bc0SZev Weiss 
58c3963bc0SZev Weiss 	const u16 *REG_TOLERANCE_H;
59c3963bc0SZev Weiss 
60c3963bc0SZev Weiss 	const u8 *REG_PWM_MODE;
61c3963bc0SZev Weiss 	const u8 *PWM_MODE_MASK;
62c3963bc0SZev Weiss 
63c3963bc0SZev Weiss 	const u16 *REG_PWM[7];	/* [0]=pwm, [1]=pwm_start, [2]=pwm_floor,
64c3963bc0SZev Weiss 				 * [3]=pwm_max, [4]=pwm_step,
65c3963bc0SZev Weiss 				 * [5]=weight_duty_step, [6]=weight_duty_base
66c3963bc0SZev Weiss 				 */
67c3963bc0SZev Weiss 	const u16 *REG_PWM_READ;
68c3963bc0SZev Weiss 
69c3963bc0SZev Weiss 	const u16 *REG_CRITICAL_PWM_ENABLE;
70c3963bc0SZev Weiss 	u8 CRITICAL_PWM_ENABLE_MASK;
71c3963bc0SZev Weiss 	const u16 *REG_CRITICAL_PWM;
72c3963bc0SZev Weiss 
73c3963bc0SZev Weiss 	const u16 *REG_AUTO_TEMP;
74c3963bc0SZev Weiss 	const u16 *REG_AUTO_PWM;
75c3963bc0SZev Weiss 
76c3963bc0SZev Weiss 	const u16 *REG_CRITICAL_TEMP;
77c3963bc0SZev Weiss 	const u16 *REG_CRITICAL_TEMP_TOLERANCE;
78c3963bc0SZev Weiss 
79c3963bc0SZev Weiss 	const u16 *REG_TEMP_SOURCE;	/* temp register sources */
80c3963bc0SZev Weiss 	const u16 *REG_TEMP_SEL;
81c3963bc0SZev Weiss 	const u16 *REG_WEIGHT_TEMP_SEL;
82c3963bc0SZev Weiss 	const u16 *REG_WEIGHT_TEMP[3];	/* 0=base, 1=tolerance, 2=step */
83c3963bc0SZev Weiss 
84c3963bc0SZev Weiss 	const u16 *REG_TEMP_OFFSET;
85c3963bc0SZev Weiss 
86c3963bc0SZev Weiss 	const u16 *REG_ALARM;
87c3963bc0SZev Weiss 	const u16 *REG_BEEP;
88c3963bc0SZev Weiss 
89c3963bc0SZev Weiss 	const u16 *REG_TSI_TEMP;
90c3963bc0SZev Weiss 
91c3963bc0SZev Weiss 	unsigned int (*fan_from_reg)(u16 reg, unsigned int divreg);
92c3963bc0SZev Weiss 	unsigned int (*fan_from_reg_min)(u16 reg, unsigned int divreg);
93c3963bc0SZev Weiss 
94c3963bc0SZev Weiss 	struct mutex update_lock;
95c3963bc0SZev Weiss 	bool valid;		/* true if following fields are valid */
96c3963bc0SZev Weiss 	unsigned long last_updated;	/* In jiffies */
97c3963bc0SZev Weiss 
98c3963bc0SZev Weiss 	/* Register values */
99c3963bc0SZev Weiss 	u8 bank;		/* current register bank */
100c3963bc0SZev Weiss 	u8 in_num;		/* number of in inputs we have */
1014f65c15cSAhmad Khalifa 	u8 in[NUM_IN][3];	/* [0]=in, [1]=in_max, [2]=in_min */
10213558a2eSAhmad Khalifa 	const u16 *scale_in;	/* internal scaling factors */
103c3963bc0SZev Weiss 	unsigned int rpm[NUM_FAN];
104c3963bc0SZev Weiss 	u16 fan_min[NUM_FAN];
105c3963bc0SZev Weiss 	u8 fan_pulses[NUM_FAN];
106c3963bc0SZev Weiss 	u8 fan_div[NUM_FAN];
107c3963bc0SZev Weiss 	u8 has_pwm;
108c3963bc0SZev Weiss 	u8 has_fan;		/* some fan inputs can be disabled */
109c3963bc0SZev Weiss 	u8 has_fan_min;		/* some fans don't have min register */
110c3963bc0SZev Weiss 	bool has_fan_div;
111c3963bc0SZev Weiss 
112c3963bc0SZev Weiss 	u8 num_temp_alarms;	/* 2, 3, or 6 */
113c3963bc0SZev Weiss 	u8 num_temp_beeps;	/* 2, 3, or 6 */
114c3963bc0SZev Weiss 	u8 temp_fixed_num;	/* 3 or 6 */
115c3963bc0SZev Weiss 	u8 temp_type[NUM_TEMP_FIXED];
116c3963bc0SZev Weiss 	s8 temp_offset[NUM_TEMP_FIXED];
117c3963bc0SZev Weiss 	s16 temp[5][NUM_TEMP]; /* 0=temp, 1=temp_over, 2=temp_hyst,
118c3963bc0SZev Weiss 				* 3=temp_crit, 4=temp_lcrit
119c3963bc0SZev Weiss 				*/
120c3963bc0SZev Weiss 	s16 tsi_temp[NUM_TSI_TEMP];
121c3963bc0SZev Weiss 	u64 alarms;
122c3963bc0SZev Weiss 	u64 beeps;
123c3963bc0SZev Weiss 
124c3963bc0SZev Weiss 	u8 pwm_num;	/* number of pwm */
125c3963bc0SZev Weiss 	u8 pwm_mode[NUM_FAN];	/* 0->DC variable voltage,
126c3963bc0SZev Weiss 				 * 1->PWM variable duty cycle
127c3963bc0SZev Weiss 				 */
128c3963bc0SZev Weiss 	enum pwm_enable pwm_enable[NUM_FAN];
129c3963bc0SZev Weiss 			/* 0->off
130c3963bc0SZev Weiss 			 * 1->manual
131c3963bc0SZev Weiss 			 * 2->thermal cruise mode (also called SmartFan I)
132c3963bc0SZev Weiss 			 * 3->fan speed cruise mode
133c3963bc0SZev Weiss 			 * 4->SmartFan III
134c3963bc0SZev Weiss 			 * 5->enhanced variable thermal cruise (SmartFan IV)
135c3963bc0SZev Weiss 			 */
136c3963bc0SZev Weiss 	u8 pwm[7][NUM_FAN];	/* [0]=pwm, [1]=pwm_start, [2]=pwm_floor,
137c3963bc0SZev Weiss 				 * [3]=pwm_max, [4]=pwm_step,
138c3963bc0SZev Weiss 				 * [5]=weight_duty_step, [6]=weight_duty_base
139c3963bc0SZev Weiss 				 */
140c3963bc0SZev Weiss 
141c3963bc0SZev Weiss 	u8 target_temp[NUM_FAN];
142c3963bc0SZev Weiss 	u8 target_temp_mask;
143c3963bc0SZev Weiss 	u32 target_speed[NUM_FAN];
144c3963bc0SZev Weiss 	u32 target_speed_tolerance[NUM_FAN];
145c3963bc0SZev Weiss 	u8 speed_tolerance_limit;
146c3963bc0SZev Weiss 
147c3963bc0SZev Weiss 	u8 temp_tolerance[2][NUM_FAN];
148c3963bc0SZev Weiss 	u8 tolerance_mask;
149c3963bc0SZev Weiss 
150c3963bc0SZev Weiss 	u8 fan_time[3][NUM_FAN]; /* 0 = stop_time, 1 = step_up, 2 = step_down */
151c3963bc0SZev Weiss 
152c3963bc0SZev Weiss 	/* Automatic fan speed control registers */
153c3963bc0SZev Weiss 	int auto_pwm_num;
154c3963bc0SZev Weiss 	u8 auto_pwm[NUM_FAN][7];
155c3963bc0SZev Weiss 	u8 auto_temp[NUM_FAN][7];
156c3963bc0SZev Weiss 	u8 pwm_temp_sel[NUM_FAN];
157c3963bc0SZev Weiss 	u8 pwm_weight_temp_sel[NUM_FAN];
158c3963bc0SZev Weiss 	u8 weight_temp[3][NUM_FAN];	/* 0->temp_step, 1->temp_step_tol,
159c3963bc0SZev Weiss 					 * 2->temp_base
160c3963bc0SZev Weiss 					 */
161c3963bc0SZev Weiss 
162c3963bc0SZev Weiss 	u8 vid;
163c3963bc0SZev Weiss 	u8 vrm;
164c3963bc0SZev Weiss 
165c3963bc0SZev Weiss 	bool have_vid;
166c3963bc0SZev Weiss 
167c3963bc0SZev Weiss 	u16 have_temp;
168c3963bc0SZev Weiss 	u16 have_temp_fixed;
169c3963bc0SZev Weiss 	u16 have_tsi_temp;
1704f65c15cSAhmad Khalifa 	u32 have_in;
171c3963bc0SZev Weiss 
172c3963bc0SZev Weiss 	/* Remember extra register values over suspend/resume */
173c3963bc0SZev Weiss 	u8 vbat;
174c3963bc0SZev Weiss 	u8 fandiv1;
175c3963bc0SZev Weiss 	u8 fandiv2;
176c3963bc0SZev Weiss 	u8 sio_reg_enable;
177c3963bc0SZev Weiss 
178c3963bc0SZev Weiss 	struct regmap *regmap;
179c3963bc0SZev Weiss 	bool read_only;
180c3963bc0SZev Weiss 
181c3963bc0SZev Weiss 	/* driver-specific (platform, i2c) initialization hook and data */
182c3963bc0SZev Weiss 	int (*driver_init)(struct nct6775_data *data);
183c3963bc0SZev Weiss 	void *driver_data;
184c3963bc0SZev Weiss };
185c3963bc0SZev Weiss 
nct6775_read_value(struct nct6775_data * data,u16 reg,u16 * value)186c3963bc0SZev Weiss static inline int nct6775_read_value(struct nct6775_data *data, u16 reg, u16 *value)
187c3963bc0SZev Weiss {
188c3963bc0SZev Weiss 	unsigned int tmp;
189c3963bc0SZev Weiss 	int ret = regmap_read(data->regmap, reg, &tmp);
190c3963bc0SZev Weiss 
191c3963bc0SZev Weiss 	if (!ret)
192c3963bc0SZev Weiss 		*value = tmp;
193c3963bc0SZev Weiss 	return ret;
194c3963bc0SZev Weiss }
195c3963bc0SZev Weiss 
nct6775_write_value(struct nct6775_data * data,u16 reg,u16 value)196c3963bc0SZev Weiss static inline int nct6775_write_value(struct nct6775_data *data, u16 reg, u16 value)
197c3963bc0SZev Weiss {
198c3963bc0SZev Weiss 	return regmap_write(data->regmap, reg, value);
199c3963bc0SZev Weiss }
200c3963bc0SZev Weiss 
201f4e6960fSZev Weiss struct nct6775_data *nct6775_update_device(struct device *dev);
202f4e6960fSZev Weiss 
203c3963bc0SZev Weiss bool nct6775_reg_is_word_sized(struct nct6775_data *data, u16 reg);
204c3963bc0SZev Weiss int nct6775_probe(struct device *dev, struct nct6775_data *data,
205c3963bc0SZev Weiss 		  const struct regmap_config *regmapcfg);
206c3963bc0SZev Weiss 
207c3963bc0SZev Weiss ssize_t nct6775_show_alarm(struct device *dev, struct device_attribute *attr, char *buf);
208c3963bc0SZev Weiss ssize_t nct6775_show_beep(struct device *dev, struct device_attribute *attr, char *buf);
209c3963bc0SZev Weiss ssize_t nct6775_store_beep(struct device *dev, struct device_attribute *attr, const char *buf,
210c3963bc0SZev Weiss 			   size_t count);
211c3963bc0SZev Weiss 
nct6775_write_temp(struct nct6775_data * data,u16 reg,u16 value)212c3963bc0SZev Weiss static inline int nct6775_write_temp(struct nct6775_data *data, u16 reg, u16 value)
213c3963bc0SZev Weiss {
214c3963bc0SZev Weiss 	if (!nct6775_reg_is_word_sized(data, reg))
215c3963bc0SZev Weiss 		value >>= 8;
216c3963bc0SZev Weiss 	return nct6775_write_value(data, reg, value);
217c3963bc0SZev Weiss }
218c3963bc0SZev Weiss 
nct6775_attr_mode(struct nct6775_data * data,struct attribute * attr)219c3963bc0SZev Weiss static inline umode_t nct6775_attr_mode(struct nct6775_data *data, struct attribute *attr)
220c3963bc0SZev Weiss {
221c3963bc0SZev Weiss 	return data->read_only ? (attr->mode & ~0222) : attr->mode;
222c3963bc0SZev Weiss }
223c3963bc0SZev Weiss 
224c3963bc0SZev Weiss static inline int
nct6775_add_attr_group(struct nct6775_data * data,const struct attribute_group * group)225c3963bc0SZev Weiss nct6775_add_attr_group(struct nct6775_data *data, const struct attribute_group *group)
226c3963bc0SZev Weiss {
227c3963bc0SZev Weiss 	/* Need to leave a NULL terminator at the end of data->groups */
228c3963bc0SZev Weiss 	if (data->num_groups == ARRAY_SIZE(data->groups) - 1)
229c3963bc0SZev Weiss 		return -ENOBUFS;
230c3963bc0SZev Weiss 
231c3963bc0SZev Weiss 	data->groups[data->num_groups++] = group;
232c3963bc0SZev Weiss 	return 0;
233c3963bc0SZev Weiss }
234c3963bc0SZev Weiss 
235c3963bc0SZev Weiss #define NCT6775_REG_BANK	0x4E
236c3963bc0SZev Weiss #define NCT6775_REG_CONFIG	0x40
237c3963bc0SZev Weiss 
238c3963bc0SZev Weiss #define NCT6775_REG_FANDIV1		0x506
239c3963bc0SZev Weiss #define NCT6775_REG_FANDIV2		0x507
240c3963bc0SZev Weiss 
241c3963bc0SZev Weiss #define NCT6791_REG_HM_IO_SPACE_LOCK_ENABLE	0x28
242c3963bc0SZev Weiss 
2433b7f4bdeSAhmad Khalifa /*
2443b7f4bdeSAhmad Khalifa  * ALARM_BITS and BEEP_BITS store bit-index for the mask of the registers
2453b7f4bdeSAhmad Khalifa  * loaded into data->alarm and data->beep.
2463b7f4bdeSAhmad Khalifa  *
2473b7f4bdeSAhmad Khalifa  * Every input register (IN/TEMP/FAN) must have a corresponding
2483b7f4bdeSAhmad Khalifa  *   ALARM/BEEP bit at the same index BITS[BASE + index]
2493b7f4bdeSAhmad Khalifa  * Set value to -1 to disable the visibility of that '*_alarm' attribute and
2503b7f4bdeSAhmad Khalifa  * to pad the bits until the next BASE
2513b7f4bdeSAhmad Khalifa  *
2523b7f4bdeSAhmad Khalifa  * Beep has an additional GLOBAL_BEEP_ENABLE bit
2533b7f4bdeSAhmad Khalifa  */
2543b7f4bdeSAhmad Khalifa #define VIN_ALARM_BASE		 0
2553b7f4bdeSAhmad Khalifa #define FAN_ALARM_BASE		24
2563b7f4bdeSAhmad Khalifa #define TEMP_ALARM_BASE		36
2573b7f4bdeSAhmad Khalifa #define INTRUSION_ALARM_BASE	48
2583b7f4bdeSAhmad Khalifa #define BEEP_ENABLE_BASE	50
2593b7f4bdeSAhmad Khalifa 
2603b7f4bdeSAhmad Khalifa #define NUM_ALARM_BITS		(INTRUSION_ALARM_BASE + 4)
2613b7f4bdeSAhmad Khalifa #define NUM_BEEP_BITS		(BEEP_ENABLE_BASE + 1)
262c3963bc0SZev Weiss 
263c3963bc0SZev Weiss /*
264c3963bc0SZev Weiss  * Not currently used:
265c3963bc0SZev Weiss  * REG_MAN_ID has the value 0x5ca3 for all supported chips.
266c3963bc0SZev Weiss  * REG_CHIP_ID == 0x88/0xa1/0xc1 depending on chip model.
267c3963bc0SZev Weiss  * REG_MAN_ID is at port 0x4f
268c3963bc0SZev Weiss  * REG_CHIP_ID is at port 0x58
269c3963bc0SZev Weiss  */
270c3963bc0SZev Weiss 
271c3963bc0SZev Weiss #endif /* __HWMON_NCT6775_H__ */
272