mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2026-04-29 12:28:27 +02:00
hwmon: (mlxreg-fan) Extend driver to support multiply cooling devices
Add support for additional cooling devices in order to support the systems, which can be equipped with up-to four PWM controllers. Signed-off-by: Vadim Pasternak <vadimp@nvidia.com> Signed-off-by: Guenter Roeck <linux@roeck-us.net>
This commit is contained in:
committed by
Guenter Roeck
parent
150f1e0c6f
commit
d7efb2ebc7
+47
-26
@@ -63,6 +63,8 @@
|
|||||||
MLXREG_FAN_MAX_DUTY, \
|
MLXREG_FAN_MAX_DUTY, \
|
||||||
MLXREG_FAN_MAX_STATE))
|
MLXREG_FAN_MAX_STATE))
|
||||||
|
|
||||||
|
struct mlxreg_fan;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* struct mlxreg_fan_tacho - tachometer data (internal use):
|
* struct mlxreg_fan_tacho - tachometer data (internal use):
|
||||||
*
|
*
|
||||||
@@ -81,12 +83,18 @@ struct mlxreg_fan_tacho {
|
|||||||
/*
|
/*
|
||||||
* struct mlxreg_fan_pwm - PWM data (internal use):
|
* struct mlxreg_fan_pwm - PWM data (internal use):
|
||||||
*
|
*
|
||||||
|
* @fan: private data;
|
||||||
* @connected: indicates if PWM is connected;
|
* @connected: indicates if PWM is connected;
|
||||||
* @reg: register offset;
|
* @reg: register offset;
|
||||||
|
* @cooling: cooling device levels;
|
||||||
|
* @cdev: cooling device;
|
||||||
*/
|
*/
|
||||||
struct mlxreg_fan_pwm {
|
struct mlxreg_fan_pwm {
|
||||||
|
struct mlxreg_fan *fan;
|
||||||
bool connected;
|
bool connected;
|
||||||
u32 reg;
|
u32 reg;
|
||||||
|
u8 cooling_levels[MLXREG_FAN_MAX_STATE + 1];
|
||||||
|
struct thermal_cooling_device *cdev;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -99,8 +107,6 @@ struct mlxreg_fan_pwm {
|
|||||||
* @tachos_per_drwr - number of tachometers per drawer;
|
* @tachos_per_drwr - number of tachometers per drawer;
|
||||||
* @samples: minimum allowed samples per pulse;
|
* @samples: minimum allowed samples per pulse;
|
||||||
* @divider: divider value for tachometer RPM calculation;
|
* @divider: divider value for tachometer RPM calculation;
|
||||||
* @cooling: cooling device levels;
|
|
||||||
* @cdev: cooling device;
|
|
||||||
*/
|
*/
|
||||||
struct mlxreg_fan {
|
struct mlxreg_fan {
|
||||||
struct device *dev;
|
struct device *dev;
|
||||||
@@ -111,8 +117,6 @@ struct mlxreg_fan {
|
|||||||
int tachos_per_drwr;
|
int tachos_per_drwr;
|
||||||
int samples;
|
int samples;
|
||||||
int divider;
|
int divider;
|
||||||
u8 cooling_levels[MLXREG_FAN_MAX_STATE + 1];
|
|
||||||
struct thermal_cooling_device *cdev;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static int
|
static int
|
||||||
@@ -305,11 +309,12 @@ static int mlxreg_fan_get_cur_state(struct thermal_cooling_device *cdev,
|
|||||||
unsigned long *state)
|
unsigned long *state)
|
||||||
|
|
||||||
{
|
{
|
||||||
struct mlxreg_fan *fan = cdev->devdata;
|
struct mlxreg_fan_pwm *pwm = cdev->devdata;
|
||||||
|
struct mlxreg_fan *fan = pwm->fan;
|
||||||
u32 regval;
|
u32 regval;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
err = regmap_read(fan->regmap, fan->pwm[0].reg, ®val);
|
err = regmap_read(fan->regmap, pwm->reg, ®val);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(fan->dev, "Failed to query PWM duty\n");
|
dev_err(fan->dev, "Failed to query PWM duty\n");
|
||||||
return err;
|
return err;
|
||||||
@@ -324,7 +329,8 @@ static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev,
|
|||||||
unsigned long state)
|
unsigned long state)
|
||||||
|
|
||||||
{
|
{
|
||||||
struct mlxreg_fan *fan = cdev->devdata;
|
struct mlxreg_fan_pwm *pwm = cdev->devdata;
|
||||||
|
struct mlxreg_fan *fan = pwm->fan;
|
||||||
unsigned long cur_state;
|
unsigned long cur_state;
|
||||||
int i, config = 0;
|
int i, config = 0;
|
||||||
u32 regval;
|
u32 regval;
|
||||||
@@ -348,11 +354,11 @@ static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev,
|
|||||||
config = 1;
|
config = 1;
|
||||||
state -= MLXREG_FAN_MAX_STATE;
|
state -= MLXREG_FAN_MAX_STATE;
|
||||||
for (i = 0; i < state; i++)
|
for (i = 0; i < state; i++)
|
||||||
fan->cooling_levels[i] = state;
|
pwm->cooling_levels[i] = state;
|
||||||
for (i = state; i <= MLXREG_FAN_MAX_STATE; i++)
|
for (i = state; i <= MLXREG_FAN_MAX_STATE; i++)
|
||||||
fan->cooling_levels[i] = i;
|
pwm->cooling_levels[i] = i;
|
||||||
|
|
||||||
err = regmap_read(fan->regmap, fan->pwm[0].reg, ®val);
|
err = regmap_read(fan->regmap, pwm->reg, ®val);
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(fan->dev, "Failed to query PWM duty\n");
|
dev_err(fan->dev, "Failed to query PWM duty\n");
|
||||||
return err;
|
return err;
|
||||||
@@ -369,8 +375,8 @@ static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
/* Normalize the state to the valid speed range. */
|
/* Normalize the state to the valid speed range. */
|
||||||
state = fan->cooling_levels[state];
|
state = pwm->cooling_levels[state];
|
||||||
err = regmap_write(fan->regmap, fan->pwm[0].reg,
|
err = regmap_write(fan->regmap, pwm->reg,
|
||||||
MLXREG_FAN_PWM_STATE2DUTY(state));
|
MLXREG_FAN_PWM_STATE2DUTY(state));
|
||||||
if (err) {
|
if (err) {
|
||||||
dev_err(fan->dev, "Failed to write PWM duty\n");
|
dev_err(fan->dev, "Failed to write PWM duty\n");
|
||||||
@@ -541,11 +547,32 @@ static int mlxreg_fan_config(struct mlxreg_fan *fan,
|
|||||||
fan->tachos_per_drwr = tacho_avail / drwr_avail;
|
fan->tachos_per_drwr = tacho_avail / drwr_avail;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Init cooling levels per PWM state. */
|
return 0;
|
||||||
for (i = 0; i < MLXREG_FAN_SPEED_MIN_LEVEL; i++)
|
}
|
||||||
fan->cooling_levels[i] = MLXREG_FAN_SPEED_MIN_LEVEL;
|
|
||||||
for (i = MLXREG_FAN_SPEED_MIN_LEVEL; i <= MLXREG_FAN_MAX_STATE; i++)
|
static int mlxreg_fan_cooling_config(struct device *dev, struct mlxreg_fan *fan)
|
||||||
fan->cooling_levels[i] = i;
|
{
|
||||||
|
int i, j;
|
||||||
|
|
||||||
|
for (i = 0; i <= MLXREG_FAN_MAX_PWM; i++) {
|
||||||
|
struct mlxreg_fan_pwm *pwm = &fan->pwm[i];
|
||||||
|
|
||||||
|
if (!pwm->connected)
|
||||||
|
continue;
|
||||||
|
pwm->fan = fan;
|
||||||
|
pwm->cdev = devm_thermal_of_cooling_device_register(dev, NULL, "mlxreg_fan", pwm,
|
||||||
|
&mlxreg_fan_cooling_ops);
|
||||||
|
if (IS_ERR(pwm->cdev)) {
|
||||||
|
dev_err(dev, "Failed to register cooling device\n");
|
||||||
|
return PTR_ERR(pwm->cdev);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Init cooling levels per PWM state. */
|
||||||
|
for (j = 0; j < MLXREG_FAN_SPEED_MIN_LEVEL; j++)
|
||||||
|
pwm->cooling_levels[j] = MLXREG_FAN_SPEED_MIN_LEVEL;
|
||||||
|
for (j = MLXREG_FAN_SPEED_MIN_LEVEL; j <= MLXREG_FAN_MAX_STATE; j++)
|
||||||
|
pwm->cooling_levels[j] = j;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -584,16 +611,10 @@ static int mlxreg_fan_probe(struct platform_device *pdev)
|
|||||||
return PTR_ERR(hwm);
|
return PTR_ERR(hwm);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (IS_REACHABLE(CONFIG_THERMAL)) {
|
if (IS_REACHABLE(CONFIG_THERMAL))
|
||||||
fan->cdev = devm_thermal_of_cooling_device_register(dev,
|
err = mlxreg_fan_cooling_config(dev, fan);
|
||||||
NULL, "mlxreg_fan", fan, &mlxreg_fan_cooling_ops);
|
|
||||||
if (IS_ERR(fan->cdev)) {
|
|
||||||
dev_err(dev, "Failed to register cooling device\n");
|
|
||||||
return PTR_ERR(fan->cdev);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct platform_driver mlxreg_fan_driver = {
|
static struct platform_driver mlxreg_fan_driver = {
|
||||||
|
|||||||
Reference in New Issue
Block a user