Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
77eb5b37
Commit
77eb5b37
authored
Dec 04, 2012
by
Guenter Roeck
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
hwmon: (nct6775) Add support for pwm, pwm_mode, and pwm_enable
Signed-off-by:
Guenter Roeck
<
linux@roeck-us.net
>
parent
84d19d92
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
347 additions
and
0 deletions
+347
-0
Documentation/hwmon/nct6775
Documentation/hwmon/nct6775
+18
-0
drivers/hwmon/nct6775.c
drivers/hwmon/nct6775.c
+329
-0
No files found.
Documentation/hwmon/nct6775
View file @
77eb5b37
...
@@ -69,6 +69,24 @@ is driven slower/faster to reach the predefined range again.
...
@@ -69,6 +69,24 @@ is driven slower/faster to reach the predefined range again.
The mode works for fan1-fan5.
The mode works for fan1-fan5.
sysfs attributes
----------------
pwm[1-5] - this file stores PWM duty cycle or DC value (fan speed) in range:
0 (lowest speed) to 255 (full)
pwm[1-5]_enable - this file controls mode of fan/temperature control:
* 0 Fan control disabled (fans set to maximum speed)
* 1 Manual mode, write to pwm[0-5] any value 0-255
* 2 "Thermal Cruise" mode
* 3 "Fan Speed Cruise" mode
* 4 "Smart Fan III" mode (NCT6775F only)
* 5 "Smart Fan IV" mode
pwm[1-5]_mode - controls if output is PWM or DC level
* 0 DC output
* 1 PWM output
Usage Notes
Usage Notes
-----------
-----------
...
...
drivers/hwmon/nct6775.c
View file @
77eb5b37
...
@@ -96,6 +96,8 @@ MODULE_PARM_DESC(fan_debounce, "Enable debouncing for fan RPM signal");
...
@@ -96,6 +96,8 @@ MODULE_PARM_DESC(fan_debounce, "Enable debouncing for fan RPM signal");
#define SIO_NCT6779_ID 0xc560
#define SIO_NCT6779_ID 0xc560
#define SIO_ID_MASK 0xFFF0
#define SIO_ID_MASK 0xFFF0
enum
pwm_enable
{
off
,
manual
,
thermal_cruise
,
speed_cruise
,
sf3
,
sf4
};
static
inline
void
static
inline
void
superio_outb
(
int
ioreg
,
int
reg
,
int
val
)
superio_outb
(
int
ioreg
,
int
reg
,
int
val
)
{
{
...
@@ -209,6 +211,15 @@ static const s8 NCT6775_ALARM_BITS[] = {
...
@@ -209,6 +211,15 @@ static const s8 NCT6775_ALARM_BITS[] = {
static
const
u8
NCT6775_REG_CR_CASEOPEN_CLR
[]
=
{
0xe6
,
0xee
};
static
const
u8
NCT6775_REG_CR_CASEOPEN_CLR
[]
=
{
0xe6
,
0xee
};
static
const
u8
NCT6775_CR_CASEOPEN_CLR_MASK
[]
=
{
0x20
,
0x01
};
static
const
u8
NCT6775_CR_CASEOPEN_CLR_MASK
[]
=
{
0x20
,
0x01
};
/* DC or PWM output fan configuration */
static
const
u8
NCT6775_REG_PWM_MODE
[]
=
{
0x04
,
0x04
,
0x12
};
static
const
u8
NCT6775_PWM_MODE_MASK
[]
=
{
0x01
,
0x02
,
0x01
};
static
const
u16
NCT6775_REG_FAN_MODE
[]
=
{
0x102
,
0x202
,
0x302
,
0x802
,
0x902
};
static
const
u16
NCT6775_REG_PWM
[]
=
{
0x109
,
0x209
,
0x309
,
0x809
,
0x909
};
static
const
u16
NCT6775_REG_PWM_READ
[]
=
{
0x01
,
0x03
,
0x11
,
0x13
,
0x15
};
static
const
u16
NCT6775_REG_FAN
[]
=
{
0x630
,
0x632
,
0x634
,
0x636
,
0x638
};
static
const
u16
NCT6775_REG_FAN
[]
=
{
0x630
,
0x632
,
0x634
,
0x636
,
0x638
};
static
const
u16
NCT6775_REG_FAN_MIN
[]
=
{
0x3b
,
0x3c
,
0x3d
};
static
const
u16
NCT6775_REG_FAN_MIN
[]
=
{
0x3b
,
0x3c
,
0x3d
};
static
const
u16
NCT6775_REG_FAN_PULSES
[]
=
{
0x641
,
0x642
,
0x643
,
0x644
,
0
};
static
const
u16
NCT6775_REG_FAN_PULSES
[]
=
{
0x641
,
0x642
,
0x643
,
0x644
,
0
};
...
@@ -270,6 +281,9 @@ static const s8 NCT6776_ALARM_BITS[] = {
...
@@ -270,6 +281,9 @@ static const s8 NCT6776_ALARM_BITS[] = {
4
,
5
,
13
,
-
1
,
-
1
,
-
1
,
/* temp1..temp6 */
4
,
5
,
13
,
-
1
,
-
1
,
-
1
,
/* temp1..temp6 */
12
,
9
};
/* intrusion0, intrusion1 */
12
,
9
};
/* intrusion0, intrusion1 */
static
const
u8
NCT6776_REG_PWM_MODE
[]
=
{
0x04
,
0
,
0
};
static
const
u8
NCT6776_PWM_MODE_MASK
[]
=
{
0x01
,
0
,
0
};
static
const
u16
NCT6776_REG_FAN_MIN
[]
=
{
0x63a
,
0x63c
,
0x63e
,
0x640
,
0x642
};
static
const
u16
NCT6776_REG_FAN_MIN
[]
=
{
0x63a
,
0x63c
,
0x63e
,
0x640
,
0x642
};
static
const
u16
NCT6776_REG_FAN_PULSES
[]
=
{
0x644
,
0x645
,
0x646
,
0
,
0
};
static
const
u16
NCT6776_REG_FAN_PULSES
[]
=
{
0x644
,
0x645
,
0x646
,
0
,
0
};
...
@@ -380,6 +394,20 @@ static const u16 NCT6779_REG_TEMP_ALTERNATE[ARRAY_SIZE(nct6779_temp_label) - 1]
...
@@ -380,6 +394,20 @@ static const u16 NCT6779_REG_TEMP_ALTERNATE[ARRAY_SIZE(nct6779_temp_label) - 1]
static
const
u16
NCT6779_REG_TEMP_CRIT
[
ARRAY_SIZE
(
nct6779_temp_label
)
-
1
]
static
const
u16
NCT6779_REG_TEMP_CRIT
[
ARRAY_SIZE
(
nct6779_temp_label
)
-
1
]
=
{
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0x709
,
0x70a
};
=
{
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0
,
0x709
,
0x70a
};
static
enum
pwm_enable
reg_to_pwm_enable
(
int
pwm
,
int
mode
)
{
if
(
mode
==
0
&&
pwm
==
255
)
return
off
;
return
mode
+
1
;
}
static
int
pwm_enable_to_reg
(
enum
pwm_enable
mode
)
{
if
(
mode
==
off
)
return
0
;
return
mode
-
1
;
}
/*
/*
* Conversions
* Conversions
*/
*/
...
@@ -471,9 +499,16 @@ struct nct6775_data {
...
@@ -471,9 +499,16 @@ struct nct6775_data {
const
u16
*
REG_IN_MINMAX
[
2
];
const
u16
*
REG_IN_MINMAX
[
2
];
const
u16
*
REG_FAN
;
const
u16
*
REG_FAN
;
const
u16
*
REG_FAN_MODE
;
const
u16
*
REG_FAN_MIN
;
const
u16
*
REG_FAN_MIN
;
const
u16
*
REG_FAN_PULSES
;
const
u16
*
REG_FAN_PULSES
;
const
u8
*
REG_PWM_MODE
;
const
u8
*
PWM_MODE_MASK
;
const
u16
*
REG_PWM
[
1
];
/* [0]=pwm */
const
u16
*
REG_PWM_READ
;
const
u16
*
REG_TEMP_SOURCE
;
/* temp register sources */
const
u16
*
REG_TEMP_SOURCE
;
/* temp register sources */
const
u16
*
REG_TEMP_OFFSET
;
const
u16
*
REG_TEMP_OFFSET
;
...
@@ -494,6 +529,7 @@ struct nct6775_data {
...
@@ -494,6 +529,7 @@ struct nct6775_data {
u16
fan_min
[
5
];
u16
fan_min
[
5
];
u8
fan_pulses
[
5
];
u8
fan_pulses
[
5
];
u8
fan_div
[
5
];
u8
fan_div
[
5
];
u8
has_pwm
;
u8
has_fan
;
/* some fan inputs can be disabled */
u8
has_fan
;
/* some fan inputs can be disabled */
u8
has_fan_min
;
/* some fans don't have min register */
u8
has_fan_min
;
/* some fans don't have min register */
bool
has_fan_div
;
bool
has_fan_div
;
...
@@ -505,6 +541,18 @@ struct nct6775_data {
...
@@ -505,6 +541,18 @@ struct nct6775_data {
* 3=temp_crit */
* 3=temp_crit */
u64
alarms
;
u64
alarms
;
u8
pwm_num
;
/* number of pwm */
u8
pwm_mode
[
5
];
/* 1->DC variable voltage, 0->PWM variable duty cycle */
enum
pwm_enable
pwm_enable
[
5
];
/* 0->off
* 1->manual
* 2->thermal cruise mode (also called SmartFan I)
* 3->fan speed cruise mode
* 4->SmartFan III
* 5->enhanced variable thermal cruise (SmartFan IV)
*/
u8
pwm
[
1
][
5
];
/* [0]=pwm */
u8
vid
;
u8
vid
;
u8
vrm
;
u8
vrm
;
...
@@ -781,6 +829,36 @@ static void nct6775_select_fan_div(struct device *dev,
...
@@ -781,6 +829,36 @@ static void nct6775_select_fan_div(struct device *dev,
}
}
}
}
static
void
nct6775_update_pwm
(
struct
device
*
dev
)
{
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
int
i
,
j
;
int
fanmodecfg
;
bool
duty_is_dc
;
for
(
i
=
0
;
i
<
data
->
pwm_num
;
i
++
)
{
if
(
!
(
data
->
has_pwm
&
(
1
<<
i
)))
continue
;
duty_is_dc
=
data
->
REG_PWM_MODE
[
i
]
&&
(
nct6775_read_value
(
data
,
data
->
REG_PWM_MODE
[
i
])
&
data
->
PWM_MODE_MASK
[
i
]);
data
->
pwm_mode
[
i
]
=
duty_is_dc
;
fanmodecfg
=
nct6775_read_value
(
data
,
data
->
REG_FAN_MODE
[
i
]);
for
(
j
=
0
;
j
<
ARRAY_SIZE
(
data
->
REG_PWM
);
j
++
)
{
if
(
data
->
REG_PWM
[
j
]
&&
data
->
REG_PWM
[
j
][
i
])
{
data
->
pwm
[
j
][
i
]
=
nct6775_read_value
(
data
,
data
->
REG_PWM
[
j
][
i
]);
}
}
data
->
pwm_enable
[
i
]
=
reg_to_pwm_enable
(
data
->
pwm
[
0
][
i
],
(
fanmodecfg
>>
4
)
&
7
);
}
}
static
struct
nct6775_data
*
nct6775_update_device
(
struct
device
*
dev
)
static
struct
nct6775_data
*
nct6775_update_device
(
struct
device
*
dev
)
{
{
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
...
@@ -826,6 +904,8 @@ static struct nct6775_data *nct6775_update_device(struct device *dev)
...
@@ -826,6 +904,8 @@ static struct nct6775_data *nct6775_update_device(struct device *dev)
nct6775_select_fan_div
(
dev
,
data
,
i
,
reg
);
nct6775_select_fan_div
(
dev
,
data
,
i
,
reg
);
}
}
nct6775_update_pwm
(
dev
);
/* Measured temperatures and limits */
/* Measured temperatures and limits */
for
(
i
=
0
;
i
<
NUM_TEMP
;
i
++
)
{
for
(
i
=
0
;
i
<
NUM_TEMP
;
i
++
)
{
if
(
!
(
data
->
have_temp
&
(
1
<<
i
)))
if
(
!
(
data
->
have_temp
&
(
1
<<
i
)))
...
@@ -1599,6 +1679,170 @@ static struct sensor_device_attribute sda_temp_alarm[] = {
...
@@ -1599,6 +1679,170 @@ static struct sensor_device_attribute sda_temp_alarm[] = {
#define NUM_TEMP_ALARM ARRAY_SIZE(sda_temp_alarm)
#define NUM_TEMP_ALARM ARRAY_SIZE(sda_temp_alarm)
static
ssize_t
show_pwm_mode
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
char
*
buf
)
{
struct
nct6775_data
*
data
=
nct6775_update_device
(
dev
);
struct
sensor_device_attribute
*
sattr
=
to_sensor_dev_attr
(
attr
);
return
sprintf
(
buf
,
"%d
\n
"
,
!
data
->
pwm_mode
[
sattr
->
index
]);
}
static
ssize_t
store_pwm_mode
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
const
char
*
buf
,
size_t
count
)
{
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
struct
sensor_device_attribute
*
sattr
=
to_sensor_dev_attr
(
attr
);
int
nr
=
sattr
->
index
;
unsigned
long
val
;
int
err
;
u8
reg
;
err
=
kstrtoul
(
buf
,
10
,
&
val
);
if
(
err
<
0
)
return
err
;
if
(
val
>
1
)
return
-
EINVAL
;
/* Setting DC mode is not supported for all chips/channels */
if
(
data
->
REG_PWM_MODE
[
nr
]
==
0
)
{
if
(
val
)
return
-
EINVAL
;
return
count
;
}
mutex_lock
(
&
data
->
update_lock
);
data
->
pwm_mode
[
nr
]
=
val
;
reg
=
nct6775_read_value
(
data
,
data
->
REG_PWM_MODE
[
nr
]);
reg
&=
~
data
->
PWM_MODE_MASK
[
nr
];
if
(
val
)
reg
|=
data
->
PWM_MODE_MASK
[
nr
];
nct6775_write_value
(
data
,
data
->
REG_PWM_MODE
[
nr
],
reg
);
mutex_unlock
(
&
data
->
update_lock
);
return
count
;
}
static
ssize_t
show_pwm
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
char
*
buf
)
{
struct
nct6775_data
*
data
=
nct6775_update_device
(
dev
);
struct
sensor_device_attribute_2
*
sattr
=
to_sensor_dev_attr_2
(
attr
);
int
nr
=
sattr
->
nr
;
int
index
=
sattr
->
index
;
int
pwm
;
/*
* For automatic fan control modes, show current pwm readings.
* Otherwise, show the configured value.
*/
if
(
index
==
0
&&
data
->
pwm_enable
[
nr
]
>
manual
)
pwm
=
nct6775_read_value
(
data
,
data
->
REG_PWM_READ
[
nr
]);
else
pwm
=
data
->
pwm
[
index
][
nr
];
return
sprintf
(
buf
,
"%d
\n
"
,
pwm
);
}
static
ssize_t
store_pwm
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
const
char
*
buf
,
size_t
count
)
{
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
struct
sensor_device_attribute_2
*
sattr
=
to_sensor_dev_attr_2
(
attr
);
int
nr
=
sattr
->
nr
;
int
index
=
sattr
->
index
;
unsigned
long
val
;
int
err
;
err
=
kstrtoul
(
buf
,
10
,
&
val
);
if
(
err
<
0
)
return
err
;
val
=
clamp_val
(
val
,
0
,
255
);
mutex_lock
(
&
data
->
update_lock
);
data
->
pwm
[
index
][
nr
]
=
val
;
nct6775_write_value
(
data
,
data
->
REG_PWM
[
index
][
nr
],
val
);
mutex_unlock
(
&
data
->
update_lock
);
return
count
;
}
static
ssize_t
show_pwm_enable
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
char
*
buf
)
{
struct
nct6775_data
*
data
=
nct6775_update_device
(
dev
);
struct
sensor_device_attribute
*
sattr
=
to_sensor_dev_attr
(
attr
);
return
sprintf
(
buf
,
"%d
\n
"
,
data
->
pwm_enable
[
sattr
->
index
]);
}
static
ssize_t
store_pwm_enable
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
const
char
*
buf
,
size_t
count
)
{
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
struct
sensor_device_attribute
*
sattr
=
to_sensor_dev_attr
(
attr
);
int
nr
=
sattr
->
index
;
unsigned
long
val
;
int
err
;
u16
reg
;
err
=
kstrtoul
(
buf
,
10
,
&
val
);
if
(
err
<
0
)
return
err
;
if
(
val
>
sf4
)
return
-
EINVAL
;
if
(
val
==
sf3
&&
data
->
kind
!=
nct6775
)
return
-
EINVAL
;
mutex_lock
(
&
data
->
update_lock
);
data
->
pwm_enable
[
nr
]
=
val
;
if
(
val
==
off
)
{
/*
* turn off pwm control: select manual mode, set pwm to maximum
*/
data
->
pwm
[
0
][
nr
]
=
255
;
nct6775_write_value
(
data
,
data
->
REG_PWM
[
0
][
nr
],
255
);
}
reg
=
nct6775_read_value
(
data
,
data
->
REG_FAN_MODE
[
nr
]);
reg
&=
0x0f
;
reg
|=
pwm_enable_to_reg
(
val
)
<<
4
;
nct6775_write_value
(
data
,
data
->
REG_FAN_MODE
[
nr
],
reg
);
mutex_unlock
(
&
data
->
update_lock
);
return
count
;
}
static
SENSOR_DEVICE_ATTR_2
(
pwm1
,
S_IWUSR
|
S_IRUGO
,
show_pwm
,
store_pwm
,
0
,
0
);
static
SENSOR_DEVICE_ATTR_2
(
pwm2
,
S_IWUSR
|
S_IRUGO
,
show_pwm
,
store_pwm
,
1
,
0
);
static
SENSOR_DEVICE_ATTR_2
(
pwm3
,
S_IWUSR
|
S_IRUGO
,
show_pwm
,
store_pwm
,
2
,
0
);
static
SENSOR_DEVICE_ATTR_2
(
pwm4
,
S_IWUSR
|
S_IRUGO
,
show_pwm
,
store_pwm
,
3
,
0
);
static
SENSOR_DEVICE_ATTR_2
(
pwm5
,
S_IWUSR
|
S_IRUGO
,
show_pwm
,
store_pwm
,
4
,
0
);
static
SENSOR_DEVICE_ATTR
(
pwm1_mode
,
S_IWUSR
|
S_IRUGO
,
show_pwm_mode
,
store_pwm_mode
,
0
);
static
SENSOR_DEVICE_ATTR
(
pwm2_mode
,
S_IWUSR
|
S_IRUGO
,
show_pwm_mode
,
store_pwm_mode
,
1
);
static
SENSOR_DEVICE_ATTR
(
pwm3_mode
,
S_IWUSR
|
S_IRUGO
,
show_pwm_mode
,
store_pwm_mode
,
2
);
static
SENSOR_DEVICE_ATTR
(
pwm4_mode
,
S_IWUSR
|
S_IRUGO
,
show_pwm_mode
,
store_pwm_mode
,
3
);
static
SENSOR_DEVICE_ATTR
(
pwm5_mode
,
S_IWUSR
|
S_IRUGO
,
show_pwm_mode
,
store_pwm_mode
,
4
);
static
SENSOR_DEVICE_ATTR
(
pwm1_enable
,
S_IWUSR
|
S_IRUGO
,
show_pwm_enable
,
store_pwm_enable
,
0
);
static
SENSOR_DEVICE_ATTR
(
pwm2_enable
,
S_IWUSR
|
S_IRUGO
,
show_pwm_enable
,
store_pwm_enable
,
1
);
static
SENSOR_DEVICE_ATTR
(
pwm3_enable
,
S_IWUSR
|
S_IRUGO
,
show_pwm_enable
,
store_pwm_enable
,
2
);
static
SENSOR_DEVICE_ATTR
(
pwm4_enable
,
S_IWUSR
|
S_IRUGO
,
show_pwm_enable
,
store_pwm_enable
,
3
);
static
SENSOR_DEVICE_ATTR
(
pwm5_enable
,
S_IWUSR
|
S_IRUGO
,
show_pwm_enable
,
store_pwm_enable
,
4
);
static
ssize_t
static
ssize_t
show_name
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
char
*
buf
)
show_name
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
char
*
buf
)
{
{
...
@@ -1609,6 +1853,47 @@ show_name(struct device *dev, struct device_attribute *attr, char *buf)
...
@@ -1609,6 +1853,47 @@ show_name(struct device *dev, struct device_attribute *attr, char *buf)
static
DEVICE_ATTR
(
name
,
S_IRUGO
,
show_name
,
NULL
);
static
DEVICE_ATTR
(
name
,
S_IRUGO
,
show_name
,
NULL
);
static
struct
attribute
*
nct6775_attributes_pwm
[
5
][
4
]
=
{
{
&
sensor_dev_attr_pwm1
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm1_mode
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm1_enable
.
dev_attr
.
attr
,
NULL
},
{
&
sensor_dev_attr_pwm2
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm2_mode
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm2_enable
.
dev_attr
.
attr
,
NULL
},
{
&
sensor_dev_attr_pwm3
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm3_mode
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm3_enable
.
dev_attr
.
attr
,
NULL
},
{
&
sensor_dev_attr_pwm4
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm4_mode
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm4_enable
.
dev_attr
.
attr
,
NULL
},
{
&
sensor_dev_attr_pwm5
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm5_mode
.
dev_attr
.
attr
,
&
sensor_dev_attr_pwm5_enable
.
dev_attr
.
attr
,
NULL
},
};
static
const
struct
attribute_group
nct6775_group_pwm
[
5
]
=
{
{
.
attrs
=
nct6775_attributes_pwm
[
0
]
},
{
.
attrs
=
nct6775_attributes_pwm
[
1
]
},
{
.
attrs
=
nct6775_attributes_pwm
[
2
]
},
{
.
attrs
=
nct6775_attributes_pwm
[
3
]
},
{
.
attrs
=
nct6775_attributes_pwm
[
4
]
},
};
static
ssize_t
static
ssize_t
show_vid
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
char
*
buf
)
show_vid
(
struct
device
*
dev
,
struct
device_attribute
*
attr
,
char
*
buf
)
{
{
...
@@ -1681,6 +1966,9 @@ static void nct6775_device_remove_files(struct device *dev)
...
@@ -1681,6 +1966,9 @@ static void nct6775_device_remove_files(struct device *dev)
int
i
;
int
i
;
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
struct
nct6775_data
*
data
=
dev_get_drvdata
(
dev
);
for
(
i
=
0
;
i
<
data
->
pwm_num
;
i
++
)
sysfs_remove_group
(
&
dev
->
kobj
,
&
nct6775_group_pwm
[
i
]);
for
(
i
=
0
;
i
<
data
->
in_num
;
i
++
)
for
(
i
=
0
;
i
<
data
->
in_num
;
i
++
)
sysfs_remove_group
(
&
dev
->
kobj
,
&
nct6775_group_in
[
i
]);
sysfs_remove_group
(
&
dev
->
kobj
,
&
nct6775_group_in
[
i
]);
...
@@ -1763,6 +2051,7 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
...
@@ -1763,6 +2051,7 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
{
{
int
regval
;
int
regval
;
bool
fan3pin
,
fan3min
,
fan4pin
,
fan4min
,
fan5pin
;
bool
fan3pin
,
fan3min
,
fan4pin
,
fan4min
,
fan5pin
;
bool
pwm3pin
,
pwm4pin
,
pwm5pin
;
int
ret
;
int
ret
;
ret
=
superio_enter
(
sio_data
->
sioreg
);
ret
=
superio_enter
(
sio_data
->
sioreg
);
...
@@ -1775,11 +2064,14 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
...
@@ -1775,11 +2064,14 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
fan3pin
=
regval
&
(
1
<<
6
);
fan3pin
=
regval
&
(
1
<<
6
);
fan3min
=
fan3pin
;
fan3min
=
fan3pin
;
pwm3pin
=
regval
&
(
1
<<
7
);
/* On NCT6775, fan4 shares pins with the fdc interface */
/* On NCT6775, fan4 shares pins with the fdc interface */
fan4pin
=
!
(
superio_inb
(
sio_data
->
sioreg
,
0x2A
)
&
0x80
);
fan4pin
=
!
(
superio_inb
(
sio_data
->
sioreg
,
0x2A
)
&
0x80
);
fan4min
=
0
;
fan4min
=
0
;
fan5pin
=
0
;
fan5pin
=
0
;
pwm4pin
=
0
;
pwm5pin
=
0
;
}
else
if
(
data
->
kind
==
nct6776
)
{
}
else
if
(
data
->
kind
==
nct6776
)
{
bool
gpok
=
superio_inb
(
sio_data
->
sioreg
,
0x27
)
&
0x80
;
bool
gpok
=
superio_inb
(
sio_data
->
sioreg
,
0x27
)
&
0x80
;
...
@@ -1803,6 +2095,9 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
...
@@ -1803,6 +2095,9 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
fan4min
=
fan4pin
;
fan4min
=
fan4pin
;
fan3min
=
fan3pin
;
fan3min
=
fan3pin
;
pwm3pin
=
fan3pin
;
pwm4pin
=
0
;
pwm5pin
=
0
;
}
else
{
/* NCT6779D */
}
else
{
/* NCT6779D */
regval
=
superio_inb
(
sio_data
->
sioreg
,
0x1c
);
regval
=
superio_inb
(
sio_data
->
sioreg
,
0x1c
);
...
@@ -1810,6 +2105,10 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
...
@@ -1810,6 +2105,10 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
fan4pin
=
!
(
regval
&
(
1
<<
6
));
fan4pin
=
!
(
regval
&
(
1
<<
6
));
fan5pin
=
!
(
regval
&
(
1
<<
7
));
fan5pin
=
!
(
regval
&
(
1
<<
7
));
pwm3pin
=
!
(
regval
&
(
1
<<
0
));
pwm4pin
=
!
(
regval
&
(
1
<<
1
));
pwm5pin
=
!
(
regval
&
(
1
<<
2
));
fan3min
=
fan3pin
;
fan3min
=
fan3pin
;
fan4min
=
fan4pin
;
fan4min
=
fan4pin
;
}
}
...
@@ -1823,6 +2122,8 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
...
@@ -1823,6 +2122,8 @@ nct6775_check_fan_inputs(const struct nct6775_sio_data *sio_data,
data
->
has_fan
|=
(
fan4pin
<<
3
)
|
(
fan5pin
<<
4
);
data
->
has_fan
|=
(
fan4pin
<<
3
)
|
(
fan5pin
<<
4
);
data
->
has_fan_min
|=
(
fan4min
<<
3
)
|
(
fan5pin
<<
4
);
data
->
has_fan_min
|=
(
fan4min
<<
3
)
|
(
fan5pin
<<
4
);
data
->
has_pwm
=
0x03
|
(
pwm3pin
<<
2
)
|
(
pwm4pin
<<
3
)
|
(
pwm5pin
<<
4
);
return
0
;
return
0
;
}
}
...
@@ -1859,6 +2160,7 @@ static int nct6775_probe(struct platform_device *pdev)
...
@@ -1859,6 +2160,7 @@ static int nct6775_probe(struct platform_device *pdev)
switch
(
data
->
kind
)
{
switch
(
data
->
kind
)
{
case
nct6775
:
case
nct6775
:
data
->
in_num
=
9
;
data
->
in_num
=
9
;
data
->
pwm_num
=
3
;
data
->
has_fan_div
=
true
;
data
->
has_fan_div
=
true
;
data
->
temp_fixed_num
=
3
;
data
->
temp_fixed_num
=
3
;
...
@@ -1877,8 +2179,13 @@ static int nct6775_probe(struct platform_device *pdev)
...
@@ -1877,8 +2179,13 @@ static int nct6775_probe(struct platform_device *pdev)
data
->
REG_IN_MINMAX
[
0
]
=
NCT6775_REG_IN_MIN
;
data
->
REG_IN_MINMAX
[
0
]
=
NCT6775_REG_IN_MIN
;
data
->
REG_IN_MINMAX
[
1
]
=
NCT6775_REG_IN_MAX
;
data
->
REG_IN_MINMAX
[
1
]
=
NCT6775_REG_IN_MAX
;
data
->
REG_FAN
=
NCT6775_REG_FAN
;
data
->
REG_FAN
=
NCT6775_REG_FAN
;
data
->
REG_FAN_MODE
=
NCT6775_REG_FAN_MODE
;
data
->
REG_FAN_MIN
=
NCT6775_REG_FAN_MIN
;
data
->
REG_FAN_MIN
=
NCT6775_REG_FAN_MIN
;
data
->
REG_FAN_PULSES
=
NCT6775_REG_FAN_PULSES
;
data
->
REG_FAN_PULSES
=
NCT6775_REG_FAN_PULSES
;
data
->
REG_PWM
[
0
]
=
NCT6775_REG_PWM
;
data
->
REG_PWM_READ
=
NCT6775_REG_PWM_READ
;
data
->
REG_PWM_MODE
=
NCT6775_REG_PWM_MODE
;
data
->
PWM_MODE_MASK
=
NCT6775_PWM_MODE_MASK
;
data
->
REG_TEMP_OFFSET
=
NCT6775_REG_TEMP_OFFSET
;
data
->
REG_TEMP_OFFSET
=
NCT6775_REG_TEMP_OFFSET
;
data
->
REG_TEMP_SOURCE
=
NCT6775_REG_TEMP_SOURCE
;
data
->
REG_TEMP_SOURCE
=
NCT6775_REG_TEMP_SOURCE
;
data
->
REG_ALARM
=
NCT6775_REG_ALARM
;
data
->
REG_ALARM
=
NCT6775_REG_ALARM
;
...
@@ -1894,6 +2201,7 @@ static int nct6775_probe(struct platform_device *pdev)
...
@@ -1894,6 +2201,7 @@ static int nct6775_probe(struct platform_device *pdev)
break
;
break
;
case
nct6776
:
case
nct6776
:
data
->
in_num
=
9
;
data
->
in_num
=
9
;
data
->
pwm_num
=
3
;
data
->
has_fan_div
=
false
;
data
->
has_fan_div
=
false
;
data
->
temp_fixed_num
=
3
;
data
->
temp_fixed_num
=
3
;
...
@@ -1912,8 +2220,13 @@ static int nct6775_probe(struct platform_device *pdev)
...
@@ -1912,8 +2220,13 @@ static int nct6775_probe(struct platform_device *pdev)
data
->
REG_IN_MINMAX
[
0
]
=
NCT6775_REG_IN_MIN
;
data
->
REG_IN_MINMAX
[
0
]
=
NCT6775_REG_IN_MIN
;
data
->
REG_IN_MINMAX
[
1
]
=
NCT6775_REG_IN_MAX
;
data
->
REG_IN_MINMAX
[
1
]
=
NCT6775_REG_IN_MAX
;
data
->
REG_FAN
=
NCT6775_REG_FAN
;
data
->
REG_FAN
=
NCT6775_REG_FAN
;
data
->
REG_FAN_MODE
=
NCT6775_REG_FAN_MODE
;
data
->
REG_FAN_MIN
=
NCT6776_REG_FAN_MIN
;
data
->
REG_FAN_MIN
=
NCT6776_REG_FAN_MIN
;
data
->
REG_FAN_PULSES
=
NCT6776_REG_FAN_PULSES
;
data
->
REG_FAN_PULSES
=
NCT6776_REG_FAN_PULSES
;
data
->
REG_PWM
[
0
]
=
NCT6775_REG_PWM
;
data
->
REG_PWM_READ
=
NCT6775_REG_PWM_READ
;
data
->
REG_PWM_MODE
=
NCT6776_REG_PWM_MODE
;
data
->
PWM_MODE_MASK
=
NCT6776_PWM_MODE_MASK
;
data
->
REG_TEMP_OFFSET
=
NCT6775_REG_TEMP_OFFSET
;
data
->
REG_TEMP_OFFSET
=
NCT6775_REG_TEMP_OFFSET
;
data
->
REG_TEMP_SOURCE
=
NCT6775_REG_TEMP_SOURCE
;
data
->
REG_TEMP_SOURCE
=
NCT6775_REG_TEMP_SOURCE
;
data
->
REG_ALARM
=
NCT6775_REG_ALARM
;
data
->
REG_ALARM
=
NCT6775_REG_ALARM
;
...
@@ -1929,6 +2242,7 @@ static int nct6775_probe(struct platform_device *pdev)
...
@@ -1929,6 +2242,7 @@ static int nct6775_probe(struct platform_device *pdev)
break
;
break
;
case
nct6779
:
case
nct6779
:
data
->
in_num
=
15
;
data
->
in_num
=
15
;
data
->
pwm_num
=
5
;
data
->
has_fan_div
=
false
;
data
->
has_fan_div
=
false
;
data
->
temp_fixed_num
=
6
;
data
->
temp_fixed_num
=
6
;
...
@@ -1947,8 +2261,13 @@ static int nct6775_probe(struct platform_device *pdev)
...
@@ -1947,8 +2261,13 @@ static int nct6775_probe(struct platform_device *pdev)
data
->
REG_IN_MINMAX
[
0
]
=
NCT6775_REG_IN_MIN
;
data
->
REG_IN_MINMAX
[
0
]
=
NCT6775_REG_IN_MIN
;
data
->
REG_IN_MINMAX
[
1
]
=
NCT6775_REG_IN_MAX
;
data
->
REG_IN_MINMAX
[
1
]
=
NCT6775_REG_IN_MAX
;
data
->
REG_FAN
=
NCT6779_REG_FAN
;
data
->
REG_FAN
=
NCT6779_REG_FAN
;
data
->
REG_FAN_MODE
=
NCT6775_REG_FAN_MODE
;
data
->
REG_FAN_MIN
=
NCT6776_REG_FAN_MIN
;
data
->
REG_FAN_MIN
=
NCT6776_REG_FAN_MIN
;
data
->
REG_FAN_PULSES
=
NCT6779_REG_FAN_PULSES
;
data
->
REG_FAN_PULSES
=
NCT6779_REG_FAN_PULSES
;
data
->
REG_PWM
[
0
]
=
NCT6775_REG_PWM
;
data
->
REG_PWM_READ
=
NCT6775_REG_PWM_READ
;
data
->
REG_PWM_MODE
=
NCT6776_REG_PWM_MODE
;
data
->
PWM_MODE_MASK
=
NCT6776_PWM_MODE_MASK
;
data
->
REG_TEMP_OFFSET
=
NCT6779_REG_TEMP_OFFSET
;
data
->
REG_TEMP_OFFSET
=
NCT6779_REG_TEMP_OFFSET
;
data
->
REG_TEMP_SOURCE
=
NCT6775_REG_TEMP_SOURCE
;
data
->
REG_TEMP_SOURCE
=
NCT6775_REG_TEMP_SOURCE
;
data
->
REG_ALARM
=
NCT6779_REG_ALARM
;
data
->
REG_ALARM
=
NCT6779_REG_ALARM
;
...
@@ -2157,6 +2476,16 @@ static int nct6775_probe(struct platform_device *pdev)
...
@@ -2157,6 +2476,16 @@ static int nct6775_probe(struct platform_device *pdev)
/* Read fan clock dividers immediately */
/* Read fan clock dividers immediately */
nct6775_init_fan_common
(
dev
,
data
);
nct6775_init_fan_common
(
dev
,
data
);
/* Register sysfs hooks */
for
(
i
=
0
;
i
<
data
->
pwm_num
;
i
++
)
{
if
(
!
(
data
->
has_pwm
&
(
1
<<
i
)))
continue
;
err
=
sysfs_create_group
(
&
dev
->
kobj
,
&
nct6775_group_pwm
[
i
]);
if
(
err
)
goto
exit_remove
;
}
for
(
i
=
0
;
i
<
data
->
in_num
;
i
++
)
{
for
(
i
=
0
;
i
<
data
->
in_num
;
i
++
)
{
if
(
!
(
data
->
have_in
&
(
1
<<
i
)))
if
(
!
(
data
->
have_in
&
(
1
<<
i
)))
continue
;
continue
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment