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