diff options
Diffstat (limited to 'drivers/hwmon/mlxreg-fan.c')
| -rw-r--r-- | drivers/hwmon/mlxreg-fan.c | 47 | 
1 files changed, 33 insertions, 14 deletions
| diff --git a/drivers/hwmon/mlxreg-fan.c b/drivers/hwmon/mlxreg-fan.c index a5f89aab3fb4..137a90dd2075 100644 --- a/drivers/hwmon/mlxreg-fan.c +++ b/drivers/hwmon/mlxreg-fan.c @@ -63,12 +63,14 @@ struct mlxreg_fan;   * @reg: register offset;   * @mask: fault mask;   * @prsnt: present register offset; + * @shift: tacho presence bit shift;   */  struct mlxreg_fan_tacho {  	bool connected;  	u32 reg;  	u32 mask;  	u32 prsnt; +	u32 shift;  };  /* @@ -113,8 +115,8 @@ struct mlxreg_fan {  	int divider;  }; -static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev, -				    unsigned long state); +static int _mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev, +				     unsigned long state, bool thermal);  static int  mlxreg_fan_read(struct device *dev, enum hwmon_sensor_types type, u32 attr, @@ -143,8 +145,10 @@ mlxreg_fan_read(struct device *dev, enum hwmon_sensor_types type, u32 attr,  				/*  				 * Map channel to presence bit - drawer can be equipped with  				 * one or few FANs, while presence is indicated per drawer. +				 * Shift channel value if necessary to align with register value.  				 */ -				if (BIT(channel / fan->tachos_per_drwr) & regval) { +				if (BIT(rol32(channel, tacho->shift) / fan->tachos_per_drwr) & +					regval) {  					/* FAN is not connected - return zero for FAN speed. */  					*val = 0;  					return 0; @@ -224,8 +228,9 @@ mlxreg_fan_write(struct device *dev, enum hwmon_sensor_types type, u32 attr,  				 * last thermal state.  				 */  				if (pwm->last_hwmon_state >= pwm->last_thermal_state) -					return mlxreg_fan_set_cur_state(pwm->cdev, -									pwm->last_hwmon_state); +					return _mlxreg_fan_set_cur_state(pwm->cdev, +									 pwm->last_hwmon_state, +									 false);  				return 0;  			}  			return regmap_write(fan->regmap, pwm->reg, val); @@ -357,9 +362,8 @@ static int mlxreg_fan_get_cur_state(struct thermal_cooling_device *cdev,  	return 0;  } -static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev, -				    unsigned long state) - +static int _mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev, +				     unsigned long state, bool thermal)  {  	struct mlxreg_fan_pwm *pwm = cdev->devdata;  	struct mlxreg_fan *fan = pwm->fan; @@ -369,7 +373,8 @@ static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev,  		return -EINVAL;  	/* Save thermal state. */ -	pwm->last_thermal_state = state; +	if (thermal) +		pwm->last_thermal_state = state;  	state = max_t(unsigned long, state, pwm->last_hwmon_state);  	err = regmap_write(fan->regmap, pwm->reg, @@ -381,6 +386,13 @@ static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev,  	return 0;  } +static int mlxreg_fan_set_cur_state(struct thermal_cooling_device *cdev, +				    unsigned long state) + +{ +	return _mlxreg_fan_set_cur_state(cdev, state, true); +} +  static const struct thermal_cooling_device_ops mlxreg_fan_cooling_ops = {  	.get_max_state	= mlxreg_fan_get_max_state,  	.get_cur_state	= mlxreg_fan_get_cur_state, @@ -400,7 +412,7 @@ static int mlxreg_fan_connect_verify(struct mlxreg_fan *fan,  		return err;  	} -	return !!(regval & data->bit); +	return data->slot ? (data->slot <= regval ? 1 : 0) : !!(regval & data->bit);  }  static int mlxreg_pwm_connect_verify(struct mlxreg_fan *fan, @@ -537,7 +549,15 @@ static int mlxreg_fan_config(struct mlxreg_fan *fan,  			return err;  		} -		drwr_avail = hweight32(regval); +		/* +		 * The number of drawers could be specified in registers by counters for newer +		 * systems, or by bitmasks for older systems. In case the data is provided by +		 * counter, it is indicated through 'version' field. +		 */ +		if (pdata->version) +			drwr_avail = regval; +		else +			drwr_avail = hweight32(regval);  		if (!tacho_avail || !drwr_avail || tacho_avail < drwr_avail) {  			dev_err(fan->dev, "Configuration is invalid: drawers num %d tachos num %d\n",  				drwr_avail, tacho_avail); @@ -561,15 +581,14 @@ static int mlxreg_fan_cooling_config(struct device *dev, struct mlxreg_fan *fan)  		if (!pwm->connected)  			continue;  		pwm->fan = fan; +		/* Set minimal PWM speed. */ +		pwm->last_hwmon_state = MLXREG_FAN_PWM_DUTY2STATE(MLXREG_FAN_MIN_DUTY);  		pwm->cdev = devm_thermal_of_cooling_device_register(dev, NULL, mlxreg_fan_name[i],  								    pwm, &mlxreg_fan_cooling_ops);  		if (IS_ERR(pwm->cdev)) {  			dev_err(dev, "Failed to register cooling device\n");  			return PTR_ERR(pwm->cdev);  		} - -		/* Set minimal PWM speed. */ -		pwm->last_hwmon_state = MLXREG_FAN_PWM_DUTY2STATE(MLXREG_FAN_MIN_DUTY);  	}  	return 0; | 
