Blame view

drivers/hwmon/f71882fg.c 82.8 KB
45fb36694   Hans de Goede   hwmon: add suppor...
1
2
  /***************************************************************************
   *   Copyright (C) 2006 by Hans Edgington <hans@edgington.nl>              *
44c4dc521   Hans de Goede   hwmon/f71882fg: F...
3
   *   Copyright (C) 2007-2011 Hans de Goede <hdegoede@redhat.com>           *
45fb36694   Hans de Goede   hwmon: add suppor...
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
   *                                                                         *
   *   This program is free software; you can redistribute it and/or modify  *
   *   it under the terms of the GNU General Public License as published by  *
   *   the Free Software Foundation; either version 2 of the License, or     *
   *   (at your option) any later version.                                   *
   *                                                                         *
   *   This program is distributed in the hope that it will be useful,       *
   *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
   *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *
   *   GNU General Public License for more details.                          *
   *                                                                         *
   *   You should have received a copy of the GNU General Public License     *
   *   along with this program; if not, write to the                         *
   *   Free Software Foundation, Inc.,                                       *
   *   59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.             *
   ***************************************************************************/
22d3b4128   Joe Perches   hwmon: (f71882fg)...
20
  #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
45fb36694   Hans de Goede   hwmon: add suppor...
21
22
23
24
25
26
27
28
29
  #include <linux/module.h>
  #include <linux/init.h>
  #include <linux/slab.h>
  #include <linux/jiffies.h>
  #include <linux/platform_device.h>
  #include <linux/hwmon.h>
  #include <linux/hwmon-sysfs.h>
  #include <linux/err.h>
  #include <linux/mutex.h>
77a4a3e2a   Mark van Doesburg   hwmon: (f71882fg)...
30
  #include <linux/io.h>
b9acb64a3   Jean Delvare   hwmon: Check for ...
31
  #include <linux/acpi.h>
45fb36694   Hans de Goede   hwmon: add suppor...
32
33
  
  #define DRVNAME "f71882fg"
09475d32e   Hans de Goede   hwmon: (f71882fg)...
34
  #define SIO_F71858FG_LD_HWM	0x02	/* Hardware monitor logical device */
77a4a3e2a   Mark van Doesburg   hwmon: (f71882fg)...
35
  #define SIO_F71882FG_LD_HWM	0x04	/* Hardware monitor logical device */
45fb36694   Hans de Goede   hwmon: add suppor...
36
  #define SIO_UNLOCK_KEY		0x87	/* Key to enable Super-I/O */
14a4019de   Hans de Goede   hwmon/f71882fg: F...
37
  #define SIO_LOCK_KEY		0xAA	/* Key to disable Super-I/O */
45fb36694   Hans de Goede   hwmon: add suppor...
38
39
40
41
42
43
44
45
46
  
  #define SIO_REG_LDSEL		0x07	/* Logical device select */
  #define SIO_REG_DEVID		0x20	/* Device ID (2 bytes) */
  #define SIO_REG_DEVREV		0x22	/* Device revision */
  #define SIO_REG_MANID		0x23	/* Fintek ID (2 bytes) */
  #define SIO_REG_ENABLE		0x30	/* Logical device enable */
  #define SIO_REG_ADDR		0x60	/* Logical device address (2 bytes) */
  
  #define SIO_FINTEK_ID		0x1934	/* Manufacturers ID */
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
47
  #define SIO_F71808E_ID		0x0901	/* Chipset ID */
629c58bac   Hans de Goede   hwmon: (f71882fg)...
48
  #define SIO_F71808A_ID		0x1001	/* Chipset ID */
09475d32e   Hans de Goede   hwmon: (f71882fg)...
49
  #define SIO_F71858_ID		0x0507  /* Chipset ID */
498be9683   Hans de Goede   hwmon: (f71882fg)...
50
  #define SIO_F71862_ID		0x0601	/* Chipset ID */
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
51
  #define SIO_F71869_ID		0x0814	/* Chipset ID */
5da556e33   Hans de Goede   hwmon: (f71882fg)...
52
  #define SIO_F71869A_ID		0x1007	/* Chipset ID */
45fb36694   Hans de Goede   hwmon: add suppor...
53
  #define SIO_F71882_ID		0x0541	/* Chipset ID */
7669896f4   Hans de Goede   hwmon: (f71882fg)...
54
  #define SIO_F71889_ID		0x0723	/* Chipset ID */
3cad40228   Hans de Goede   hwmon/f71882fg: A...
55
  #define SIO_F71889E_ID		0x0909	/* Chipset ID */
a66c10887   Hans de Goede   hwmon: (f71882fg)...
56
  #define SIO_F71889A_ID		0x1005	/* Chipset ID */
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
57
  #define SIO_F8000_ID		0x0581	/* Chipset ID */
383586b12   Jean Delvare   hwmon: (f71882fg)...
58
  #define SIO_F81865_ID		0x0704	/* Chipset ID */
45fb36694   Hans de Goede   hwmon: add suppor...
59
60
61
62
  
  #define REGION_LENGTH		8
  #define ADDR_REG_OFFSET		5
  #define DATA_REG_OFFSET		6
3cad40228   Hans de Goede   hwmon/f71882fg: A...
63
64
  #define F71882FG_REG_IN_STATUS		0x12 /* f7188x only */
  #define F71882FG_REG_IN_BEEP		0x13 /* f7188x only */
45fb36694   Hans de Goede   hwmon: add suppor...
65
  #define F71882FG_REG_IN(nr)		(0x20  + (nr))
3cad40228   Hans de Goede   hwmon/f71882fg: A...
66
  #define F71882FG_REG_IN1_HIGH		0x32 /* f7188x only */
45fb36694   Hans de Goede   hwmon: add suppor...
67
68
  
  #define F71882FG_REG_FAN(nr)		(0xA0 + (16 * (nr)))
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
69
70
  #define F71882FG_REG_FAN_TARGET(nr)	(0xA2 + (16 * (nr)))
  #define F71882FG_REG_FAN_FULL_SPEED(nr)	(0xA4 + (16 * (nr)))
45fb36694   Hans de Goede   hwmon: add suppor...
71
72
  #define F71882FG_REG_FAN_STATUS		0x92
  #define F71882FG_REG_FAN_BEEP		0x93
7567a0435   Hans de Goede   hwmon: (f71882fg)...
73
74
75
  #define F71882FG_REG_TEMP(nr)		(0x70 + 2 * (nr))
  #define F71882FG_REG_TEMP_OVT(nr)	(0x80 + 2 * (nr))
  #define F71882FG_REG_TEMP_HIGH(nr)	(0x81 + 2 * (nr))
45fb36694   Hans de Goede   hwmon: add suppor...
76
77
  #define F71882FG_REG_TEMP_STATUS	0x62
  #define F71882FG_REG_TEMP_BEEP		0x63
09475d32e   Hans de Goede   hwmon: (f71882fg)...
78
  #define F71882FG_REG_TEMP_CONFIG	0x69
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
79
  #define F71882FG_REG_TEMP_HYST(nr)	(0x6C + (nr))
45fb36694   Hans de Goede   hwmon: add suppor...
80
81
  #define F71882FG_REG_TEMP_TYPE		0x6B
  #define F71882FG_REG_TEMP_DIODE_OPEN	0x6F
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
82
83
84
  #define F71882FG_REG_PWM(nr)		(0xA3 + (16 * (nr)))
  #define F71882FG_REG_PWM_TYPE		0x94
  #define F71882FG_REG_PWM_ENABLE		0x96
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
85
  #define F71882FG_REG_FAN_HYST(nr)	(0x98 + (nr))
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
86

98f7ba197   Hans de Goede   hwmon/f71882fg: O...
87
88
  #define F71882FG_REG_FAN_FAULT_T	0x9F
  #define F71882FG_FAN_NEG_TEMP_EN	0x20
3cad40228   Hans de Goede   hwmon/f71882fg: A...
89
  #define F71882FG_FAN_PROG_SEL		0x80
98f7ba197   Hans de Goede   hwmon/f71882fg: O...
90

9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
91
92
93
  #define F71882FG_REG_POINT_PWM(pwm, point)	(0xAA + (point) + (16 * (pwm)))
  #define F71882FG_REG_POINT_TEMP(pwm, point)	(0xA6 + (point) + (16 * (pwm)))
  #define F71882FG_REG_POINT_MAPPING(nr)		(0xAF + 16 * (nr))
45fb36694   Hans de Goede   hwmon: add suppor...
94
  #define	F71882FG_REG_START		0x01
0bae64008   Hans de Goede   hwmon/f71882fg: M...
95
  #define F71882FG_MAX_INS		9
45fb36694   Hans de Goede   hwmon: add suppor...
96
  #define FAN_MIN_DETECT			366 /* Lowest detectable fanspeed */
67b671bce   Jean Delvare   hwmon: Let the us...
97
98
99
  static unsigned short force_id;
  module_param(force_id, ushort, 0);
  MODULE_PARM_DESC(force_id, "Override the detected device ID");
5da556e33   Hans de Goede   hwmon: (f71882fg)...
100
101
  enum chips { f71808e, f71808a, f71858fg, f71862fg, f71869, f71869a, f71882fg,
  	     f71889fg, f71889ed, f71889a, f8000, f81865f };
498be9683   Hans de Goede   hwmon: (f71882fg)...
102
103
  
  static const char *f71882fg_names[] = {
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
104
  	"f71808e",
629c58bac   Hans de Goede   hwmon: (f71882fg)...
105
  	"f71808a",
09475d32e   Hans de Goede   hwmon: (f71882fg)...
106
  	"f71858fg",
498be9683   Hans de Goede   hwmon: (f71882fg)...
107
  	"f71862fg",
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
108
  	"f71869", /* Both f71869f and f71869e, reg. compatible and same id */
5da556e33   Hans de Goede   hwmon: (f71882fg)...
109
  	"f71869a",
498be9683   Hans de Goede   hwmon: (f71882fg)...
110
  	"f71882fg",
5d7f77bf6   Jean Delvare   hwmon: (f71882fg)...
111
  	"f71889fg", /* f81801u too, same id */
3cad40228   Hans de Goede   hwmon/f71882fg: A...
112
  	"f71889ed",
a66c10887   Hans de Goede   hwmon: (f71882fg)...
113
  	"f71889a",
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
114
  	"f8000",
383586b12   Jean Delvare   hwmon: (f71882fg)...
115
  	"f81865f",
498be9683   Hans de Goede   hwmon: (f71882fg)...
116
  };
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
117
118
  static const char f71882fg_has_in[][F71882FG_MAX_INS] = {
  	[f71808e]	= { 1, 1, 1, 1, 1, 1, 0, 1, 1 },
629c58bac   Hans de Goede   hwmon: (f71882fg)...
119
  	[f71808a]	= { 1, 1, 1, 1, 0, 0, 0, 1, 1 },
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
120
121
122
  	[f71858fg]	= { 1, 1, 1, 0, 0, 0, 0, 0, 0 },
  	[f71862fg]	= { 1, 1, 1, 1, 1, 1, 1, 1, 1 },
  	[f71869]	= { 1, 1, 1, 1, 1, 1, 1, 1, 1 },
5da556e33   Hans de Goede   hwmon: (f71882fg)...
123
  	[f71869a]	= { 1, 1, 1, 1, 1, 1, 1, 1, 1 },
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
124
125
126
  	[f71882fg]	= { 1, 1, 1, 1, 1, 1, 1, 1, 1 },
  	[f71889fg]	= { 1, 1, 1, 1, 1, 1, 1, 1, 1 },
  	[f71889ed]	= { 1, 1, 1, 1, 1, 1, 1, 1, 1 },
a66c10887   Hans de Goede   hwmon: (f71882fg)...
127
  	[f71889a]	= { 1, 1, 1, 1, 1, 1, 1, 1, 1 },
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
128
  	[f8000]		= { 1, 1, 1, 0, 0, 0, 0, 0, 0 },
383586b12   Jean Delvare   hwmon: (f71882fg)...
129
  	[f81865f]	= { 1, 1, 1, 1, 1, 1, 1, 0, 0 },
0bae64008   Hans de Goede   hwmon/f71882fg: M...
130
  };
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
131
132
  static const char f71882fg_has_in1_alarm[] = {
  	[f71808e]	= 0,
629c58bac   Hans de Goede   hwmon: (f71882fg)...
133
  	[f71808a]	= 0,
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
134
135
136
  	[f71858fg]	= 0,
  	[f71862fg]	= 0,
  	[f71869]	= 0,
5da556e33   Hans de Goede   hwmon: (f71882fg)...
137
  	[f71869a]	= 0,
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
138
139
140
  	[f71882fg]	= 1,
  	[f71889fg]	= 1,
  	[f71889ed]	= 1,
a66c10887   Hans de Goede   hwmon: (f71882fg)...
141
  	[f71889a]	= 1,
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
142
  	[f8000]		= 0,
383586b12   Jean Delvare   hwmon: (f71882fg)...
143
  	[f81865f]	= 1,
0bae64008   Hans de Goede   hwmon/f71882fg: M...
144
  };
4d53811a2   Hans de Goede   hwmon: (f71882fg)...
145
  static const char f71882fg_fan_has_beep[] = {
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
146
  	[f71808e]	= 0,
629c58bac   Hans de Goede   hwmon: (f71882fg)...
147
  	[f71808a]	= 0,
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
148
149
150
  	[f71858fg]	= 0,
  	[f71862fg]	= 1,
  	[f71869]	= 1,
5da556e33   Hans de Goede   hwmon: (f71882fg)...
151
  	[f71869a]	= 1,
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
152
153
154
  	[f71882fg]	= 1,
  	[f71889fg]	= 1,
  	[f71889ed]	= 1,
a66c10887   Hans de Goede   hwmon: (f71882fg)...
155
  	[f71889a]	= 1,
2740c60c4   Jean Delvare   hwmon: (f71882fg)...
156
  	[f8000]		= 0,
383586b12   Jean Delvare   hwmon: (f71882fg)...
157
  	[f81865f]	= 1,
78aa4f72b   Hans de Goede   hwmon/f71882fg: S...
158
  };
f27def07b   Jean Delvare   hwmon: (f71882fg)...
159
160
  static const char f71882fg_nr_fans[] = {
  	[f71808e]	= 3,
629c58bac   Hans de Goede   hwmon: (f71882fg)...
161
  	[f71808a]	= 2, /* +1 fan which is monitor + simple pwm only */
f27def07b   Jean Delvare   hwmon: (f71882fg)...
162
163
164
  	[f71858fg]	= 3,
  	[f71862fg]	= 3,
  	[f71869]	= 3,
5da556e33   Hans de Goede   hwmon: (f71882fg)...
165
  	[f71869a]	= 3,
f27def07b   Jean Delvare   hwmon: (f71882fg)...
166
167
168
  	[f71882fg]	= 4,
  	[f71889fg]	= 3,
  	[f71889ed]	= 3,
a66c10887   Hans de Goede   hwmon: (f71882fg)...
169
  	[f71889a]	= 3,
629c58bac   Hans de Goede   hwmon: (f71882fg)...
170
  	[f8000]		= 3, /* +1 fan which is monitor only */
383586b12   Jean Delvare   hwmon: (f71882fg)...
171
  	[f81865f]	= 2,
f27def07b   Jean Delvare   hwmon: (f71882fg)...
172
  };
4d53811a2   Hans de Goede   hwmon: (f71882fg)...
173
174
  static const char f71882fg_temp_has_beep[] = {
  	[f71808e]	= 0,
629c58bac   Hans de Goede   hwmon: (f71882fg)...
175
  	[f71808a]	= 1,
4d53811a2   Hans de Goede   hwmon: (f71882fg)...
176
177
178
  	[f71858fg]	= 0,
  	[f71862fg]	= 1,
  	[f71869]	= 1,
5da556e33   Hans de Goede   hwmon: (f71882fg)...
179
  	[f71869a]	= 1,
4d53811a2   Hans de Goede   hwmon: (f71882fg)...
180
181
182
183
184
185
186
  	[f71882fg]	= 1,
  	[f71889fg]	= 1,
  	[f71889ed]	= 1,
  	[f71889a]	= 1,
  	[f8000]		= 0,
  	[f81865f]	= 1,
  };
f27def07b   Jean Delvare   hwmon: (f71882fg)...
187
188
  static const char f71882fg_nr_temps[] = {
  	[f71808e]	= 2,
629c58bac   Hans de Goede   hwmon: (f71882fg)...
189
  	[f71808a]	= 2,
f27def07b   Jean Delvare   hwmon: (f71882fg)...
190
191
192
  	[f71858fg]	= 3,
  	[f71862fg]	= 3,
  	[f71869]	= 3,
5da556e33   Hans de Goede   hwmon: (f71882fg)...
193
  	[f71869a]	= 3,
f27def07b   Jean Delvare   hwmon: (f71882fg)...
194
195
196
  	[f71882fg]	= 3,
  	[f71889fg]	= 3,
  	[f71889ed]	= 3,
a66c10887   Hans de Goede   hwmon: (f71882fg)...
197
  	[f71889a]	= 3,
f27def07b   Jean Delvare   hwmon: (f71882fg)...
198
  	[f8000]		= 3,
383586b12   Jean Delvare   hwmon: (f71882fg)...
199
  	[f81865f]	= 2,
f27def07b   Jean Delvare   hwmon: (f71882fg)...
200
  };
77a4a3e2a   Mark van Doesburg   hwmon: (f71882fg)...
201
  static struct platform_device *f71882fg_pdev;
45fb36694   Hans de Goede   hwmon: add suppor...
202
203
204
205
  
  /* Super-I/O Function prototypes */
  static inline int superio_inb(int base, int reg);
  static inline int superio_inw(int base, int reg);
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
206
  static inline int superio_enter(int base);
45fb36694   Hans de Goede   hwmon: add suppor...
207
208
  static inline void superio_select(int base, int ld);
  static inline void superio_exit(int base);
498be9683   Hans de Goede   hwmon: (f71882fg)...
209
210
211
  struct f71882fg_sio_data {
  	enum chips type;
  };
45fb36694   Hans de Goede   hwmon: add suppor...
212
213
  struct f71882fg_data {
  	unsigned short addr;
498be9683   Hans de Goede   hwmon: (f71882fg)...
214
  	enum chips type;
1beeffe43   Tony Jones   hwmon: Convert fr...
215
  	struct device *hwmon_dev;
45fb36694   Hans de Goede   hwmon: add suppor...
216
217
  
  	struct mutex update_lock;
09475d32e   Hans de Goede   hwmon: (f71882fg)...
218
  	int temp_start;			/* temp numbering start (0 or 1) */
45fb36694   Hans de Goede   hwmon: add suppor...
219
  	char valid;			/* !=0 if following fields are valid */
98f7ba197   Hans de Goede   hwmon/f71882fg: O...
220
  	char auto_point_temp_signed;
45fb36694   Hans de Goede   hwmon: add suppor...
221
222
223
224
  	unsigned long last_updated;	/* In jiffies */
  	unsigned long last_limits;	/* In jiffies */
  
  	/* Register Values */
0bae64008   Hans de Goede   hwmon/f71882fg: M...
225
  	u8	in[F71882FG_MAX_INS];
45fb36694   Hans de Goede   hwmon: add suppor...
226
227
228
229
  	u8	in1_max;
  	u8	in_status;
  	u8	in_beep;
  	u16	fan[4];
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
230
231
  	u16	fan_target[4];
  	u16	fan_full_speed[4];
45fb36694   Hans de Goede   hwmon: add suppor...
232
233
  	u8	fan_status;
  	u8	fan_beep;
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
234
  	/* Note: all models have max 3 temperature channels, but on some
7567a0435   Hans de Goede   hwmon: (f71882fg)...
235
236
  	   they are addressed as 0-2 and on others as 1-3, so for coding
  	   convenience we reserve space for 4 channels */
09475d32e   Hans de Goede   hwmon: (f71882fg)...
237
  	u16	temp[4];
7567a0435   Hans de Goede   hwmon: (f71882fg)...
238
239
  	u8	temp_ovt[4];
  	u8	temp_high[4];
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
240
  	u8	temp_hyst[2]; /* 2 hysts stored per reg */
7567a0435   Hans de Goede   hwmon: (f71882fg)...
241
  	u8	temp_type[4];
45fb36694   Hans de Goede   hwmon: add suppor...
242
243
244
  	u8	temp_status;
  	u8	temp_beep;
  	u8	temp_diode_open;
09475d32e   Hans de Goede   hwmon: (f71882fg)...
245
  	u8	temp_config;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
246
247
248
249
250
  	u8	pwm[4];
  	u8	pwm_enable;
  	u8	pwm_auto_point_hyst[2];
  	u8	pwm_auto_point_mapping[4];
  	u8	pwm_auto_point_pwm[4][5];
7669896f4   Hans de Goede   hwmon: (f71882fg)...
251
  	s8	pwm_auto_point_temp[4][4];
45fb36694   Hans de Goede   hwmon: add suppor...
252
  };
77a4a3e2a   Mark van Doesburg   hwmon: (f71882fg)...
253
  /* Sysfs in */
45fb36694   Hans de Goede   hwmon: add suppor...
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
  static ssize_t show_in(struct device *dev, struct device_attribute *devattr,
  	char *buf);
  static ssize_t show_in_max(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t store_in_max(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count);
  static ssize_t show_in_beep(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t store_in_beep(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count);
  static ssize_t show_in_alarm(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  /* Sysfs Fan */
  static ssize_t show_fan(struct device *dev, struct device_attribute *devattr,
  	char *buf);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
269
270
271
272
  static ssize_t show_fan_full_speed(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_fan_full_speed(struct device *dev,
  	struct device_attribute *devattr, const char *buf, size_t count);
45fb36694   Hans de Goede   hwmon: add suppor...
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
  static ssize_t show_fan_beep(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t store_fan_beep(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count);
  static ssize_t show_fan_alarm(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  /* Sysfs Temp */
  static ssize_t show_temp(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t show_temp_max(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t store_temp_max(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count);
  static ssize_t show_temp_max_hyst(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t store_temp_max_hyst(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count);
  static ssize_t show_temp_crit(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t store_temp_crit(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count);
  static ssize_t show_temp_crit_hyst(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t show_temp_type(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t show_temp_beep(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t store_temp_beep(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count);
  static ssize_t show_temp_alarm(struct device *dev, struct device_attribute
  	*devattr, char *buf);
  static ssize_t show_temp_fault(struct device *dev, struct device_attribute
  	*devattr, char *buf);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
306
307
308
309
310
  /* PWM and Auto point control */
  static ssize_t show_pwm(struct device *dev, struct device_attribute *devattr,
  	char *buf);
  static ssize_t store_pwm(struct device *dev, struct device_attribute *devattr,
  	const char *buf, size_t count);
629c58bac   Hans de Goede   hwmon: (f71882fg)...
311
312
313
314
  static ssize_t show_simple_pwm(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_simple_pwm(struct device *dev,
  	struct device_attribute *devattr, const char *buf, size_t count);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
  static ssize_t show_pwm_enable(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_pwm_enable(struct device *dev,
  	struct device_attribute	*devattr, const char *buf, size_t count);
  static ssize_t show_pwm_interpolate(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_pwm_interpolate(struct device *dev,
  	struct device_attribute *devattr, const char *buf, size_t count);
  static ssize_t show_pwm_auto_point_channel(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_pwm_auto_point_channel(struct device *dev,
  	struct device_attribute *devattr, const char *buf, size_t count);
  static ssize_t show_pwm_auto_point_temp_hyst(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_pwm_auto_point_temp_hyst(struct device *dev,
  	struct device_attribute *devattr, const char *buf, size_t count);
  static ssize_t show_pwm_auto_point_pwm(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_pwm_auto_point_pwm(struct device *dev,
  	struct device_attribute *devattr, const char *buf, size_t count);
  static ssize_t show_pwm_auto_point_temp(struct device *dev,
  	struct device_attribute *devattr, char *buf);
  static ssize_t store_pwm_auto_point_temp(struct device *dev,
  	struct device_attribute *devattr, const char *buf, size_t count);
45fb36694   Hans de Goede   hwmon: add suppor...
339
340
341
342
343
  /* Sysfs misc */
  static ssize_t show_name(struct device *dev, struct device_attribute *devattr,
  	char *buf);
  
  static int __devinit f71882fg_probe(struct platform_device * pdev);
c13548c53   Hans de Goede   hwmon: (f71882fg)...
344
  static int f71882fg_remove(struct platform_device *pdev);
45fb36694   Hans de Goede   hwmon: add suppor...
345
346
347
348
349
350
351
  
  static struct platform_driver f71882fg_driver = {
  	.driver = {
  		.owner	= THIS_MODULE,
  		.name	= DRVNAME,
  	},
  	.probe		= f71882fg_probe,
cd659fd0d   Jean Delvare   hwmon: Fix more _...
352
  	.remove		= f71882fg_remove,
45fb36694   Hans de Goede   hwmon: add suppor...
353
  };
c13548c53   Hans de Goede   hwmon: (f71882fg)...
354
  static DEVICE_ATTR(name, S_IRUGO, show_name, NULL);
45fb36694   Hans de Goede   hwmon: add suppor...
355

0bae64008   Hans de Goede   hwmon/f71882fg: M...
356
357
358
  /* Temp attr for the f71858fg, the f71858fg is special as it has its
     temperature indexes start at 0 (the others start at 1) */
  static struct sensor_device_attribute_2 f71858fg_temp_attr[] = {
09475d32e   Hans de Goede   hwmon: (f71882fg)...
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
  	SENSOR_ATTR_2(temp1_input, S_IRUGO, show_temp, NULL, 0, 0),
  	SENSOR_ATTR_2(temp1_max, S_IRUGO|S_IWUSR, show_temp_max,
  		store_temp_max, 0, 0),
  	SENSOR_ATTR_2(temp1_max_hyst, S_IRUGO|S_IWUSR, show_temp_max_hyst,
  		store_temp_max_hyst, 0, 0),
  	SENSOR_ATTR_2(temp1_max_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 0),
  	SENSOR_ATTR_2(temp1_crit, S_IRUGO|S_IWUSR, show_temp_crit,
  		store_temp_crit, 0, 0),
  	SENSOR_ATTR_2(temp1_crit_hyst, S_IRUGO, show_temp_crit_hyst, NULL,
  		0, 0),
  	SENSOR_ATTR_2(temp1_crit_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 4),
  	SENSOR_ATTR_2(temp1_fault, S_IRUGO, show_temp_fault, NULL, 0, 0),
  	SENSOR_ATTR_2(temp2_input, S_IRUGO, show_temp, NULL, 0, 1),
  	SENSOR_ATTR_2(temp2_max, S_IRUGO|S_IWUSR, show_temp_max,
  		store_temp_max, 0, 1),
  	SENSOR_ATTR_2(temp2_max_hyst, S_IRUGO|S_IWUSR, show_temp_max_hyst,
  		store_temp_max_hyst, 0, 1),
  	SENSOR_ATTR_2(temp2_max_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 1),
  	SENSOR_ATTR_2(temp2_crit, S_IRUGO|S_IWUSR, show_temp_crit,
  		store_temp_crit, 0, 1),
  	SENSOR_ATTR_2(temp2_crit_hyst, S_IRUGO, show_temp_crit_hyst, NULL,
  		0, 1),
  	SENSOR_ATTR_2(temp2_crit_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 5),
09475d32e   Hans de Goede   hwmon: (f71882fg)...
382
383
384
385
386
387
388
389
390
391
392
393
394
395
  	SENSOR_ATTR_2(temp2_fault, S_IRUGO, show_temp_fault, NULL, 0, 1),
  	SENSOR_ATTR_2(temp3_input, S_IRUGO, show_temp, NULL, 0, 2),
  	SENSOR_ATTR_2(temp3_max, S_IRUGO|S_IWUSR, show_temp_max,
  		store_temp_max, 0, 2),
  	SENSOR_ATTR_2(temp3_max_hyst, S_IRUGO|S_IWUSR, show_temp_max_hyst,
  		store_temp_max_hyst, 0, 2),
  	SENSOR_ATTR_2(temp3_max_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 2),
  	SENSOR_ATTR_2(temp3_crit, S_IRUGO|S_IWUSR, show_temp_crit,
  		store_temp_crit, 0, 2),
  	SENSOR_ATTR_2(temp3_crit_hyst, S_IRUGO, show_temp_crit_hyst, NULL,
  		0, 2),
  	SENSOR_ATTR_2(temp3_crit_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 6),
  	SENSOR_ATTR_2(temp3_fault, S_IRUGO, show_temp_fault, NULL, 0, 2),
  };
0bae64008   Hans de Goede   hwmon/f71882fg: M...
396
  /* Temp attr for the standard models */
78aa4f72b   Hans de Goede   hwmon/f71882fg: S...
397
  static struct sensor_device_attribute_2 fxxxx_temp_attr[3][9] = { {
7567a0435   Hans de Goede   hwmon: (f71882fg)...
398
  	SENSOR_ATTR_2(temp1_input, S_IRUGO, show_temp, NULL, 0, 1),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
399
  	SENSOR_ATTR_2(temp1_max, S_IRUGO|S_IWUSR, show_temp_max,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
400
  		store_temp_max, 0, 1),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
401
  	SENSOR_ATTR_2(temp1_max_hyst, S_IRUGO|S_IWUSR, show_temp_max_hyst,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
402
  		store_temp_max_hyst, 0, 1),
754a5907b   Hans de Goede   hwmon: (f71882fg)...
403
404
405
406
  	/* Should really be temp1_max_alarm, but older versions did not handle
  	   the max and crit alarms separately and lm_sensors v2 depends on the
  	   presence of temp#_alarm files. The same goes for temp2/3 _alarm. */
  	SENSOR_ATTR_2(temp1_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 1),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
407
  	SENSOR_ATTR_2(temp1_crit, S_IRUGO|S_IWUSR, show_temp_crit,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
408
  		store_temp_crit, 0, 1),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
409
  	SENSOR_ATTR_2(temp1_crit_hyst, S_IRUGO, show_temp_crit_hyst, NULL,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
410
  		0, 1),
754a5907b   Hans de Goede   hwmon: (f71882fg)...
411
  	SENSOR_ATTR_2(temp1_crit_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 5),
7567a0435   Hans de Goede   hwmon: (f71882fg)...
412
  	SENSOR_ATTR_2(temp1_type, S_IRUGO, show_temp_type, NULL, 0, 1),
7567a0435   Hans de Goede   hwmon: (f71882fg)...
413
  	SENSOR_ATTR_2(temp1_fault, S_IRUGO, show_temp_fault, NULL, 0, 1),
60d2b378b   Hans de Goede   hwmon/f71882fg: M...
414
  }, {
7567a0435   Hans de Goede   hwmon: (f71882fg)...
415
  	SENSOR_ATTR_2(temp2_input, S_IRUGO, show_temp, NULL, 0, 2),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
416
  	SENSOR_ATTR_2(temp2_max, S_IRUGO|S_IWUSR, show_temp_max,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
417
  		store_temp_max, 0, 2),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
418
  	SENSOR_ATTR_2(temp2_max_hyst, S_IRUGO|S_IWUSR, show_temp_max_hyst,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
419
  		store_temp_max_hyst, 0, 2),
754a5907b   Hans de Goede   hwmon: (f71882fg)...
420
421
  	/* Should be temp2_max_alarm, see temp1_alarm note */
  	SENSOR_ATTR_2(temp2_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 2),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
422
  	SENSOR_ATTR_2(temp2_crit, S_IRUGO|S_IWUSR, show_temp_crit,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
423
  		store_temp_crit, 0, 2),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
424
  	SENSOR_ATTR_2(temp2_crit_hyst, S_IRUGO, show_temp_crit_hyst, NULL,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
425
  		0, 2),
754a5907b   Hans de Goede   hwmon: (f71882fg)...
426
  	SENSOR_ATTR_2(temp2_crit_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 6),
7567a0435   Hans de Goede   hwmon: (f71882fg)...
427
  	SENSOR_ATTR_2(temp2_type, S_IRUGO, show_temp_type, NULL, 0, 2),
7567a0435   Hans de Goede   hwmon: (f71882fg)...
428
  	SENSOR_ATTR_2(temp2_fault, S_IRUGO, show_temp_fault, NULL, 0, 2),
60d2b378b   Hans de Goede   hwmon/f71882fg: M...
429
  }, {
7567a0435   Hans de Goede   hwmon: (f71882fg)...
430
  	SENSOR_ATTR_2(temp3_input, S_IRUGO, show_temp, NULL, 0, 3),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
431
  	SENSOR_ATTR_2(temp3_max, S_IRUGO|S_IWUSR, show_temp_max,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
432
  		store_temp_max, 0, 3),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
433
  	SENSOR_ATTR_2(temp3_max_hyst, S_IRUGO|S_IWUSR, show_temp_max_hyst,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
434
  		store_temp_max_hyst, 0, 3),
754a5907b   Hans de Goede   hwmon: (f71882fg)...
435
436
  	/* Should be temp3_max_alarm, see temp1_alarm note */
  	SENSOR_ATTR_2(temp3_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 3),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
437
  	SENSOR_ATTR_2(temp3_crit, S_IRUGO|S_IWUSR, show_temp_crit,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
438
  		store_temp_crit, 0, 3),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
439
  	SENSOR_ATTR_2(temp3_crit_hyst, S_IRUGO, show_temp_crit_hyst, NULL,
7567a0435   Hans de Goede   hwmon: (f71882fg)...
440
  		0, 3),
754a5907b   Hans de Goede   hwmon: (f71882fg)...
441
  	SENSOR_ATTR_2(temp3_crit_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 7),
7567a0435   Hans de Goede   hwmon: (f71882fg)...
442
  	SENSOR_ATTR_2(temp3_type, S_IRUGO, show_temp_type, NULL, 0, 3),
7567a0435   Hans de Goede   hwmon: (f71882fg)...
443
  	SENSOR_ATTR_2(temp3_fault, S_IRUGO, show_temp_fault, NULL, 0, 3),
60d2b378b   Hans de Goede   hwmon/f71882fg: M...
444
  } };
45fb36694   Hans de Goede   hwmon: add suppor...
445

78aa4f72b   Hans de Goede   hwmon/f71882fg: S...
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
  /* Temp attr for models which can beep on temp alarm */
  static struct sensor_device_attribute_2 fxxxx_temp_beep_attr[3][2] = { {
  	SENSOR_ATTR_2(temp1_max_beep, S_IRUGO|S_IWUSR, show_temp_beep,
  		store_temp_beep, 0, 1),
  	SENSOR_ATTR_2(temp1_crit_beep, S_IRUGO|S_IWUSR, show_temp_beep,
  		store_temp_beep, 0, 5),
  }, {
  	SENSOR_ATTR_2(temp2_max_beep, S_IRUGO|S_IWUSR, show_temp_beep,
  		store_temp_beep, 0, 2),
  	SENSOR_ATTR_2(temp2_crit_beep, S_IRUGO|S_IWUSR, show_temp_beep,
  		store_temp_beep, 0, 6),
  }, {
  	SENSOR_ATTR_2(temp3_max_beep, S_IRUGO|S_IWUSR, show_temp_beep,
  		store_temp_beep, 0, 3),
  	SENSOR_ATTR_2(temp3_crit_beep, S_IRUGO|S_IWUSR, show_temp_beep,
  		store_temp_beep, 0, 7),
  } };
0bae64008   Hans de Goede   hwmon/f71882fg: M...
463
  /* Temp attr for the f8000
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
464
465
     Note on the f8000 temp_ovt (crit) is used as max, and temp_high (max)
     is used as hysteresis value to clear alarms
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
466
     Also like the f71858fg its temperature indexes start at 0
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
467
   */
0bae64008   Hans de Goede   hwmon/f71882fg: M...
468
  static struct sensor_device_attribute_2 f8000_temp_attr[] = {
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
469
470
471
472
473
474
  	SENSOR_ATTR_2(temp1_input, S_IRUGO, show_temp, NULL, 0, 0),
  	SENSOR_ATTR_2(temp1_max, S_IRUGO|S_IWUSR, show_temp_crit,
  		store_temp_crit, 0, 0),
  	SENSOR_ATTR_2(temp1_max_hyst, S_IRUGO|S_IWUSR, show_temp_max,
  		store_temp_max, 0, 0),
  	SENSOR_ATTR_2(temp1_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 4),
b6858bca8   Hans de Goede   hwmon: (f71882fg)...
475
  	SENSOR_ATTR_2(temp1_fault, S_IRUGO, show_temp_fault, NULL, 0, 0),
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
476
477
478
479
480
481
  	SENSOR_ATTR_2(temp2_input, S_IRUGO, show_temp, NULL, 0, 1),
  	SENSOR_ATTR_2(temp2_max, S_IRUGO|S_IWUSR, show_temp_crit,
  		store_temp_crit, 0, 1),
  	SENSOR_ATTR_2(temp2_max_hyst, S_IRUGO|S_IWUSR, show_temp_max,
  		store_temp_max, 0, 1),
  	SENSOR_ATTR_2(temp2_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 5),
b6858bca8   Hans de Goede   hwmon: (f71882fg)...
482
  	SENSOR_ATTR_2(temp2_fault, S_IRUGO, show_temp_fault, NULL, 0, 1),
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
483
484
485
486
487
488
  	SENSOR_ATTR_2(temp3_input, S_IRUGO, show_temp, NULL, 0, 2),
  	SENSOR_ATTR_2(temp3_max, S_IRUGO|S_IWUSR, show_temp_crit,
  		store_temp_crit, 0, 2),
  	SENSOR_ATTR_2(temp3_max_hyst, S_IRUGO|S_IWUSR, show_temp_max,
  		store_temp_max, 0, 2),
  	SENSOR_ATTR_2(temp3_alarm, S_IRUGO, show_temp_alarm, NULL, 0, 6),
b6858bca8   Hans de Goede   hwmon: (f71882fg)...
489
  	SENSOR_ATTR_2(temp3_fault, S_IRUGO, show_temp_fault, NULL, 0, 2),
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
490
  };
0bae64008   Hans de Goede   hwmon/f71882fg: M...
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
  /* in attr for all models */
  static struct sensor_device_attribute_2 fxxxx_in_attr[] = {
  	SENSOR_ATTR_2(in0_input, S_IRUGO, show_in, NULL, 0, 0),
  	SENSOR_ATTR_2(in1_input, S_IRUGO, show_in, NULL, 0, 1),
  	SENSOR_ATTR_2(in2_input, S_IRUGO, show_in, NULL, 0, 2),
  	SENSOR_ATTR_2(in3_input, S_IRUGO, show_in, NULL, 0, 3),
  	SENSOR_ATTR_2(in4_input, S_IRUGO, show_in, NULL, 0, 4),
  	SENSOR_ATTR_2(in5_input, S_IRUGO, show_in, NULL, 0, 5),
  	SENSOR_ATTR_2(in6_input, S_IRUGO, show_in, NULL, 0, 6),
  	SENSOR_ATTR_2(in7_input, S_IRUGO, show_in, NULL, 0, 7),
  	SENSOR_ATTR_2(in8_input, S_IRUGO, show_in, NULL, 0, 8),
  };
  
  /* For models with in1 alarm capability */
  static struct sensor_device_attribute_2 fxxxx_in1_alarm_attr[] = {
  	SENSOR_ATTR_2(in1_max, S_IRUGO|S_IWUSR, show_in_max, store_in_max,
  		0, 1),
  	SENSOR_ATTR_2(in1_beep, S_IRUGO|S_IWUSR, show_in_beep, store_in_beep,
  		0, 1),
  	SENSOR_ATTR_2(in1_alarm, S_IRUGO, show_in_alarm, NULL, 0, 1),
  };
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
512
  /* Fan / PWM attr common to all models */
b69b03992   Hans de Goede   hwmon: (f71882fg)...
513
  static struct sensor_device_attribute_2 fxxxx_fan_attr[4][6] = { {
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
514
  	SENSOR_ATTR_2(fan1_input, S_IRUGO, show_fan, NULL, 0, 0),
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
515
516
517
  	SENSOR_ATTR_2(fan1_full_speed, S_IRUGO|S_IWUSR,
  		      show_fan_full_speed,
  		      store_fan_full_speed, 0, 0),
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
518
  	SENSOR_ATTR_2(fan1_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 0),
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
519
520
521
522
523
  	SENSOR_ATTR_2(pwm1, S_IRUGO|S_IWUSR, show_pwm, store_pwm, 0, 0),
  	SENSOR_ATTR_2(pwm1_enable, S_IRUGO|S_IWUSR, show_pwm_enable,
  		      store_pwm_enable, 0, 0),
  	SENSOR_ATTR_2(pwm1_interpolate, S_IRUGO|S_IWUSR,
  		      show_pwm_interpolate, store_pwm_interpolate, 0, 0),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
524
525
526
527
528
529
  }, {
  	SENSOR_ATTR_2(fan2_input, S_IRUGO, show_fan, NULL, 0, 1),
  	SENSOR_ATTR_2(fan2_full_speed, S_IRUGO|S_IWUSR,
  		      show_fan_full_speed,
  		      store_fan_full_speed, 0, 1),
  	SENSOR_ATTR_2(fan2_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 1),
498be9683   Hans de Goede   hwmon: (f71882fg)...
530
531
532
533
534
  	SENSOR_ATTR_2(pwm2, S_IRUGO|S_IWUSR, show_pwm, store_pwm, 0, 1),
  	SENSOR_ATTR_2(pwm2_enable, S_IRUGO|S_IWUSR, show_pwm_enable,
  		      store_pwm_enable, 0, 1),
  	SENSOR_ATTR_2(pwm2_interpolate, S_IRUGO|S_IWUSR,
  		      show_pwm_interpolate, store_pwm_interpolate, 0, 1),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
535
536
537
538
539
540
  }, {
  	SENSOR_ATTR_2(fan3_input, S_IRUGO, show_fan, NULL, 0, 2),
  	SENSOR_ATTR_2(fan3_full_speed, S_IRUGO|S_IWUSR,
  		      show_fan_full_speed,
  		      store_fan_full_speed, 0, 2),
  	SENSOR_ATTR_2(fan3_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 2),
3fc7838a2   Hans de Goede   hwmon: (f71882fg)...
541
542
543
  	SENSOR_ATTR_2(pwm3, S_IRUGO|S_IWUSR, show_pwm, store_pwm, 0, 2),
  	SENSOR_ATTR_2(pwm3_enable, S_IRUGO|S_IWUSR, show_pwm_enable,
  		      store_pwm_enable, 0, 2),
498be9683   Hans de Goede   hwmon: (f71882fg)...
544
545
  	SENSOR_ATTR_2(pwm3_interpolate, S_IRUGO|S_IWUSR,
  		      show_pwm_interpolate, store_pwm_interpolate, 0, 2),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
546
547
548
549
550
551
552
553
554
555
556
557
  }, {
  	SENSOR_ATTR_2(fan4_input, S_IRUGO, show_fan, NULL, 0, 3),
  	SENSOR_ATTR_2(fan4_full_speed, S_IRUGO|S_IWUSR,
  		      show_fan_full_speed,
  		      store_fan_full_speed, 0, 3),
  	SENSOR_ATTR_2(fan4_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 3),
  	SENSOR_ATTR_2(pwm4, S_IRUGO|S_IWUSR, show_pwm, store_pwm, 0, 3),
  	SENSOR_ATTR_2(pwm4_enable, S_IRUGO|S_IWUSR, show_pwm_enable,
  		      store_pwm_enable, 0, 3),
  	SENSOR_ATTR_2(pwm4_interpolate, S_IRUGO|S_IWUSR,
  		      show_pwm_interpolate, store_pwm_interpolate, 0, 3),
  } };
498be9683   Hans de Goede   hwmon: (f71882fg)...
558

629c58bac   Hans de Goede   hwmon: (f71882fg)...
559
560
561
562
563
564
565
  /* Attr for the third fan of the f71808a, which only has manual pwm */
  static struct sensor_device_attribute_2 f71808a_fan3_attr[] = {
  	SENSOR_ATTR_2(fan3_input, S_IRUGO, show_fan, NULL, 0, 2),
  	SENSOR_ATTR_2(fan3_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 2),
  	SENSOR_ATTR_2(pwm3, S_IRUGO|S_IWUSR,
  		      show_simple_pwm, store_simple_pwm, 0, 2),
  };
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
566
567
  /* Attr for models which can beep on Fan alarm */
  static struct sensor_device_attribute_2 fxxxx_fan_beep_attr[] = {
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
568
569
570
571
572
573
  	SENSOR_ATTR_2(fan1_beep, S_IRUGO|S_IWUSR, show_fan_beep,
  		store_fan_beep, 0, 0),
  	SENSOR_ATTR_2(fan2_beep, S_IRUGO|S_IWUSR, show_fan_beep,
  		store_fan_beep, 0, 1),
  	SENSOR_ATTR_2(fan3_beep, S_IRUGO|S_IWUSR, show_fan_beep,
  		store_fan_beep, 0, 2),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
574
575
  	SENSOR_ATTR_2(fan4_beep, S_IRUGO|S_IWUSR, show_fan_beep,
  		store_fan_beep, 0, 3),
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
576
  };
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
577

66344aa6a   Hans de Goede   hwmon: (f71882fg)...
578
  /* PWM attr for the f71862fg, fewer pwms and fewer zones per pwm than the
3cad40228   Hans de Goede   hwmon/f71882fg: A...
579
     standard models */
558401425   Hans de Goede   hwmon/f71882fg: M...
580
  static struct sensor_device_attribute_2 f71862fg_auto_pwm_attr[3][7] = { {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
581
582
583
  	SENSOR_ATTR_2(pwm1_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 0),
498be9683   Hans de Goede   hwmon: (f71882fg)...
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
  	SENSOR_ATTR_2(pwm1_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 0),
  	SENSOR_ATTR_2(pwm1_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 0),
  	SENSOR_ATTR_2(pwm1_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 0),
558401425   Hans de Goede   hwmon/f71882fg: M...
602
  }, {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
603
604
605
  	SENSOR_ATTR_2(pwm2_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 1),
498be9683   Hans de Goede   hwmon: (f71882fg)...
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
  	SENSOR_ATTR_2(pwm2_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 1),
  	SENSOR_ATTR_2(pwm2_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 1),
  	SENSOR_ATTR_2(pwm2_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 1),
558401425   Hans de Goede   hwmon/f71882fg: M...
624
  }, {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
625
626
627
  	SENSOR_ATTR_2(pwm3_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 2),
4901062f7   Hans de Goede   hwmon: (f71882fg)...
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
  	SENSOR_ATTR_2(pwm3_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 2),
  	SENSOR_ATTR_2(pwm3_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 2),
  	SENSOR_ATTR_2(pwm3_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 2),
558401425   Hans de Goede   hwmon/f71882fg: M...
646
  } };
498be9683   Hans de Goede   hwmon: (f71882fg)...
647

e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
648
  /* PWM attr for the f71808e/f71869, almost identical to the f71862fg, but the
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
649
650
     pwm setting when the temperature is above the pwmX_auto_point1_temp can be
     programmed instead of being hardcoded to 0xff */
558401425   Hans de Goede   hwmon/f71882fg: M...
651
  static struct sensor_device_attribute_2 f71869_auto_pwm_attr[3][8] = { {
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
  	SENSOR_ATTR_2(pwm1_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 0),
  	SENSOR_ATTR_2(pwm1_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 0),
  	SENSOR_ATTR_2(pwm1_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 0),
  	SENSOR_ATTR_2(pwm1_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 0),
558401425   Hans de Goede   hwmon/f71882fg: M...
676
  }, {
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
  	SENSOR_ATTR_2(pwm2_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 1),
  	SENSOR_ATTR_2(pwm2_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 1),
  	SENSOR_ATTR_2(pwm2_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 1),
  	SENSOR_ATTR_2(pwm2_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 1),
558401425   Hans de Goede   hwmon/f71882fg: M...
701
  }, {
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
  	SENSOR_ATTR_2(pwm3_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 2),
  	SENSOR_ATTR_2(pwm3_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 2),
  	SENSOR_ATTR_2(pwm3_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 2),
  	SENSOR_ATTR_2(pwm3_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 2),
558401425   Hans de Goede   hwmon/f71882fg: M...
726
  } };
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
727

3cad40228   Hans de Goede   hwmon/f71882fg: A...
728
  /* PWM attr for the standard models */
b69b03992   Hans de Goede   hwmon: (f71882fg)...
729
  static struct sensor_device_attribute_2 fxxxx_auto_pwm_attr[4][14] = { {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
730
731
732
  	SENSOR_ATTR_2(pwm1_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 0),
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
  	SENSOR_ATTR_2(pwm1_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 0),
  	SENSOR_ATTR_2(pwm1_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      2, 0),
  	SENSOR_ATTR_2(pwm1_auto_point4_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      3, 0),
  	SENSOR_ATTR_2(pwm1_auto_point5_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 0),
  	SENSOR_ATTR_2(pwm1_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      1, 0),
  	SENSOR_ATTR_2(pwm1_auto_point3_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      2, 0),
  	SENSOR_ATTR_2(pwm1_auto_point4_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 0),
  	SENSOR_ATTR_2(pwm1_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 0),
  	SENSOR_ATTR_2(pwm1_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 1, 0),
  	SENSOR_ATTR_2(pwm1_auto_point3_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 2, 0),
  	SENSOR_ATTR_2(pwm1_auto_point4_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 0),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
770
  }, {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
771
772
773
  	SENSOR_ATTR_2(pwm2_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 1),
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
  	SENSOR_ATTR_2(pwm2_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 1),
  	SENSOR_ATTR_2(pwm2_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      2, 1),
  	SENSOR_ATTR_2(pwm2_auto_point4_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      3, 1),
  	SENSOR_ATTR_2(pwm2_auto_point5_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 1),
  	SENSOR_ATTR_2(pwm2_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      1, 1),
  	SENSOR_ATTR_2(pwm2_auto_point3_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      2, 1),
  	SENSOR_ATTR_2(pwm2_auto_point4_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 1),
  	SENSOR_ATTR_2(pwm2_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 1),
  	SENSOR_ATTR_2(pwm2_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 1, 1),
  	SENSOR_ATTR_2(pwm2_auto_point3_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 2, 1),
  	SENSOR_ATTR_2(pwm2_auto_point4_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 1),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
811
  }, {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
812
813
814
  	SENSOR_ATTR_2(pwm3_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 2),
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
  	SENSOR_ATTR_2(pwm3_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 2),
  	SENSOR_ATTR_2(pwm3_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      2, 2),
  	SENSOR_ATTR_2(pwm3_auto_point4_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      3, 2),
  	SENSOR_ATTR_2(pwm3_auto_point5_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 2),
  	SENSOR_ATTR_2(pwm3_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      1, 2),
  	SENSOR_ATTR_2(pwm3_auto_point3_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      2, 2),
  	SENSOR_ATTR_2(pwm3_auto_point4_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 2),
  	SENSOR_ATTR_2(pwm3_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 2),
  	SENSOR_ATTR_2(pwm3_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 1, 2),
  	SENSOR_ATTR_2(pwm3_auto_point3_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 2, 2),
  	SENSOR_ATTR_2(pwm3_auto_point4_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 2),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
852
  }, {
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
  	SENSOR_ATTR_2(pwm4_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 3),
  	SENSOR_ATTR_2(pwm4_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 3),
  	SENSOR_ATTR_2(pwm4_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 3),
  	SENSOR_ATTR_2(pwm4_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      2, 3),
  	SENSOR_ATTR_2(pwm4_auto_point4_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      3, 3),
  	SENSOR_ATTR_2(pwm4_auto_point5_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 3),
  	SENSOR_ATTR_2(pwm4_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 3),
  	SENSOR_ATTR_2(pwm4_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      1, 3),
  	SENSOR_ATTR_2(pwm4_auto_point3_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      2, 3),
  	SENSOR_ATTR_2(pwm4_auto_point4_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 3),
  	SENSOR_ATTR_2(pwm4_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 3),
  	SENSOR_ATTR_2(pwm4_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 1, 3),
  	SENSOR_ATTR_2(pwm4_auto_point3_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 2, 3),
  	SENSOR_ATTR_2(pwm4_auto_point4_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 3),
b69b03992   Hans de Goede   hwmon: (f71882fg)...
893
  } };
45fb36694   Hans de Goede   hwmon: add suppor...
894

66344aa6a   Hans de Goede   hwmon: (f71882fg)...
895
  /* Fan attr specific to the f8000 (4th fan input can only measure speed) */
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
896
897
  static struct sensor_device_attribute_2 f8000_fan_attr[] = {
  	SENSOR_ATTR_2(fan4_input, S_IRUGO, show_fan, NULL, 0, 3),
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
898
  };
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
899

66344aa6a   Hans de Goede   hwmon: (f71882fg)...
900
901
902
  /* PWM attr for the f8000, zones mapped to temp instead of to pwm!
     Also the register block at offset A0 maps to TEMP1 (so our temp2, as the
     F8000 starts counting temps at 0), B0 maps the TEMP2 and C0 maps to TEMP0 */
558401425   Hans de Goede   hwmon/f71882fg: M...
903
  static struct sensor_device_attribute_2 f8000_auto_pwm_attr[3][14] = { {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
904
905
906
  	SENSOR_ATTR_2(pwm1_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 0),
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
  	SENSOR_ATTR_2(temp1_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 2),
  	SENSOR_ATTR_2(temp1_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 2),
  	SENSOR_ATTR_2(temp1_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      2, 2),
  	SENSOR_ATTR_2(temp1_auto_point4_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      3, 2),
  	SENSOR_ATTR_2(temp1_auto_point5_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 2),
  	SENSOR_ATTR_2(temp1_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 2),
  	SENSOR_ATTR_2(temp1_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      1, 2),
  	SENSOR_ATTR_2(temp1_auto_point3_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      2, 2),
  	SENSOR_ATTR_2(temp1_auto_point4_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 2),
  	SENSOR_ATTR_2(temp1_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 2),
  	SENSOR_ATTR_2(temp1_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 1, 2),
  	SENSOR_ATTR_2(temp1_auto_point3_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 2, 2),
  	SENSOR_ATTR_2(temp1_auto_point4_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 2),
558401425   Hans de Goede   hwmon/f71882fg: M...
944
  }, {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
945
946
947
  	SENSOR_ATTR_2(pwm2_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 1),
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
  	SENSOR_ATTR_2(temp2_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 0),
  	SENSOR_ATTR_2(temp2_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 0),
  	SENSOR_ATTR_2(temp2_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      2, 0),
  	SENSOR_ATTR_2(temp2_auto_point4_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      3, 0),
  	SENSOR_ATTR_2(temp2_auto_point5_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 0),
  	SENSOR_ATTR_2(temp2_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 0),
  	SENSOR_ATTR_2(temp2_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      1, 0),
  	SENSOR_ATTR_2(temp2_auto_point3_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      2, 0),
  	SENSOR_ATTR_2(temp2_auto_point4_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 0),
  	SENSOR_ATTR_2(temp2_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 0),
  	SENSOR_ATTR_2(temp2_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 1, 0),
  	SENSOR_ATTR_2(temp2_auto_point3_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 2, 0),
  	SENSOR_ATTR_2(temp2_auto_point4_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 0),
558401425   Hans de Goede   hwmon/f71882fg: M...
985
  }, {
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
986
987
988
  	SENSOR_ATTR_2(pwm3_auto_channels_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_channel,
  		      store_pwm_auto_point_channel, 0, 2),
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
  	SENSOR_ATTR_2(temp3_auto_point1_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      0, 1),
  	SENSOR_ATTR_2(temp3_auto_point2_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      1, 1),
  	SENSOR_ATTR_2(temp3_auto_point3_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      2, 1),
  	SENSOR_ATTR_2(temp3_auto_point4_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      3, 1),
  	SENSOR_ATTR_2(temp3_auto_point5_pwm, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
  		      4, 1),
  	SENSOR_ATTR_2(temp3_auto_point1_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      0, 1),
  	SENSOR_ATTR_2(temp3_auto_point2_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      1, 1),
  	SENSOR_ATTR_2(temp3_auto_point3_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      2, 1),
  	SENSOR_ATTR_2(temp3_auto_point4_temp, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp, store_pwm_auto_point_temp,
  		      3, 1),
  	SENSOR_ATTR_2(temp3_auto_point1_temp_hyst, S_IRUGO|S_IWUSR,
  		      show_pwm_auto_point_temp_hyst,
  		      store_pwm_auto_point_temp_hyst,
  		      0, 1),
  	SENSOR_ATTR_2(temp3_auto_point2_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 1, 1),
  	SENSOR_ATTR_2(temp3_auto_point3_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 2, 1),
  	SENSOR_ATTR_2(temp3_auto_point4_temp_hyst, S_IRUGO,
  		      show_pwm_auto_point_temp_hyst, NULL, 3, 1),
558401425   Hans de Goede   hwmon/f71882fg: M...
1026
  } };
45fb36694   Hans de Goede   hwmon: add suppor...
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
  
  /* Super I/O functions */
  static inline int superio_inb(int base, int reg)
  {
  	outb(reg, base);
  	return inb(base + 1);
  }
  
  static int superio_inw(int base, int reg)
  {
  	int val;
bd328acdc   Giel van Schijndel   hwmon: (f71882fg)...
1038
1039
  	val  = superio_inb(base, reg) << 8;
  	val |= superio_inb(base, reg + 1);
45fb36694   Hans de Goede   hwmon: add suppor...
1040
1041
  	return val;
  }
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
1042
  static inline int superio_enter(int base)
45fb36694   Hans de Goede   hwmon: add suppor...
1043
  {
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
1044
1045
  	/* Don't step on other drivers' I/O space by accident */
  	if (!request_muxed_region(base, 2, DRVNAME)) {
22d3b4128   Joe Perches   hwmon: (f71882fg)...
1046
1047
  		pr_err("I/O address 0x%04x already in use
  ", base);
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
1048
1049
  		return -EBUSY;
  	}
45fb36694   Hans de Goede   hwmon: add suppor...
1050
  	/* according to the datasheet the key must be send twice! */
162bb59e4   Giel van Schijndel   hwmon: (f71882fg)...
1051
1052
  	outb(SIO_UNLOCK_KEY, base);
  	outb(SIO_UNLOCK_KEY, base);
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
1053
1054
  
  	return 0;
45fb36694   Hans de Goede   hwmon: add suppor...
1055
  }
162bb59e4   Giel van Schijndel   hwmon: (f71882fg)...
1056
  static inline void superio_select(int base, int ld)
45fb36694   Hans de Goede   hwmon: add suppor...
1057
1058
1059
1060
1061
1062
1063
1064
  {
  	outb(SIO_REG_LDSEL, base);
  	outb(ld, base + 1);
  }
  
  static inline void superio_exit(int base)
  {
  	outb(SIO_LOCK_KEY, base);
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
1065
  	release_region(base, 2);
45fb36694   Hans de Goede   hwmon: add suppor...
1066
  }
2f650631b   Hans de Goede   hwmon: (f71882fg)...
1067
  static inline int fan_from_reg(u16 reg)
45fb36694   Hans de Goede   hwmon: add suppor...
1068
1069
1070
  {
  	return reg ? (1500000 / reg) : 0;
  }
2f650631b   Hans de Goede   hwmon: (f71882fg)...
1071
  static inline u16 fan_to_reg(int fan)
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1072
1073
1074
  {
  	return fan ? (1500000 / fan) : 0;
  }
45fb36694   Hans de Goede   hwmon: add suppor...
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
  static u8 f71882fg_read8(struct f71882fg_data *data, u8 reg)
  {
  	u8 val;
  
  	outb(reg, data->addr + ADDR_REG_OFFSET);
  	val = inb(data->addr + DATA_REG_OFFSET);
  
  	return val;
  }
  
  static u16 f71882fg_read16(struct f71882fg_data *data, u8 reg)
  {
  	u16 val;
bd328acdc   Giel van Schijndel   hwmon: (f71882fg)...
1088
1089
  	val  = f71882fg_read8(data, reg) << 8;
  	val |= f71882fg_read8(data, reg + 1);
45fb36694   Hans de Goede   hwmon: add suppor...
1090
1091
1092
1093
1094
1095
1096
1097
1098
  
  	return val;
  }
  
  static void f71882fg_write8(struct f71882fg_data *data, u8 reg, u8 val)
  {
  	outb(reg, data->addr + ADDR_REG_OFFSET);
  	outb(val, data->addr + DATA_REG_OFFSET);
  }
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1099
1100
  static void f71882fg_write16(struct f71882fg_data *data, u8 reg, u16 val)
  {
bd328acdc   Giel van Schijndel   hwmon: (f71882fg)...
1101
1102
  	f71882fg_write8(data, reg,     val >> 8);
  	f71882fg_write8(data, reg + 1, val & 0xff);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1103
  }
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1104
1105
1106
1107
1108
1109
1110
  static u16 f71882fg_read_temp(struct f71882fg_data *data, int nr)
  {
  	if (data->type == f71858fg)
  		return f71882fg_read16(data, F71882FG_REG_TEMP(nr));
  	else
  		return f71882fg_read8(data, F71882FG_REG_TEMP(nr));
  }
77a4a3e2a   Mark van Doesburg   hwmon: (f71882fg)...
1111
  static struct f71882fg_data *f71882fg_update_device(struct device *dev)
45fb36694   Hans de Goede   hwmon: add suppor...
1112
1113
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
f27def07b   Jean Delvare   hwmon: (f71882fg)...
1114
1115
  	int nr_fans = f71882fg_nr_fans[data->type];
  	int nr_temps = f71882fg_nr_temps[data->type];
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
1116
  	int nr, reg, point;
45fb36694   Hans de Goede   hwmon: add suppor...
1117
1118
1119
1120
  
  	mutex_lock(&data->update_lock);
  
  	/* Update once every 60 seconds */
162bb59e4   Giel van Schijndel   hwmon: (f71882fg)...
1121
  	if (time_after(jiffies, data->last_limits + 60 * HZ) ||
45fb36694   Hans de Goede   hwmon: add suppor...
1122
  			!data->valid) {
0bae64008   Hans de Goede   hwmon/f71882fg: M...
1123
  		if (f71882fg_has_in1_alarm[data->type]) {
498be9683   Hans de Goede   hwmon: (f71882fg)...
1124
1125
1126
1127
1128
  			data->in1_max =
  				f71882fg_read8(data, F71882FG_REG_IN1_HIGH);
  			data->in_beep =
  				f71882fg_read8(data, F71882FG_REG_IN_BEEP);
  		}
45fb36694   Hans de Goede   hwmon: add suppor...
1129
1130
  
  		/* Get High & boundary temps*/
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
1131
1132
  		for (nr = data->temp_start; nr < nr_temps + data->temp_start;
  									nr++) {
45fb36694   Hans de Goede   hwmon: add suppor...
1133
1134
1135
1136
1137
  			data->temp_ovt[nr] = f71882fg_read8(data,
  						F71882FG_REG_TEMP_OVT(nr));
  			data->temp_high[nr] = f71882fg_read8(data,
  						F71882FG_REG_TEMP_HIGH(nr));
  		}
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1138
  		if (data->type != f8000) {
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1139
1140
1141
1142
  			data->temp_hyst[0] = f71882fg_read8(data,
  						F71882FG_REG_TEMP_HYST(0));
  			data->temp_hyst[1] = f71882fg_read8(data,
  						F71882FG_REG_TEMP_HYST(1));
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1143
  		}
78aa4f72b   Hans de Goede   hwmon/f71882fg: S...
1144
1145
1146
1147
1148
1149
1150
  		/* All but the f71858fg / f8000 have this register */
  		if ((data->type != f71858fg) && (data->type != f8000)) {
  			reg  = f71882fg_read8(data, F71882FG_REG_TEMP_TYPE);
  			data->temp_type[1] = (reg & 0x02) ? 2 : 4;
  			data->temp_type[2] = (reg & 0x04) ? 2 : 4;
  			data->temp_type[3] = (reg & 0x08) ? 2 : 4;
  		}
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1151

4d53811a2   Hans de Goede   hwmon: (f71882fg)...
1152
  		if (f71882fg_fan_has_beep[data->type])
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1153
1154
  			data->fan_beep = f71882fg_read8(data,
  						F71882FG_REG_FAN_BEEP);
4d53811a2   Hans de Goede   hwmon: (f71882fg)...
1155
1156
  
  		if (f71882fg_temp_has_beep[data->type])
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1157
1158
  			data->temp_beep = f71882fg_read8(data,
  						F71882FG_REG_TEMP_BEEP);
45fb36694   Hans de Goede   hwmon: add suppor...
1159

9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1160
1161
  		data->pwm_enable = f71882fg_read8(data,
  						  F71882FG_REG_PWM_ENABLE);
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
1162
1163
1164
1165
  		data->pwm_auto_point_hyst[0] =
  			f71882fg_read8(data, F71882FG_REG_FAN_HYST(0));
  		data->pwm_auto_point_hyst[1] =
  			f71882fg_read8(data, F71882FG_REG_FAN_HYST(1));
498be9683   Hans de Goede   hwmon: (f71882fg)...
1166
  		for (nr = 0; nr < nr_fans; nr++) {
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1167
1168
1169
  			data->pwm_auto_point_mapping[nr] =
  			    f71882fg_read8(data,
  					   F71882FG_REG_POINT_MAPPING(nr));
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
1170
1171
  			switch (data->type) {
  			default:
498be9683   Hans de Goede   hwmon: (f71882fg)...
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
  				for (point = 0; point < 5; point++) {
  					data->pwm_auto_point_pwm[nr][point] =
  						f71882fg_read8(data,
  							F71882FG_REG_POINT_PWM
  							(nr, point));
  				}
  				for (point = 0; point < 4; point++) {
  					data->pwm_auto_point_temp[nr][point] =
  						f71882fg_read8(data,
  							F71882FG_REG_POINT_TEMP
  							(nr, point));
  				}
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
1184
1185
1186
1187
1188
1189
1190
1191
  				break;
  			case f71808e:
  			case f71869:
  				data->pwm_auto_point_pwm[nr][0] =
  					f71882fg_read8(data,
  						F71882FG_REG_POINT_PWM(nr, 0));
  				/* Fall through */
  			case f71862fg:
498be9683   Hans de Goede   hwmon: (f71882fg)...
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
  				data->pwm_auto_point_pwm[nr][1] =
  					f71882fg_read8(data,
  						F71882FG_REG_POINT_PWM
  						(nr, 1));
  				data->pwm_auto_point_pwm[nr][4] =
  					f71882fg_read8(data,
  						F71882FG_REG_POINT_PWM
  						(nr, 4));
  				data->pwm_auto_point_temp[nr][0] =
  					f71882fg_read8(data,
  						F71882FG_REG_POINT_TEMP
  						(nr, 0));
  				data->pwm_auto_point_temp[nr][3] =
  					f71882fg_read8(data,
  						F71882FG_REG_POINT_TEMP
  						(nr, 3));
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
1208
  				break;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1209
1210
  			}
  		}
45fb36694   Hans de Goede   hwmon: add suppor...
1211
1212
1213
1214
  		data->last_limits = jiffies;
  	}
  
  	/* Update every second */
8afb10490   Mark M. Hoffman   hwmon: (f71882fg)...
1215
  	if (time_after(jiffies, data->last_updated + HZ) || !data->valid) {
45fb36694   Hans de Goede   hwmon: add suppor...
1216
1217
1218
1219
  		data->temp_status = f71882fg_read8(data,
  						F71882FG_REG_TEMP_STATUS);
  		data->temp_diode_open = f71882fg_read8(data,
  						F71882FG_REG_TEMP_DIODE_OPEN);
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
1220
1221
  		for (nr = data->temp_start; nr < nr_temps + data->temp_start;
  									nr++)
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1222
  			data->temp[nr] = f71882fg_read_temp(data, nr);
45fb36694   Hans de Goede   hwmon: add suppor...
1223
1224
1225
  
  		data->fan_status = f71882fg_read8(data,
  						F71882FG_REG_FAN_STATUS);
498be9683   Hans de Goede   hwmon: (f71882fg)...
1226
  		for (nr = 0; nr < nr_fans; nr++) {
45fb36694   Hans de Goede   hwmon: add suppor...
1227
1228
  			data->fan[nr] = f71882fg_read16(data,
  						F71882FG_REG_FAN(nr));
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1229
1230
1231
1232
1233
1234
1235
1236
  			data->fan_target[nr] =
  			    f71882fg_read16(data, F71882FG_REG_FAN_TARGET(nr));
  			data->fan_full_speed[nr] =
  			    f71882fg_read16(data,
  					    F71882FG_REG_FAN_FULL_SPEED(nr));
  			data->pwm[nr] =
  			    f71882fg_read8(data, F71882FG_REG_PWM(nr));
  		}
629c58bac   Hans de Goede   hwmon: (f71882fg)...
1237
1238
1239
1240
1241
1242
1243
  		/* Some models have 1 more fan with limited capabilities */
  		if (data->type == f71808a) {
  			data->fan[2] = f71882fg_read16(data,
  						F71882FG_REG_FAN(2));
  			data->pwm[2] = f71882fg_read8(data,
  							F71882FG_REG_PWM(2));
  		}
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1244
1245
1246
  		if (data->type == f8000)
  			data->fan[3] = f71882fg_read16(data,
  						F71882FG_REG_FAN(3));
0bae64008   Hans de Goede   hwmon/f71882fg: M...
1247
1248
  
  		if (f71882fg_has_in1_alarm[data->type])
498be9683   Hans de Goede   hwmon: (f71882fg)...
1249
  			data->in_status = f71882fg_read8(data,
45fb36694   Hans de Goede   hwmon: add suppor...
1250
  						F71882FG_REG_IN_STATUS);
0bae64008   Hans de Goede   hwmon/f71882fg: M...
1251
1252
1253
1254
  		for (nr = 0; nr < F71882FG_MAX_INS; nr++)
  			if (f71882fg_has_in[data->type][nr])
  				data->in[nr] = f71882fg_read8(data,
  							F71882FG_REG_IN(nr));
45fb36694   Hans de Goede   hwmon: add suppor...
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
  
  		data->last_updated = jiffies;
  		data->valid = 1;
  	}
  
  	mutex_unlock(&data->update_lock);
  
  	return data;
  }
  
  /* Sysfs Interface */
  static ssize_t show_fan(struct device *dev, struct device_attribute *devattr,
  	char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1270
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1271
1272
1273
1274
1275
1276
1277
1278
  	int speed = fan_from_reg(data->fan[nr]);
  
  	if (speed == FAN_MIN_DETECT)
  		speed = 0;
  
  	return sprintf(buf, "%d
  ", speed);
  }
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
  static ssize_t show_fan_full_speed(struct device *dev,
  				   struct device_attribute *devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int nr = to_sensor_dev_attr_2(devattr)->index;
  	int speed = fan_from_reg(data->fan_full_speed[nr]);
  	return sprintf(buf, "%d
  ", speed);
  }
  
  static ssize_t store_fan_full_speed(struct device *dev,
  				    struct device_attribute *devattr,
  				    const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1294
1295
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1296
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1297
1298
  	if (err)
  		return err;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1299
1300
1301
1302
1303
  
  	val = SENSORS_LIMIT(val, 23, 1500000);
  	val = fan_to_reg(val);
  
  	mutex_lock(&data->update_lock);
4c82c38ae   Hans de Goede   hwmon: (f71882fg)...
1304
1305
  	f71882fg_write16(data, F71882FG_REG_FAN_FULL_SPEED(nr), val);
  	data->fan_full_speed[nr] = val;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1306
1307
1308
1309
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
45fb36694   Hans de Goede   hwmon: add suppor...
1310
1311
1312
1313
  static ssize_t show_fan_beep(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1314
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
  
  	if (data->fan_beep & (1 << nr))
  		return sprintf(buf, "1
  ");
  	else
  		return sprintf(buf, "0
  ");
  }
  
  static ssize_t store_fan_beep(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1328
1329
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	unsigned long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1330
  	err = kstrtoul(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1331
1332
  	if (err)
  		return err;
45fb36694   Hans de Goede   hwmon: add suppor...
1333
1334
  
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1335
  	data->fan_beep = f71882fg_read8(data, F71882FG_REG_FAN_BEEP);
45fb36694   Hans de Goede   hwmon: add suppor...
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
  	if (val)
  		data->fan_beep |= 1 << nr;
  	else
  		data->fan_beep &= ~(1 << nr);
  
  	f71882fg_write8(data, F71882FG_REG_FAN_BEEP, data->fan_beep);
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_fan_alarm(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1351
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
  
  	if (data->fan_status & (1 << nr))
  		return sprintf(buf, "1
  ");
  	else
  		return sprintf(buf, "0
  ");
  }
  
  static ssize_t show_in(struct device *dev, struct device_attribute *devattr,
  	char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1365
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
  
  	return sprintf(buf, "%d
  ", data->in[nr] * 8);
  }
  
  static ssize_t show_in_max(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  
  	return sprintf(buf, "%d
  ", data->in1_max * 8);
  }
  
  static ssize_t store_in_max(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1384
1385
  	int err;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1386
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1387
1388
1389
1390
  	if (err)
  		return err;
  
  	val /= 8;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1391
  	val = SENSORS_LIMIT(val, 0, 255);
45fb36694   Hans de Goede   hwmon: add suppor...
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
  
  	mutex_lock(&data->update_lock);
  	f71882fg_write8(data, F71882FG_REG_IN1_HIGH, val);
  	data->in1_max = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_in_beep(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1405
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
  
  	if (data->in_beep & (1 << nr))
  		return sprintf(buf, "1
  ");
  	else
  		return sprintf(buf, "0
  ");
  }
  
  static ssize_t store_in_beep(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1419
1420
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	unsigned long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1421
  	err = kstrtoul(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1422
1423
  	if (err)
  		return err;
45fb36694   Hans de Goede   hwmon: add suppor...
1424
1425
  
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1426
  	data->in_beep = f71882fg_read8(data, F71882FG_REG_IN_BEEP);
45fb36694   Hans de Goede   hwmon: add suppor...
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
  	if (val)
  		data->in_beep |= 1 << nr;
  	else
  		data->in_beep &= ~(1 << nr);
  
  	f71882fg_write8(data, F71882FG_REG_IN_BEEP, data->in_beep);
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_in_alarm(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1442
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1443
1444
1445
1446
1447
1448
1449
1450
1451
1452
1453
1454
1455
  
  	if (data->in_status & (1 << nr))
  		return sprintf(buf, "1
  ");
  	else
  		return sprintf(buf, "0
  ");
  }
  
  static ssize_t show_temp(struct device *dev, struct device_attribute *devattr,
  	char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1456
  	int nr = to_sensor_dev_attr_2(devattr)->index;
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1457
1458
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
  	int sign, temp;
  
  	if (data->type == f71858fg) {
  		/* TEMP_TABLE_SEL 1 or 3 ? */
  		if (data->temp_config & 1) {
  			sign = data->temp[nr] & 0x0001;
  			temp = (data->temp[nr] >> 5) & 0x7ff;
  		} else {
  			sign = data->temp[nr] & 0x8000;
  			temp = (data->temp[nr] >> 5) & 0x3ff;
  		}
  		temp *= 125;
  		if (sign)
  			temp -= 128000;
  	} else
  		temp = data->temp[nr] * 1000;
45fb36694   Hans de Goede   hwmon: add suppor...
1473

09475d32e   Hans de Goede   hwmon: (f71882fg)...
1474
1475
  	return sprintf(buf, "%d
  ", temp);
45fb36694   Hans de Goede   hwmon: add suppor...
1476
1477
1478
1479
1480
1481
  }
  
  static ssize_t show_temp_max(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1482
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1483
1484
1485
1486
1487
1488
1489
1490
1491
  
  	return sprintf(buf, "%d
  ", data->temp_high[nr] * 1000);
  }
  
  static ssize_t store_temp_max(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1492
1493
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1494
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1495
1496
1497
1498
  	if (err)
  		return err;
  
  	val /= 1000;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1499
  	val = SENSORS_LIMIT(val, 0, 255);
45fb36694   Hans de Goede   hwmon: add suppor...
1500
1501
1502
1503
1504
1505
1506
1507
1508
1509
1510
1511
1512
  
  	mutex_lock(&data->update_lock);
  	f71882fg_write8(data, F71882FG_REG_TEMP_HIGH(nr), val);
  	data->temp_high[nr] = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_temp_max_hyst(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1513
  	int nr = to_sensor_dev_attr_2(devattr)->index;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1514
  	int temp_max_hyst;
45fb36694   Hans de Goede   hwmon: add suppor...
1515

ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1516
  	mutex_lock(&data->update_lock);
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
1517
1518
1519
1520
1521
  	if (nr & 1)
  		temp_max_hyst = data->temp_hyst[nr / 2] >> 4;
  	else
  		temp_max_hyst = data->temp_hyst[nr / 2] & 0x0f;
  	temp_max_hyst = (data->temp_high[nr] - temp_max_hyst) * 1000;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1522
1523
1524
1525
  	mutex_unlock(&data->update_lock);
  
  	return sprintf(buf, "%d
  ", temp_max_hyst);
45fb36694   Hans de Goede   hwmon: add suppor...
1526
1527
1528
1529
1530
1531
  }
  
  static ssize_t store_temp_max_hyst(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1532
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1533
  	ssize_t ret = count;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1534
  	u8 reg;
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1535
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1536
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1537
1538
1539
1540
  	if (err)
  		return err;
  
  	val /= 1000;
45fb36694   Hans de Goede   hwmon: add suppor...
1541
1542
1543
1544
  
  	mutex_lock(&data->update_lock);
  
  	/* convert abs to relative and check */
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1545
1546
1547
  	data->temp_high[nr] = f71882fg_read8(data, F71882FG_REG_TEMP_HIGH(nr));
  	val = SENSORS_LIMIT(val, data->temp_high[nr] - 15,
  			    data->temp_high[nr]);
45fb36694   Hans de Goede   hwmon: add suppor...
1548
  	val = data->temp_high[nr] - val;
45fb36694   Hans de Goede   hwmon: add suppor...
1549
1550
  
  	/* convert value to register contents */
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
1551
1552
1553
1554
1555
1556
1557
  	reg = f71882fg_read8(data, F71882FG_REG_TEMP_HYST(nr / 2));
  	if (nr & 1)
  		reg = (reg & 0x0f) | (val << 4);
  	else
  		reg = (reg & 0xf0) | val;
  	f71882fg_write8(data, F71882FG_REG_TEMP_HYST(nr / 2), reg);
  	data->temp_hyst[nr / 2] = reg;
45fb36694   Hans de Goede   hwmon: add suppor...
1558

45fb36694   Hans de Goede   hwmon: add suppor...
1559
1560
1561
1562
1563
1564
1565
1566
  	mutex_unlock(&data->update_lock);
  	return ret;
  }
  
  static ssize_t show_temp_crit(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1567
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1568
1569
1570
1571
1572
1573
1574
1575
1576
  
  	return sprintf(buf, "%d
  ", data->temp_ovt[nr] * 1000);
  }
  
  static ssize_t store_temp_crit(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1577
1578
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1579
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1580
1581
1582
1583
  	if (err)
  		return err;
  
  	val /= 1000;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1584
  	val = SENSORS_LIMIT(val, 0, 255);
45fb36694   Hans de Goede   hwmon: add suppor...
1585
1586
1587
1588
1589
1590
1591
1592
1593
1594
1595
1596
1597
  
  	mutex_lock(&data->update_lock);
  	f71882fg_write8(data, F71882FG_REG_TEMP_OVT(nr), val);
  	data->temp_ovt[nr] = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_temp_crit_hyst(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1598
  	int nr = to_sensor_dev_attr_2(devattr)->index;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1599
  	int temp_crit_hyst;
45fb36694   Hans de Goede   hwmon: add suppor...
1600

ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1601
  	mutex_lock(&data->update_lock);
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
1602
1603
1604
1605
1606
  	if (nr & 1)
  		temp_crit_hyst = data->temp_hyst[nr / 2] >> 4;
  	else
  		temp_crit_hyst = data->temp_hyst[nr / 2] & 0x0f;
  	temp_crit_hyst = (data->temp_ovt[nr] - temp_crit_hyst) * 1000;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1607
1608
1609
1610
  	mutex_unlock(&data->update_lock);
  
  	return sprintf(buf, "%d
  ", temp_crit_hyst);
45fb36694   Hans de Goede   hwmon: add suppor...
1611
1612
1613
1614
1615
1616
  }
  
  static ssize_t show_temp_type(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1617
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1618
1619
1620
1621
1622
1623
1624
1625
1626
  
  	return sprintf(buf, "%d
  ", data->temp_type[nr]);
  }
  
  static ssize_t show_temp_beep(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1627
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1628

7567a0435   Hans de Goede   hwmon: (f71882fg)...
1629
  	if (data->temp_beep & (1 << nr))
45fb36694   Hans de Goede   hwmon: add suppor...
1630
1631
1632
1633
1634
1635
1636
1637
1638
1639
1640
  		return sprintf(buf, "1
  ");
  	else
  		return sprintf(buf, "0
  ");
  }
  
  static ssize_t store_temp_beep(struct device *dev, struct device_attribute
  	*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1641
1642
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	unsigned long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1643
  	err = kstrtoul(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1644
1645
  	if (err)
  		return err;
45fb36694   Hans de Goede   hwmon: add suppor...
1646
1647
  
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1648
  	data->temp_beep = f71882fg_read8(data, F71882FG_REG_TEMP_BEEP);
45fb36694   Hans de Goede   hwmon: add suppor...
1649
  	if (val)
7567a0435   Hans de Goede   hwmon: (f71882fg)...
1650
  		data->temp_beep |= 1 << nr;
45fb36694   Hans de Goede   hwmon: add suppor...
1651
  	else
7567a0435   Hans de Goede   hwmon: (f71882fg)...
1652
  		data->temp_beep &= ~(1 << nr);
45fb36694   Hans de Goede   hwmon: add suppor...
1653
1654
1655
1656
1657
1658
1659
1660
1661
1662
1663
  
  	f71882fg_write8(data, F71882FG_REG_TEMP_BEEP, data->temp_beep);
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_temp_alarm(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1664
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1665

7567a0435   Hans de Goede   hwmon: (f71882fg)...
1666
  	if (data->temp_status & (1 << nr))
45fb36694   Hans de Goede   hwmon: add suppor...
1667
1668
1669
1670
1671
1672
1673
1674
1675
1676
1677
  		return sprintf(buf, "1
  ");
  	else
  		return sprintf(buf, "0
  ");
  }
  
  static ssize_t show_temp_fault(struct device *dev, struct device_attribute
  	*devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
bc37ae712   Mark van Doesburg   hwmon: (f71882fg)...
1678
  	int nr = to_sensor_dev_attr_2(devattr)->index;
45fb36694   Hans de Goede   hwmon: add suppor...
1679

7567a0435   Hans de Goede   hwmon: (f71882fg)...
1680
  	if (data->temp_diode_open & (1 << nr))
45fb36694   Hans de Goede   hwmon: add suppor...
1681
1682
1683
1684
1685
1686
  		return sprintf(buf, "1
  ");
  	else
  		return sprintf(buf, "0
  ");
  }
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1687
1688
1689
1690
1691
  static ssize_t show_pwm(struct device *dev,
  			struct device_attribute *devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int val, nr = to_sensor_dev_attr_2(devattr)->index;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1692
  	mutex_lock(&data->update_lock);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1693
1694
1695
1696
1697
  	if (data->pwm_enable & (1 << (2 * nr)))
  		/* PWM mode */
  		val = data->pwm[nr];
  	else {
  		/* RPM mode */
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1698
1699
  		val = 255 * fan_from_reg(data->fan_target[nr])
  			/ fan_from_reg(data->fan_full_speed[nr]);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1700
  	}
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1701
  	mutex_unlock(&data->update_lock);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1702
1703
1704
1705
1706
1707
1708
1709
  	return sprintf(buf, "%d
  ", val);
  }
  
  static ssize_t store_pwm(struct device *dev,
  			 struct device_attribute *devattr, const char *buf,
  			 size_t count)
  {
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1710
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1711
1712
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1713
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1714
1715
  	if (err)
  		return err;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1716
1717
1718
  	val = SENSORS_LIMIT(val, 0, 255);
  
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1719
  	data->pwm_enable = f71882fg_read8(data, F71882FG_REG_PWM_ENABLE);
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1720
1721
1722
1723
1724
  	if ((data->type == f8000 && ((data->pwm_enable >> 2 * nr) & 3) != 2) ||
  	    (data->type != f8000 && !((data->pwm_enable >> 2 * nr) & 2))) {
  		count = -EROFS;
  		goto leave;
  	}
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1725
1726
1727
1728
1729
1730
  	if (data->pwm_enable & (1 << (2 * nr))) {
  		/* PWM mode */
  		f71882fg_write8(data, F71882FG_REG_PWM(nr), val);
  		data->pwm[nr] = val;
  	} else {
  		/* RPM mode */
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1731
1732
1733
1734
1735
1736
1737
  		int target, full_speed;
  		full_speed = f71882fg_read16(data,
  					     F71882FG_REG_FAN_FULL_SPEED(nr));
  		target = fan_to_reg(val * fan_from_reg(full_speed) / 255);
  		f71882fg_write16(data, F71882FG_REG_FAN_TARGET(nr), target);
  		data->fan_target[nr] = target;
  		data->fan_full_speed[nr] = full_speed;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1738
  	}
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1739
  leave:
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1740
1741
1742
1743
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
629c58bac   Hans de Goede   hwmon: (f71882fg)...
1744
1745
1746
1747
1748
1749
1750
1751
1752
1753
1754
1755
1756
1757
1758
1759
1760
1761
  static ssize_t show_simple_pwm(struct device *dev,
  			       struct device_attribute *devattr, char *buf)
  {
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int val, nr = to_sensor_dev_attr_2(devattr)->index;
  
  	val = data->pwm[nr];
  	return sprintf(buf, "%d
  ", val);
  }
  
  static ssize_t store_simple_pwm(struct device *dev,
  				struct device_attribute *devattr,
  				const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1762
  	err = kstrtol(buf, 10, &val);
629c58bac   Hans de Goede   hwmon: (f71882fg)...
1763
1764
1765
1766
1767
1768
1769
1770
1771
1772
1773
1774
  	if (err)
  		return err;
  
  	val = SENSORS_LIMIT(val, 0, 255);
  
  	mutex_lock(&data->update_lock);
  	f71882fg_write8(data, F71882FG_REG_PWM(nr), val);
  	data->pwm[nr] = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1775
1776
1777
  static ssize_t show_pwm_enable(struct device *dev,
  			       struct device_attribute *devattr, char *buf)
  {
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1778
  	int result = 0;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1779
1780
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int nr = to_sensor_dev_attr_2(devattr)->index;
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1781
1782
1783
1784
1785
1786
1787
1788
1789
1790
1791
1792
1793
1794
1795
  	switch ((data->pwm_enable >> 2 * nr) & 3) {
  	case 0:
  	case 1:
  		result = 2; /* Normal auto mode */
  		break;
  	case 2:
  		result = 1; /* Manual mode */
  		break;
  	case 3:
  		if (data->type == f8000)
  			result = 3; /* Thermostat mode */
  		else
  			result = 1; /* Manual mode */
  		break;
  	}
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1796
1797
1798
1799
1800
1801
1802
1803
1804
  
  	return sprintf(buf, "%d
  ", result);
  }
  
  static ssize_t store_pwm_enable(struct device *dev, struct device_attribute
  				*devattr, const char *buf, size_t count)
  {
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1805
1806
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1807
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1808
1809
  	if (err)
  		return err;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1810

3fc7838a2   Hans de Goede   hwmon: (f71882fg)...
1811
1812
1813
  	/* Special case for F8000 pwm channel 3 which only does auto mode */
  	if (data->type == f8000 && nr == 2 && val != 2)
  		return -EINVAL;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1814
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1815
  	data->pwm_enable = f71882fg_read8(data, F71882FG_REG_PWM_ENABLE);
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1816
1817
1818
1819
1820
1821
1822
1823
1824
1825
1826
1827
1828
1829
1830
1831
  	/* Special case for F8000 auto PWM mode / Thermostat mode */
  	if (data->type == f8000 && ((data->pwm_enable >> 2 * nr) & 1)) {
  		switch (val) {
  		case 2:
  			data->pwm_enable &= ~(2 << (2 * nr));
  			break;		/* Normal auto mode */
  		case 3:
  			data->pwm_enable |= 2 << (2 * nr);
  			break;		/* Thermostat mode */
  		default:
  			count = -EINVAL;
  			goto leave;
  		}
  	} else {
  		switch (val) {
  		case 1:
09475d32e   Hans de Goede   hwmon: (f71882fg)...
1832
1833
1834
1835
1836
1837
  			/* The f71858fg does not support manual RPM mode */
  			if (data->type == f71858fg &&
  			    ((data->pwm_enable >> (2 * nr)) & 1)) {
  				count = -EINVAL;
  				goto leave;
  			}
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1838
1839
1840
1841
1842
1843
1844
1845
1846
  			data->pwm_enable |= 2 << (2 * nr);
  			break;		/* Manual */
  		case 2:
  			data->pwm_enable &= ~(2 << (2 * nr));
  			break;		/* Normal auto mode */
  		default:
  			count = -EINVAL;
  			goto leave;
  		}
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1847
  	}
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1848
  	f71882fg_write8(data, F71882FG_REG_PWM_ENABLE, data->pwm_enable);
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
1849
  leave:
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1850
1851
1852
1853
1854
1855
1856
1857
1858
1859
1860
1861
1862
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_pwm_auto_point_pwm(struct device *dev,
  				       struct device_attribute *devattr,
  				       char *buf)
  {
  	int result;
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int pwm = to_sensor_dev_attr_2(devattr)->index;
  	int point = to_sensor_dev_attr_2(devattr)->nr;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1863
  	mutex_lock(&data->update_lock);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1864
1865
1866
1867
1868
1869
1870
  	if (data->pwm_enable & (1 << (2 * pwm))) {
  		/* PWM mode */
  		result = data->pwm_auto_point_pwm[pwm][point];
  	} else {
  		/* RPM mode */
  		result = 32 * 255 / (32 + data->pwm_auto_point_pwm[pwm][point]);
  	}
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1871
  	mutex_unlock(&data->update_lock);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1872
1873
1874
1875
1876
1877
1878
1879
1880
  
  	return sprintf(buf, "%d
  ", result);
  }
  
  static ssize_t store_pwm_auto_point_pwm(struct device *dev,
  					struct device_attribute *devattr,
  					const char *buf, size_t count)
  {
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1881
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1882
  	int err, pwm = to_sensor_dev_attr_2(devattr)->index;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1883
  	int point = to_sensor_dev_attr_2(devattr)->nr;
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1884
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1885
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1886
1887
  	if (err)
  		return err;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1888
1889
1890
  	val = SENSORS_LIMIT(val, 0, 255);
  
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1891
  	data->pwm_enable = f71882fg_read8(data, F71882FG_REG_PWM_ENABLE);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1892
1893
1894
1895
1896
1897
1898
1899
1900
1901
1902
1903
1904
1905
1906
1907
1908
1909
1910
1911
1912
1913
1914
1915
1916
1917
  	if (data->pwm_enable & (1 << (2 * pwm))) {
  		/* PWM mode */
  	} else {
  		/* RPM mode */
  		if (val < 29)	/* Prevent negative numbers */
  			val = 255;
  		else
  			val = (255 - val) * 32 / val;
  	}
  	f71882fg_write8(data, F71882FG_REG_POINT_PWM(pwm, point), val);
  	data->pwm_auto_point_pwm[pwm][point] = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_pwm_auto_point_temp_hyst(struct device *dev,
  					     struct device_attribute *devattr,
  					     char *buf)
  {
  	int result = 0;
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int nr = to_sensor_dev_attr_2(devattr)->index;
  	int point = to_sensor_dev_attr_2(devattr)->nr;
  
  	mutex_lock(&data->update_lock);
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
1918
1919
1920
1921
  	if (nr & 1)
  		result = data->pwm_auto_point_hyst[nr / 2] >> 4;
  	else
  		result = data->pwm_auto_point_hyst[nr / 2] & 0x0f;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1922
1923
1924
1925
1926
1927
1928
1929
1930
1931
1932
  	result = 1000 * (data->pwm_auto_point_temp[nr][point] - result);
  	mutex_unlock(&data->update_lock);
  
  	return sprintf(buf, "%d
  ", result);
  }
  
  static ssize_t store_pwm_auto_point_temp_hyst(struct device *dev,
  					      struct device_attribute *devattr,
  					      const char *buf, size_t count)
  {
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1933
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1934
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1935
  	int point = to_sensor_dev_attr_2(devattr)->nr;
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
1936
  	u8 reg;
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1937
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1938
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1939
1940
1941
1942
  	if (err)
  		return err;
  
  	val /= 1000;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1943
1944
  
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1945
1946
  	data->pwm_auto_point_temp[nr][point] =
  		f71882fg_read8(data, F71882FG_REG_POINT_TEMP(nr, point));
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1947
1948
1949
  	val = SENSORS_LIMIT(val, data->pwm_auto_point_temp[nr][point] - 15,
  				data->pwm_auto_point_temp[nr][point]);
  	val = data->pwm_auto_point_temp[nr][point] - val;
bc27490f9   Hans de Goede   hwmon: (f71882fg)...
1950
1951
1952
1953
1954
1955
1956
1957
  	reg = f71882fg_read8(data, F71882FG_REG_FAN_HYST(nr / 2));
  	if (nr & 1)
  		reg = (reg & 0x0f) | (val << 4);
  	else
  		reg = (reg & 0xf0) | val;
  
  	f71882fg_write8(data, F71882FG_REG_FAN_HYST(nr / 2), reg);
  	data->pwm_auto_point_hyst[nr / 2] = reg;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1958
1959
1960
1961
1962
1963
1964
1965
1966
1967
1968
1969
1970
1971
1972
1973
1974
1975
1976
1977
1978
1979
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_pwm_interpolate(struct device *dev,
  				    struct device_attribute *devattr, char *buf)
  {
  	int result;
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int nr = to_sensor_dev_attr_2(devattr)->index;
  
  	result = (data->pwm_auto_point_mapping[nr] >> 4) & 1;
  
  	return sprintf(buf, "%d
  ", result);
  }
  
  static ssize_t store_pwm_interpolate(struct device *dev,
  				     struct device_attribute *devattr,
  				     const char *buf, size_t count)
  {
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1980
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1981
1982
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	unsigned long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
1983
  	err = kstrtoul(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
1984
1985
  	if (err)
  		return err;
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1986

9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1987
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
1988
1989
  	data->pwm_auto_point_mapping[nr] =
  		f71882fg_read8(data, F71882FG_REG_POINT_MAPPING(nr));
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
1990
1991
1992
1993
1994
1995
1996
1997
1998
1999
2000
2001
2002
2003
2004
2005
2006
2007
  	if (val)
  		val = data->pwm_auto_point_mapping[nr] | (1 << 4);
  	else
  		val = data->pwm_auto_point_mapping[nr] & (~(1 << 4));
  	f71882fg_write8(data, F71882FG_REG_POINT_MAPPING(nr), val);
  	data->pwm_auto_point_mapping[nr] = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_pwm_auto_point_channel(struct device *dev,
  					   struct device_attribute *devattr,
  					   char *buf)
  {
  	int result;
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int nr = to_sensor_dev_attr_2(devattr)->index;
09475d32e   Hans de Goede   hwmon: (f71882fg)...
2008
2009
  	result = 1 << ((data->pwm_auto_point_mapping[nr] & 3) -
  		       data->temp_start);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2010
2011
2012
2013
2014
2015
2016
2017
2018
  
  	return sprintf(buf, "%d
  ", result);
  }
  
  static ssize_t store_pwm_auto_point_channel(struct device *dev,
  					    struct device_attribute *devattr,
  					    const char *buf, size_t count)
  {
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
2019
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
2020
2021
  	int err, nr = to_sensor_dev_attr_2(devattr)->index;
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
2022
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
2023
2024
  	if (err)
  		return err;
304530186   Hans de Goede   hwmon: (f71882fg)...
2025

9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2026
2027
  	switch (val) {
  	case 1:
304530186   Hans de Goede   hwmon: (f71882fg)...
2028
  		val = 0;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2029
2030
  		break;
  	case 2:
304530186   Hans de Goede   hwmon: (f71882fg)...
2031
  		val = 1;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2032
2033
  		break;
  	case 4:
304530186   Hans de Goede   hwmon: (f71882fg)...
2034
  		val = 2;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2035
2036
2037
2038
  		break;
  	default:
  		return -EINVAL;
  	}
09475d32e   Hans de Goede   hwmon: (f71882fg)...
2039
  	val += data->temp_start;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2040
  	mutex_lock(&data->update_lock);
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
2041
2042
  	data->pwm_auto_point_mapping[nr] =
  		f71882fg_read8(data, F71882FG_REG_POINT_MAPPING(nr));
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2043
2044
2045
2046
2047
2048
2049
2050
2051
2052
2053
2054
2055
2056
2057
2058
2059
2060
2061
2062
2063
2064
2065
2066
2067
2068
  	val = (data->pwm_auto_point_mapping[nr] & 0xfc) | val;
  	f71882fg_write8(data, F71882FG_REG_POINT_MAPPING(nr), val);
  	data->pwm_auto_point_mapping[nr] = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
  
  static ssize_t show_pwm_auto_point_temp(struct device *dev,
  					struct device_attribute *devattr,
  					char *buf)
  {
  	int result;
  	struct f71882fg_data *data = f71882fg_update_device(dev);
  	int pwm = to_sensor_dev_attr_2(devattr)->index;
  	int point = to_sensor_dev_attr_2(devattr)->nr;
  
  	result = data->pwm_auto_point_temp[pwm][point];
  	return sprintf(buf, "%d
  ", 1000 * result);
  }
  
  static ssize_t store_pwm_auto_point_temp(struct device *dev,
  					 struct device_attribute *devattr,
  					 const char *buf, size_t count)
  {
ce0bfa5ee   Hans de Goede   hwmon: (f71882fg)...
2069
  	struct f71882fg_data *data = dev_get_drvdata(dev);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
2070
  	int err, pwm = to_sensor_dev_attr_2(devattr)->index;
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2071
  	int point = to_sensor_dev_attr_2(devattr)->nr;
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
2072
  	long val;
179c4fdb5   Frans Meulenbroeks   hwmon: replaced s...
2073
  	err = kstrtol(buf, 10, &val);
e8a4eacaa   Giel van Schijndel   hwmon: (f71882fg)...
2074
2075
2076
2077
  	if (err)
  		return err;
  
  	val /= 1000;
7669896f4   Hans de Goede   hwmon: (f71882fg)...
2078

98f7ba197   Hans de Goede   hwmon/f71882fg: O...
2079
  	if (data->auto_point_temp_signed)
7669896f4   Hans de Goede   hwmon: (f71882fg)...
2080
2081
2082
  		val = SENSORS_LIMIT(val, -128, 127);
  	else
  		val = SENSORS_LIMIT(val, 0, 127);
9ab796ebe   Mark van Doesburg   hwmon: (f71882fg)...
2083
2084
2085
2086
2087
2088
2089
2090
  
  	mutex_lock(&data->update_lock);
  	f71882fg_write8(data, F71882FG_REG_POINT_TEMP(pwm, point), val);
  	data->pwm_auto_point_temp[pwm][point] = val;
  	mutex_unlock(&data->update_lock);
  
  	return count;
  }
45fb36694   Hans de Goede   hwmon: add suppor...
2091
2092
2093
  static ssize_t show_name(struct device *dev, struct device_attribute *devattr,
  	char *buf)
  {
498be9683   Hans de Goede   hwmon: (f71882fg)...
2094
2095
2096
  	struct f71882fg_data *data = dev_get_drvdata(dev);
  	return sprintf(buf, "%s
  ", f71882fg_names[data->type]);
45fb36694   Hans de Goede   hwmon: add suppor...
2097
  }
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2098
2099
2100
2101
2102
2103
2104
2105
2106
2107
2108
2109
  static int __devinit f71882fg_create_sysfs_files(struct platform_device *pdev,
  	struct sensor_device_attribute_2 *attr, int count)
  {
  	int err, i;
  
  	for (i = 0; i < count; i++) {
  		err = device_create_file(&pdev->dev, &attr[i].dev_attr);
  		if (err)
  			return err;
  	}
  	return 0;
  }
45fb36694   Hans de Goede   hwmon: add suppor...
2110

fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2111
2112
2113
2114
2115
2116
2117
2118
  static void f71882fg_remove_sysfs_files(struct platform_device *pdev,
  	struct sensor_device_attribute_2 *attr, int count)
  {
  	int i;
  
  	for (i = 0; i < count; i++)
  		device_remove_file(&pdev->dev, &attr[i].dev_attr);
  }
9af0794c6   Hans de Goede   hwmon/f71882fg: A...
2119
  static int __devinit f71882fg_create_fan_sysfs_files(
6543439f1   Hans de Goede   hwmon/f71882fg: M...
2120
  	struct platform_device *pdev, int idx)
9af0794c6   Hans de Goede   hwmon/f71882fg: A...
2121
2122
2123
  {
  	struct f71882fg_data *data = platform_get_drvdata(pdev);
  	int err;
6543439f1   Hans de Goede   hwmon/f71882fg: M...
2124
2125
2126
2127
2128
2129
2130
2131
2132
2133
2134
2135
2136
2137
2138
2139
2140
2141
2142
2143
2144
2145
2146
2147
2148
2149
  	/* Sanity check the pwm setting */
  	err = 0;
  	switch (data->type) {
  	case f71858fg:
  		if (((data->pwm_enable >> (idx * 2)) & 3) == 3)
  			err = 1;
  		break;
  	case f71862fg:
  		if (((data->pwm_enable >> (idx * 2)) & 1) != 1)
  			err = 1;
  		break;
  	case f8000:
  		if (idx == 2)
  			err = data->pwm_enable & 0x20;
  		break;
  	default:
  		break;
  	}
  	if (err) {
  		dev_err(&pdev->dev,
  			"Invalid (reserved) pwm settings: 0x%02x, "
  			"skipping fan %d
  ",
  			(data->pwm_enable >> (idx * 2)) & 3, idx + 1);
  		return 0; /* This is a non fatal condition */
  	}
9af0794c6   Hans de Goede   hwmon/f71882fg: A...
2150
2151
2152
2153
2154
2155
2156
2157
2158
2159
2160
2161
  	err = f71882fg_create_sysfs_files(pdev, &fxxxx_fan_attr[idx][0],
  					  ARRAY_SIZE(fxxxx_fan_attr[0]));
  	if (err)
  		return err;
  
  	if (f71882fg_fan_has_beep[data->type]) {
  		err = f71882fg_create_sysfs_files(pdev,
  						  &fxxxx_fan_beep_attr[idx],
  						  1);
  		if (err)
  			return err;
  	}
6543439f1   Hans de Goede   hwmon/f71882fg: M...
2162
2163
2164
2165
2166
2167
2168
2169
2170
2171
2172
2173
2174
2175
2176
2177
2178
2179
2180
2181
2182
2183
2184
2185
2186
2187
2188
2189
  	dev_info(&pdev->dev, "Fan: %d is in %s mode
  ", idx + 1,
  		 (data->pwm_enable & (1 << (2 * idx))) ? "duty-cycle" : "RPM");
  
  	/* Check for unsupported auto pwm settings */
  	switch (data->type) {
  	case f71808e:
  	case f71808a:
  	case f71869:
  	case f71869a:
  	case f71889fg:
  	case f71889ed:
  	case f71889a:
  		data->pwm_auto_point_mapping[idx] =
  			f71882fg_read8(data, F71882FG_REG_POINT_MAPPING(idx));
  		if ((data->pwm_auto_point_mapping[idx] & 0x80) ||
  		    (data->pwm_auto_point_mapping[idx] & 3) == 0) {
  			dev_warn(&pdev->dev,
  				 "Auto pwm controlled by raw digital "
  				 "data, disabling pwm auto_point "
  				 "sysfs attributes for fan %d
  ", idx + 1);
  			return 0; /* This is a non fatal condition */
  		}
  		break;
  	default:
  		break;
  	}
9af0794c6   Hans de Goede   hwmon/f71882fg: A...
2190
2191
2192
2193
2194
2195
2196
2197
2198
2199
2200
2201
2202
2203
2204
2205
2206
2207
2208
2209
2210
2211
2212
2213
2214
2215
  
  	switch (data->type) {
  	case f71862fg:
  		err = f71882fg_create_sysfs_files(pdev,
  					&f71862fg_auto_pwm_attr[idx][0],
  					ARRAY_SIZE(f71862fg_auto_pwm_attr[0]));
  		break;
  	case f71808e:
  	case f71869:
  		err = f71882fg_create_sysfs_files(pdev,
  					&f71869_auto_pwm_attr[idx][0],
  					ARRAY_SIZE(f71869_auto_pwm_attr[0]));
  		break;
  	case f8000:
  		err = f71882fg_create_sysfs_files(pdev,
  					&f8000_auto_pwm_attr[idx][0],
  					ARRAY_SIZE(f8000_auto_pwm_attr[0]));
  		break;
  	default:
  		err = f71882fg_create_sysfs_files(pdev,
  					&fxxxx_auto_pwm_attr[idx][0],
  					ARRAY_SIZE(fxxxx_auto_pwm_attr[0]));
  	}
  
  	return err;
  }
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2216
  static int __devinit f71882fg_probe(struct platform_device *pdev)
45fb36694   Hans de Goede   hwmon: add suppor...
2217
2218
  {
  	struct f71882fg_data *data;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2219
  	struct f71882fg_sio_data *sio_data = pdev->dev.platform_data;
f27def07b   Jean Delvare   hwmon: (f71882fg)...
2220
2221
2222
  	int nr_fans = f71882fg_nr_fans[sio_data->type];
  	int nr_temps = f71882fg_nr_temps[sio_data->type];
  	int err, i;
98f7ba197   Hans de Goede   hwmon/f71882fg: O...
2223
  	u8 start_reg, reg;
45fb36694   Hans de Goede   hwmon: add suppor...
2224

c13548c53   Hans de Goede   hwmon: (f71882fg)...
2225
2226
  	data = kzalloc(sizeof(struct f71882fg_data), GFP_KERNEL);
  	if (!data)
45fb36694   Hans de Goede   hwmon: add suppor...
2227
2228
2229
  		return -ENOMEM;
  
  	data->addr = platform_get_resource(pdev, IORESOURCE_IO, 0)->start;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2230
  	data->type = sio_data->type;
09475d32e   Hans de Goede   hwmon: (f71882fg)...
2231
2232
  	data->temp_start =
  	    (data->type == f71858fg || data->type == f8000) ? 0 : 1;
45fb36694   Hans de Goede   hwmon: add suppor...
2233
2234
  	mutex_init(&data->update_lock);
  	platform_set_drvdata(pdev, data);
3cc74758a   Hans de Goede   hwmon: (f71882fg)...
2235
  	start_reg = f71882fg_read8(data, F71882FG_REG_START);
12d66e840   Hans de Goede   hwmon: (f71882fg)...
2236
2237
2238
2239
2240
2241
  	if (start_reg & 0x04) {
  		dev_warn(&pdev->dev, "Hardware monitor is powered down
  ");
  		err = -ENODEV;
  		goto exit_free;
  	}
3cc74758a   Hans de Goede   hwmon: (f71882fg)...
2242
2243
2244
2245
2246
2247
  	if (!(start_reg & 0x03)) {
  		dev_warn(&pdev->dev, "Hardware monitoring not activated
  ");
  		err = -ENODEV;
  		goto exit_free;
  	}
45fb36694   Hans de Goede   hwmon: add suppor...
2248
  	/* Register sysfs interface files */
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2249
2250
2251
  	err = device_create_file(&pdev->dev, &dev_attr_name);
  	if (err)
  		goto exit_unregister_sysfs;
45fb36694   Hans de Goede   hwmon: add suppor...
2252

45fb36694   Hans de Goede   hwmon: add suppor...
2253
  	if (start_reg & 0x01) {
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
2254
  		switch (data->type) {
09475d32e   Hans de Goede   hwmon: (f71882fg)...
2255
2256
2257
2258
2259
2260
2261
  		case f71858fg:
  			data->temp_config =
  				f71882fg_read8(data, F71882FG_REG_TEMP_CONFIG);
  			if (data->temp_config & 0x10)
  				/* The f71858fg temperature alarms behave as
  				   the f8000 alarms in this mode */
  				err = f71882fg_create_sysfs_files(pdev,
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2262
2263
  					f8000_temp_attr,
  					ARRAY_SIZE(f8000_temp_attr));
09475d32e   Hans de Goede   hwmon: (f71882fg)...
2264
2265
  			else
  				err = f71882fg_create_sysfs_files(pdev,
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2266
2267
  					f71858fg_temp_attr,
  					ARRAY_SIZE(f71858fg_temp_attr));
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
2268
2269
2270
  			break;
  		case f8000:
  			err = f71882fg_create_sysfs_files(pdev,
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2271
2272
  					f8000_temp_attr,
  					ARRAY_SIZE(f8000_temp_attr));
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
2273
  			break;
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2274
2275
  		default:
  			err = f71882fg_create_sysfs_files(pdev,
60d2b378b   Hans de Goede   hwmon/f71882fg: M...
2276
2277
  				&fxxxx_temp_attr[0][0],
  				ARRAY_SIZE(fxxxx_temp_attr[0]) * nr_temps);
498be9683   Hans de Goede   hwmon: (f71882fg)...
2278
  		}
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
2279
2280
  		if (err)
  			goto exit_unregister_sysfs;
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2281

4d53811a2   Hans de Goede   hwmon: (f71882fg)...
2282
  		if (f71882fg_temp_has_beep[data->type]) {
78aa4f72b   Hans de Goede   hwmon/f71882fg: S...
2283
2284
2285
2286
2287
2288
2289
  			err = f71882fg_create_sysfs_files(pdev,
  					&fxxxx_temp_beep_attr[0][0],
  					ARRAY_SIZE(fxxxx_temp_beep_attr[0])
  						* nr_temps);
  			if (err)
  				goto exit_unregister_sysfs;
  		}
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2290
2291
2292
2293
2294
2295
2296
2297
2298
2299
2300
2301
2302
2303
2304
  		for (i = 0; i < F71882FG_MAX_INS; i++) {
  			if (f71882fg_has_in[data->type][i]) {
  				err = device_create_file(&pdev->dev,
  						&fxxxx_in_attr[i].dev_attr);
  				if (err)
  					goto exit_unregister_sysfs;
  			}
  		}
  		if (f71882fg_has_in1_alarm[data->type]) {
  			err = f71882fg_create_sysfs_files(pdev,
  					fxxxx_in1_alarm_attr,
  					ARRAY_SIZE(fxxxx_in1_alarm_attr));
  			if (err)
  				goto exit_unregister_sysfs;
  		}
45fb36694   Hans de Goede   hwmon: add suppor...
2305
2306
2307
  	}
  
  	if (start_reg & 0x02) {
98f7ba197   Hans de Goede   hwmon/f71882fg: O...
2308
  		switch (data->type) {
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
2309
  		case f71808e:
629c58bac   Hans de Goede   hwmon: (f71882fg)...
2310
  		case f71808a:
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
2311
  		case f71869:
5da556e33   Hans de Goede   hwmon: (f71882fg)...
2312
  		case f71869a:
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
2313
  			/* These always have signed auto point temps */
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
2314
2315
  			data->auto_point_temp_signed = 1;
  			/* Fall through to select correct fan/pwm reg bank! */
98f7ba197   Hans de Goede   hwmon/f71882fg: O...
2316
  		case f71889fg:
3cad40228   Hans de Goede   hwmon/f71882fg: A...
2317
  		case f71889ed:
a66c10887   Hans de Goede   hwmon: (f71882fg)...
2318
  		case f71889a:
98f7ba197   Hans de Goede   hwmon/f71882fg: O...
2319
2320
2321
  			reg = f71882fg_read8(data, F71882FG_REG_FAN_FAULT_T);
  			if (reg & F71882FG_FAN_NEG_TEMP_EN)
  				data->auto_point_temp_signed = 1;
3cad40228   Hans de Goede   hwmon/f71882fg: A...
2322
2323
2324
  			/* Ensure banked pwm registers point to right bank */
  			reg &= ~F71882FG_FAN_PROG_SEL;
  			f71882fg_write8(data, F71882FG_REG_FAN_FAULT_T, reg);
98f7ba197   Hans de Goede   hwmon/f71882fg: O...
2325
2326
2327
2328
  			break;
  		default:
  			break;
  		}
996cadb2b   Hans de Goede   hwmon: (f71882fg)...
2329
2330
  		data->pwm_enable =
  			f71882fg_read8(data, F71882FG_REG_PWM_ENABLE);
9af0794c6   Hans de Goede   hwmon/f71882fg: A...
2331
  		for (i = 0; i < nr_fans; i++) {
6543439f1   Hans de Goede   hwmon/f71882fg: M...
2332
  			err = f71882fg_create_fan_sysfs_files(pdev, i);
629c58bac   Hans de Goede   hwmon: (f71882fg)...
2333
2334
  			if (err)
  				goto exit_unregister_sysfs;
9af0794c6   Hans de Goede   hwmon/f71882fg: A...
2335
2336
2337
2338
2339
  		}
  
  		/* Some types have 1 extra fan with limited functionality */
  		switch (data->type) {
  		case f71808a:
629c58bac   Hans de Goede   hwmon: (f71882fg)...
2340
2341
2342
2343
  			err = f71882fg_create_sysfs_files(pdev,
  					f71808a_fan3_attr,
  					ARRAY_SIZE(f71808a_fan3_attr));
  			break;
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
2344
2345
2346
2347
2348
  		case f8000:
  			err = f71882fg_create_sysfs_files(pdev,
  					f8000_fan_attr,
  					ARRAY_SIZE(f8000_fan_attr));
  			break;
e48a7f1a7   Hans de Goede   hwmon/f71882fg: B...
2349
  		default:
9af0794c6   Hans de Goede   hwmon/f71882fg: A...
2350
  			break;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2351
  		}
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2352
2353
  		if (err)
  			goto exit_unregister_sysfs;
45fb36694   Hans de Goede   hwmon: add suppor...
2354
  	}
1beeffe43   Tony Jones   hwmon: Convert fr...
2355
2356
2357
  	data->hwmon_dev = hwmon_device_register(&pdev->dev);
  	if (IS_ERR(data->hwmon_dev)) {
  		err = PTR_ERR(data->hwmon_dev);
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2358
  		data->hwmon_dev = NULL;
45fb36694   Hans de Goede   hwmon: add suppor...
2359
2360
2361
2362
2363
2364
  		goto exit_unregister_sysfs;
  	}
  
  	return 0;
  
  exit_unregister_sysfs:
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2365
  	f71882fg_remove(pdev); /* Will unregister the sysfs files for us */
3cc74758a   Hans de Goede   hwmon: (f71882fg)...
2366
2367
2368
  	return err; /* f71882fg_remove() also frees our data */
  exit_free:
  	kfree(data);
45fb36694   Hans de Goede   hwmon: add suppor...
2369
2370
  	return err;
  }
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2371
  static int f71882fg_remove(struct platform_device *pdev)
45fb36694   Hans de Goede   hwmon: add suppor...
2372
  {
45fb36694   Hans de Goede   hwmon: add suppor...
2373
  	struct f71882fg_data *data = platform_get_drvdata(pdev);
f27def07b   Jean Delvare   hwmon: (f71882fg)...
2374
2375
2376
  	int nr_fans = f71882fg_nr_fans[data->type];
  	int nr_temps = f71882fg_nr_temps[data->type];
  	int i;
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2377
  	u8 start_reg = f71882fg_read8(data, F71882FG_REG_START);
45fb36694   Hans de Goede   hwmon: add suppor...
2378

c13548c53   Hans de Goede   hwmon: (f71882fg)...
2379
2380
  	if (data->hwmon_dev)
  		hwmon_device_unregister(data->hwmon_dev);
45fb36694   Hans de Goede   hwmon: add suppor...
2381

c13548c53   Hans de Goede   hwmon: (f71882fg)...
2382
  	device_remove_file(&pdev->dev, &dev_attr_name);
45fb36694   Hans de Goede   hwmon: add suppor...
2383

fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2384
2385
2386
2387
2388
  	if (start_reg & 0x01) {
  		switch (data->type) {
  		case f71858fg:
  			if (data->temp_config & 0x10)
  				f71882fg_remove_sysfs_files(pdev,
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2389
2390
  					f8000_temp_attr,
  					ARRAY_SIZE(f8000_temp_attr));
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2391
2392
  			else
  				f71882fg_remove_sysfs_files(pdev,
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2393
2394
  					f71858fg_temp_attr,
  					ARRAY_SIZE(f71858fg_temp_attr));
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2395
2396
2397
  			break;
  		case f8000:
  			f71882fg_remove_sysfs_files(pdev,
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2398
2399
  					f8000_temp_attr,
  					ARRAY_SIZE(f8000_temp_attr));
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2400
  			break;
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2401
2402
  		default:
  			f71882fg_remove_sysfs_files(pdev,
60d2b378b   Hans de Goede   hwmon/f71882fg: M...
2403
2404
  				&fxxxx_temp_attr[0][0],
  				ARRAY_SIZE(fxxxx_temp_attr[0]) * nr_temps);
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2405
  		}
4d53811a2   Hans de Goede   hwmon: (f71882fg)...
2406
  		if (f71882fg_temp_has_beep[data->type]) {
78aa4f72b   Hans de Goede   hwmon/f71882fg: S...
2407
2408
2409
2410
  			f71882fg_remove_sysfs_files(pdev,
  			       &fxxxx_temp_beep_attr[0][0],
  			       ARRAY_SIZE(fxxxx_temp_beep_attr[0]) * nr_temps);
  		}
0bae64008   Hans de Goede   hwmon/f71882fg: M...
2411
2412
2413
2414
2415
2416
2417
2418
2419
2420
  		for (i = 0; i < F71882FG_MAX_INS; i++) {
  			if (f71882fg_has_in[data->type][i]) {
  				device_remove_file(&pdev->dev,
  						&fxxxx_in_attr[i].dev_attr);
  			}
  		}
  		if (f71882fg_has_in1_alarm[data->type]) {
  			f71882fg_remove_sysfs_files(pdev,
  					fxxxx_in1_alarm_attr,
  					ARRAY_SIZE(fxxxx_in1_alarm_attr));
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2421
2422
  		}
  	}
498be9683   Hans de Goede   hwmon: (f71882fg)...
2423

fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2424
2425
2426
  	if (start_reg & 0x02) {
  		f71882fg_remove_sysfs_files(pdev, &fxxxx_fan_attr[0][0],
  				ARRAY_SIZE(fxxxx_fan_attr[0]) * nr_fans);
45fb36694   Hans de Goede   hwmon: add suppor...
2427

4d53811a2   Hans de Goede   hwmon: (f71882fg)...
2428
  		if (f71882fg_fan_has_beep[data->type]) {
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2429
2430
  			f71882fg_remove_sysfs_files(pdev,
  					fxxxx_fan_beep_attr, nr_fans);
78aa4f72b   Hans de Goede   hwmon/f71882fg: S...
2431
  		}
66344aa6a   Hans de Goede   hwmon: (f71882fg)...
2432

fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2433
  		switch (data->type) {
629c58bac   Hans de Goede   hwmon: (f71882fg)...
2434
2435
2436
2437
2438
2439
2440
2441
  		case f71808a:
  			f71882fg_remove_sysfs_files(pdev,
  				&fxxxx_auto_pwm_attr[0][0],
  				ARRAY_SIZE(fxxxx_auto_pwm_attr[0]) * nr_fans);
  			f71882fg_remove_sysfs_files(pdev,
  					f71808a_fan3_attr,
  					ARRAY_SIZE(f71808a_fan3_attr));
  			break;
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2442
2443
  		case f71862fg:
  			f71882fg_remove_sysfs_files(pdev,
558401425   Hans de Goede   hwmon/f71882fg: M...
2444
2445
2446
  				&f71862fg_auto_pwm_attr[0][0],
  				ARRAY_SIZE(f71862fg_auto_pwm_attr[0]) *
  					nr_fans);
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2447
  			break;
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
2448
  		case f71808e:
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
2449
2450
  		case f71869:
  			f71882fg_remove_sysfs_files(pdev,
558401425   Hans de Goede   hwmon/f71882fg: M...
2451
2452
  				&f71869_auto_pwm_attr[0][0],
  				ARRAY_SIZE(f71869_auto_pwm_attr[0]) * nr_fans);
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
2453
  			break;
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2454
2455
2456
2457
2458
  		case f8000:
  			f71882fg_remove_sysfs_files(pdev,
  					f8000_fan_attr,
  					ARRAY_SIZE(f8000_fan_attr));
  			f71882fg_remove_sysfs_files(pdev,
558401425   Hans de Goede   hwmon/f71882fg: M...
2459
2460
  				&f8000_auto_pwm_attr[0][0],
  				ARRAY_SIZE(f8000_auto_pwm_attr[0]) * nr_fans);
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2461
  			break;
3cad40228   Hans de Goede   hwmon/f71882fg: A...
2462
  		default:
fc16c56e6   Hans de Goede   hwmon: (f71882fg)...
2463
2464
2465
2466
2467
  			f71882fg_remove_sysfs_files(pdev,
  				&fxxxx_auto_pwm_attr[0][0],
  				ARRAY_SIZE(fxxxx_auto_pwm_attr[0]) * nr_fans);
  		}
  	}
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
2468

d9ebaa454   Hans de Goede   hwmon/f71882fg: S...
2469
  	platform_set_drvdata(pdev, NULL);
45fb36694   Hans de Goede   hwmon: add suppor...
2470
2471
2472
2473
  	kfree(data);
  
  	return 0;
  }
498be9683   Hans de Goede   hwmon: (f71882fg)...
2474
2475
  static int __init f71882fg_find(int sioaddr, unsigned short *address,
  	struct f71882fg_sio_data *sio_data)
45fb36694   Hans de Goede   hwmon: add suppor...
2476
  {
45fb36694   Hans de Goede   hwmon: add suppor...
2477
  	u16 devid;
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
2478
2479
2480
  	int err = superio_enter(sioaddr);
  	if (err)
  		return err;
45fb36694   Hans de Goede   hwmon: add suppor...
2481
2482
2483
  
  	devid = superio_inw(sioaddr, SIO_REG_MANID);
  	if (devid != SIO_FINTEK_ID) {
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2484
2485
  		pr_debug("Not a Fintek device
  ");
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
2486
  		err = -ENODEV;
45fb36694   Hans de Goede   hwmon: add suppor...
2487
2488
  		goto exit;
  	}
67b671bce   Jean Delvare   hwmon: Let the us...
2489
  	devid = force_id ? force_id : superio_inw(sioaddr, SIO_REG_DEVID);
498be9683   Hans de Goede   hwmon: (f71882fg)...
2490
  	switch (devid) {
e5e713cbe   Hans de Goede   hwmon/f71882fg: A...
2491
2492
2493
  	case SIO_F71808E_ID:
  		sio_data->type = f71808e;
  		break;
629c58bac   Hans de Goede   hwmon: (f71882fg)...
2494
2495
2496
  	case SIO_F71808A_ID:
  		sio_data->type = f71808a;
  		break;
09475d32e   Hans de Goede   hwmon: (f71882fg)...
2497
2498
2499
  	case SIO_F71858_ID:
  		sio_data->type = f71858fg;
  		break;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2500
2501
2502
  	case SIO_F71862_ID:
  		sio_data->type = f71862fg;
  		break;
c11bb9936   Hans de Goede   hwmon/f71882fg: A...
2503
2504
2505
  	case SIO_F71869_ID:
  		sio_data->type = f71869;
  		break;
5da556e33   Hans de Goede   hwmon: (f71882fg)...
2506
2507
2508
  	case SIO_F71869A_ID:
  		sio_data->type = f71869a;
  		break;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2509
2510
2511
  	case SIO_F71882_ID:
  		sio_data->type = f71882fg;
  		break;
7669896f4   Hans de Goede   hwmon: (f71882fg)...
2512
2513
2514
  	case SIO_F71889_ID:
  		sio_data->type = f71889fg;
  		break;
3cad40228   Hans de Goede   hwmon/f71882fg: A...
2515
2516
2517
  	case SIO_F71889E_ID:
  		sio_data->type = f71889ed;
  		break;
a66c10887   Hans de Goede   hwmon: (f71882fg)...
2518
2519
2520
  	case SIO_F71889A_ID:
  		sio_data->type = f71889a;
  		break;
ed4f7c20b   Hans de Goede   hwmon: (f71882fg)...
2521
2522
2523
  	case SIO_F8000_ID:
  		sio_data->type = f8000;
  		break;
383586b12   Jean Delvare   hwmon: (f71882fg)...
2524
2525
2526
  	case SIO_F81865_ID:
  		sio_data->type = f81865f;
  		break;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2527
  	default:
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2528
2529
2530
  		pr_info("Unsupported Fintek device: %04x
  ",
  			(unsigned int)devid);
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
2531
  		err = -ENODEV;
45fb36694   Hans de Goede   hwmon: add suppor...
2532
2533
  		goto exit;
  	}
09475d32e   Hans de Goede   hwmon: (f71882fg)...
2534
2535
2536
2537
  	if (sio_data->type == f71858fg)
  		superio_select(sioaddr, SIO_F71858FG_LD_HWM);
  	else
  		superio_select(sioaddr, SIO_F71882FG_LD_HWM);
8afb10490   Mark M. Hoffman   hwmon: (f71882fg)...
2538
  	if (!(superio_inb(sioaddr, SIO_REG_ENABLE) & 0x01)) {
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2539
2540
  		pr_warn("Device not activated
  ");
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
2541
  		err = -ENODEV;
45fb36694   Hans de Goede   hwmon: add suppor...
2542
2543
2544
2545
  		goto exit;
  	}
  
  	*address = superio_inw(sioaddr, SIO_REG_ADDR);
162bb59e4   Giel van Schijndel   hwmon: (f71882fg)...
2546
  	if (*address == 0) {
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2547
2548
  		pr_warn("Base address not set
  ");
cadb86570   Giel van Schijndel   hwmon: f71882fg: ...
2549
  		err = -ENODEV;
45fb36694   Hans de Goede   hwmon: add suppor...
2550
2551
2552
  		goto exit;
  	}
  	*address &= ~(REGION_LENGTH - 1);	/* Ignore 3 LSB */
45fb36694   Hans de Goede   hwmon: add suppor...
2553
  	err = 0;
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2554
2555
  	pr_info("Found %s chip at %#x, revision %d
  ",
498be9683   Hans de Goede   hwmon: (f71882fg)...
2556
  		f71882fg_names[sio_data->type],	(unsigned int)*address,
45fb36694   Hans de Goede   hwmon: add suppor...
2557
2558
2559
2560
2561
  		(int)superio_inb(sioaddr, SIO_REG_DEVREV));
  exit:
  	superio_exit(sioaddr);
  	return err;
  }
498be9683   Hans de Goede   hwmon: (f71882fg)...
2562
2563
  static int __init f71882fg_device_add(unsigned short address,
  	const struct f71882fg_sio_data *sio_data)
45fb36694   Hans de Goede   hwmon: add suppor...
2564
2565
2566
2567
2568
2569
2570
2571
2572
  {
  	struct resource res = {
  		.start	= address,
  		.end	= address + REGION_LENGTH - 1,
  		.flags	= IORESOURCE_IO,
  	};
  	int err;
  
  	f71882fg_pdev = platform_device_alloc(DRVNAME, address);
8afb10490   Mark M. Hoffman   hwmon: (f71882fg)...
2573
  	if (!f71882fg_pdev)
45fb36694   Hans de Goede   hwmon: add suppor...
2574
2575
2576
  		return -ENOMEM;
  
  	res.name = f71882fg_pdev->name;
b9acb64a3   Jean Delvare   hwmon: Check for ...
2577
2578
  	err = acpi_check_resource_conflict(&res);
  	if (err)
18632f84f   Hans de Goede   hwmon: Fix ACPI r...
2579
  		goto exit_device_put;
b9acb64a3   Jean Delvare   hwmon: Check for ...
2580

45fb36694   Hans de Goede   hwmon: add suppor...
2581
  	err = platform_device_add_resources(f71882fg_pdev, &res, 1);
8afb10490   Mark M. Hoffman   hwmon: (f71882fg)...
2582
  	if (err) {
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2583
2584
  		pr_err("Device resource addition failed
  ");
45fb36694   Hans de Goede   hwmon: add suppor...
2585
2586
  		goto exit_device_put;
  	}
498be9683   Hans de Goede   hwmon: (f71882fg)...
2587
2588
2589
  	err = platform_device_add_data(f71882fg_pdev, sio_data,
  				       sizeof(struct f71882fg_sio_data));
  	if (err) {
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2590
2591
  		pr_err("Platform data allocation failed
  ");
498be9683   Hans de Goede   hwmon: (f71882fg)...
2592
2593
  		goto exit_device_put;
  	}
45fb36694   Hans de Goede   hwmon: add suppor...
2594
  	err = platform_device_add(f71882fg_pdev);
8afb10490   Mark M. Hoffman   hwmon: (f71882fg)...
2595
  	if (err) {
22d3b4128   Joe Perches   hwmon: (f71882fg)...
2596
2597
  		pr_err("Device addition failed
  ");
45fb36694   Hans de Goede   hwmon: add suppor...
2598
2599
2600
2601
2602
2603
2604
2605
2606
2607
2608
2609
2610
2611
2612
  		goto exit_device_put;
  	}
  
  	return 0;
  
  exit_device_put:
  	platform_device_put(f71882fg_pdev);
  
  	return err;
  }
  
  static int __init f71882fg_init(void)
  {
  	int err = -ENODEV;
  	unsigned short address;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2613
2614
2615
  	struct f71882fg_sio_data sio_data;
  
  	memset(&sio_data, 0, sizeof(sio_data));
45fb36694   Hans de Goede   hwmon: add suppor...
2616

498be9683   Hans de Goede   hwmon: (f71882fg)...
2617
2618
  	if (f71882fg_find(0x2e, &address, &sio_data) &&
  	    f71882fg_find(0x4e, &address, &sio_data))
45fb36694   Hans de Goede   hwmon: add suppor...
2619
  		goto exit;
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2620
2621
  	err = platform_driver_register(&f71882fg_driver);
  	if (err)
45fb36694   Hans de Goede   hwmon: add suppor...
2622
  		goto exit;
498be9683   Hans de Goede   hwmon: (f71882fg)...
2623
  	err = f71882fg_device_add(address, &sio_data);
c13548c53   Hans de Goede   hwmon: (f71882fg)...
2624
  	if (err)
45fb36694   Hans de Goede   hwmon: add suppor...
2625
2626
2627
2628
2629
2630
2631
2632
2633
2634
2635
2636
2637
2638
2639
2640
2641
  		goto exit_driver;
  
  	return 0;
  
  exit_driver:
  	platform_driver_unregister(&f71882fg_driver);
  exit:
  	return err;
  }
  
  static void __exit f71882fg_exit(void)
  {
  	platform_device_unregister(f71882fg_pdev);
  	platform_driver_unregister(&f71882fg_driver);
  }
  
  MODULE_DESCRIPTION("F71882FG Hardware Monitoring Driver");
7958e3b45   Hans de Goede   hwmon: Use <> rat...
2642
  MODULE_AUTHOR("Hans Edgington, Hans de Goede <hdegoede@redhat.com>");
45fb36694   Hans de Goede   hwmon: add suppor...
2643
2644
2645
2646
  MODULE_LICENSE("GPL");
  
  module_init(f71882fg_init);
  module_exit(f71882fg_exit);