PATCH: f71882fg: add f8000 support

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Sun, 14 Dec 2008 17:27:14 +0100, Hans de Goede wrote:
> <oops, now with attachment>
> 
> Hi Jean,
> 
> ANd finally the patch actually adding the f8000 support.
> 
> Thanks & Regards,
> 
> Hans
> 
> p.s.
> 
> While doing this patch I noticed that for some reason the fan_attr array for 
> the f71862fg is missing the attr for the 3th pwm output. I'll do a separate 
> patch adding that. When that is done too, I'll mail the 2 f8000 testers you've 
> send me mails about asking them to test the new f71882fg.c with f8000 support.

Sounds like a good plan.

Review:

> And (finally) the patch actually adding f8000 support
> 
> Signed-off-by: Hans de Goede <hdegoede at redhat.com>
> --- linux/drivers/hwmon/f71882fg.c.12-applied	2008-12-14 16:25:49.000000000 +0100
> +++ linux/drivers/hwmon/f71882fg.c	2008-12-14 17:15:27.000000000 +0100
> @@ -45,6 +45,7 @@
>  #define SIO_FINTEK_ID		0x1934	/* Manufacturers ID */
>  #define SIO_F71862_ID		0x0601	/* Chipset ID */
>  #define SIO_F71882_ID		0x0541	/* Chipset ID */
> +#define SIO_F8000_ID		0x0581	/* Chipset ID */
>  
>  #define REGION_LENGTH		8
>  #define ADDR_REG_OFFSET		5
> @@ -90,11 +91,12 @@ static unsigned short force_id;
>  module_param(force_id, ushort, 0);
>  MODULE_PARM_DESC(force_id, "Override the detected device ID");
>  
> -enum chips { f71862fg, f71882fg };
> +enum chips { f71862fg, f71882fg, f8000 };
>  
>  static const char *f71882fg_names[] = {
>  	"f71862fg",
>  	"f71882fg",
> +	"f8000",
>  };
>  
>  static struct platform_device *f71882fg_pdev;
> @@ -249,6 +251,7 @@ static struct platform_driver f71882fg_d
>  
>  static DEVICE_ATTR(name, S_IRUGO, show_name, NULL);
>  
> +/* Temp and in attr common to both the f71862fg and f71882fg */
>  static struct sensor_device_attribute_2 f718x2fg_in_temp_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),
> @@ -312,6 +315,7 @@ static struct sensor_device_attribute_2 
>  	SENSOR_ATTR_2(temp3_fault, S_IRUGO, show_temp_fault, NULL, 0, 3),
>  };
>  
> +/* Temp and in attr found only on the f71882fg */
>  static struct sensor_device_attribute_2 f71882fg_in_temp_attr[] = {
>  	SENSOR_ATTR_2(in1_max, S_IRUGO|S_IWUSR, show_in_max, store_in_max,
>  		0, 1),
> @@ -320,27 +324,51 @@ static struct sensor_device_attribute_2 
>  	SENSOR_ATTR_2(in1_alarm, S_IRUGO, show_in_alarm, NULL, 0, 1),
>  };
>  
> -static struct sensor_device_attribute_2 f718x2fg_fan_attr[] = {
> +/* Temp and in attr for the f8000
> +   Note on the f8000 temp_ovt (crit) is used as max, and temp_high (max)
> +   is used as hysteresis value to clear alarms

Ouch, that's tricky. Good catch.

> + */
> +static struct sensor_device_attribute_2 f8000_in_temp_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(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),
> +	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),
> +	SENSOR_ATTR_2(temp2_type, S_IRUGO, show_temp_type, 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_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),
> +};
> +
> +/* Fan / PWM attr common to all models */
> +static struct sensor_device_attribute_2 fxxxx_fan_attr[] = {
>  	SENSOR_ATTR_2(fan1_input, S_IRUGO, show_fan, NULL, 0, 0),
>  	SENSOR_ATTR_2(fan1_full_speed, S_IRUGO|S_IWUSR,
>  		      show_fan_full_speed,
>  		      store_fan_full_speed, 0, 0),
> -	SENSOR_ATTR_2(fan1_beep, S_IRUGO|S_IWUSR, show_fan_beep,
> -		store_fan_beep, 0, 0),
>  	SENSOR_ATTR_2(fan1_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 0),
>  	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_beep, S_IRUGO|S_IWUSR, show_fan_beep,
> -		store_fan_beep, 0, 1),
>  	SENSOR_ATTR_2(fan2_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 1),
>  	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_beep, S_IRUGO|S_IWUSR, show_fan_beep,
> -		store_fan_beep, 0, 2),
>  	SENSOR_ATTR_2(fan3_alarm, S_IRUGO, show_fan_alarm, NULL, 0, 2),
>  
>  	SENSOR_ATTR_2(pwm1, S_IRUGO|S_IWUSR, show_pwm, store_pwm, 0, 0),
> @@ -361,9 +389,6 @@ static struct sensor_device_attribute_2 
>  		      show_pwm_auto_point_channel,
>  		      store_pwm_auto_point_channel, 0, 1),
>  
> -	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),
>  	SENSOR_ATTR_2(pwm3_interpolate, S_IRUGO|S_IWUSR,
>  		      show_pwm_interpolate, store_pwm_interpolate, 0, 2),
>  	SENSOR_ATTR_2(pwm3_auto_channels_temp, S_IRUGO|S_IWUSR,
> @@ -371,7 +396,16 @@ static struct sensor_device_attribute_2 
>  		      store_pwm_auto_point_channel, 0, 2),
>  };
>  
> +/* Fan / PWM attr for the f71862fg, less pwms and less zones per pwm then the

Spelling: than.

(Fixed it myself.)

> +   f71882fg */
>  static struct sensor_device_attribute_2 f71862fg_fan_attr[] = {
> +	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),
> +
>  	SENSOR_ATTR_2(pwm1_auto_point1_pwm, S_IRUGO|S_IWUSR,
>  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
>  		      1, 0),
> @@ -411,7 +445,14 @@ static struct sensor_device_attribute_2 
>  		      show_pwm_auto_point_temp_hyst, NULL, 3, 1),
>  };
>  
> +/* Fan / PWM attr for the f71882fg */
>  static struct sensor_device_attribute_2 f71882fg_fan_attr[] = {
> +	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),
>  	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,
> @@ -496,6 +537,9 @@ static struct sensor_device_attribute_2 
>  	SENSOR_ATTR_2(pwm2_auto_point4_temp_hyst, S_IRUGO,
>  		      show_pwm_auto_point_temp_hyst, NULL, 3, 1),
>  
> +	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),
>  	SENSOR_ATTR_2(pwm3_auto_point1_pwm, S_IRUGO|S_IWUSR,
>  		      show_pwm_auto_point_pwm, store_pwm_auto_point_pwm,
>  		      0, 2),
> @@ -581,6 +625,128 @@ static struct sensor_device_attribute_2 
>  		      show_pwm_auto_point_temp_hyst, NULL, 3, 3),
>  };
>  
> +/* Fan / 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 */
> +static struct sensor_device_attribute_2 f8000_fan_attr[] = {
> +	SENSOR_ATTR_2(fan4_input, S_IRUGO, show_fan, NULL, 0, 3),
> +
> +	SENSOR_ATTR_2(pwm3, S_IRUGO, show_pwm, NULL, 0, 2),
> +
> +	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),
> +
> +	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),
> +
> +	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),
> +};
>  
>  /* Super I/O functions */
>  static inline int superio_inb(int base, int reg)
> @@ -666,8 +832,10 @@ static void f71882fg_write16(struct f718
>  static struct f71882fg_data *f71882fg_update_device(struct device *dev)
>  {
>  	struct f71882fg_data *data = dev_get_drvdata(dev);
> -	int nr, reg, reg2;
> -	int nr_fans = (data->type == f71862fg) ? 3 : 4;
> +	int nr, reg = 0, reg2;
> +	int nr_fans = (data->type == f71882fg) ? 4 : 3;
> +	int nr_ins = (data->type == f8000) ? 3 : 9;
> +	int temp_start = (data->type == f8000) ? 0 : 1;
>  
>  	mutex_lock(&data->update_lock);
>  
> @@ -682,35 +850,36 @@ static struct f71882fg_data *f71882fg_up
>  		}
>  
>  		/* Get High & boundary temps*/
> -		for (nr = 1; nr < 4; nr++) {
> +		for (nr = temp_start; nr < 3 + temp_start; nr++) {
>  			data->temp_ovt[nr] = f71882fg_read8(data,
>  						F71882FG_REG_TEMP_OVT(nr));
>  			data->temp_high[nr] = f71882fg_read8(data,
>  						F71882FG_REG_TEMP_HIGH(nr));
>  		}
>  
> -		/* hyst */
> -		data->temp_hyst[0] =
> -			f71882fg_read8(data, F71882FG_REG_TEMP_HYST(0));
> -		data->temp_hyst[1] =
> -			f71882fg_read8(data, F71882FG_REG_TEMP_HYST(1));
> -
> -		/* Have to hardcode type, because temp1 is special */
> -		reg  = f71882fg_read8(data, F71882FG_REG_TEMP_TYPE);
> +		if (data->type != f8000) {
> +			data->fan_beep = f71882fg_read8(data,
> +						F71882FG_REG_FAN_BEEP);
> +			data->temp_beep = f71882fg_read8(data,
> +						F71882FG_REG_TEMP_BEEP);
> +			data->temp_hyst[0] = f71882fg_read8(data,
> +						F71882FG_REG_TEMP_HYST(0));
> +			data->temp_hyst[1] = f71882fg_read8(data,
> +						F71882FG_REG_TEMP_HYST(1));
> +			/* Have to hardcode type, because temp1 is special */

This comment is a bit surprising as you now handle temp1 separately.

> +			reg  = f71882fg_read8(data, F71882FG_REG_TEMP_TYPE);
> +			data->temp_type[2] = (reg & 0x04) ? 2 : 4;
> +			data->temp_type[3] = (reg & 0x08) ? 2 : 4;
> +		}
>  		reg2 = f71882fg_read8(data, F71882FG_REG_PECI);
>  		if ((reg2 & 0x03) == 0x01)
>  			data->temp_type[1] = 6 /* PECI */;
>  		else if ((reg2 & 0x03) == 0x02)
>  			data->temp_type[1] = 5 /* AMDSI */;
> -		else
> +		else if (data->type != f8000)
>  			data->temp_type[1] = (reg & 0x02) ? 2 : 4;
> -
> -		data->temp_type[2] = (reg & 0x04) ? 2 : 4;
> -		data->temp_type[3] = (reg & 0x08) ? 2 : 4;
> -
> -		data->temp_beep = f71882fg_read8(data, F71882FG_REG_TEMP_BEEP);
> -
> -		data->fan_beep = f71882fg_read8(data, F71882FG_REG_FAN_BEEP);
> +		else
> +			data->temp_type[1] = 2; /* F8000 only supports BJT */
>  
>  		data->pwm_enable = f71882fg_read8(data,
>  						  F71882FG_REG_PWM_ENABLE);
> @@ -724,7 +893,7 @@ static struct f71882fg_data *f71882fg_up
>  			    f71882fg_read8(data,
>  					   F71882FG_REG_POINT_MAPPING(nr));
>  
> -			if (data->type == f71882fg) {
> +			if (data->type != f71862fg) {
>  				int point;
>  				for (point = 0; point < 5; point++) {
>  					data->pwm_auto_point_pwm[nr][point] =
> @@ -766,7 +935,7 @@ static struct f71882fg_data *f71882fg_up
>  						F71882FG_REG_TEMP_STATUS);
>  		data->temp_diode_open = f71882fg_read8(data,
>  						F71882FG_REG_TEMP_DIODE_OPEN);
> -		for (nr = 1; nr < 4; nr++)
> +		for (nr = temp_start; nr < 3 + temp_start; nr++)
>  			data->temp[nr] = f71882fg_read8(data,
>  						F71882FG_REG_TEMP(nr));
>  
> @@ -784,10 +953,14 @@ static struct f71882fg_data *f71882fg_up
>  			    f71882fg_read8(data, F71882FG_REG_PWM(nr));
>  		}
>  
> +		/* The f8000 can monitor 1 more fan, but has no pwm for it */
> +		if (data->type == f8000)
> +			data->fan[3] = f71882fg_read8(data,
> +						F71882FG_REG_FAN(3));
>  		if (data->type == f71882fg)
>  			data->in_status = f71882fg_read8(data,
>  						F71882FG_REG_IN_STATUS);
> -		for (nr = 0; nr < 9; nr++)
> +		for (nr = 0; nr < nr_ins; nr++)
>  			data->in[nr] = f71882fg_read8(data,
>  						F71882FG_REG_IN(nr));
>  
> @@ -1181,6 +1354,11 @@ static ssize_t store_pwm(struct device *
>  
>  	mutex_lock(&data->update_lock);
>  	data->pwm_enable = f71882fg_read8(data, F71882FG_REG_PWM_ENABLE);
> +	if ((data->type == f8000 && ((data->pwm_enable >> 2 * nr) & 3) != 2) ||
> +	    (data->type != f8000 && !((data->pwm_enable >> 2 * nr) & 2))) {
> +		count = -EROFS;
> +		goto leave;
> +	}
>  	if (data->pwm_enable & (1 << (2 * nr))) {
>  		/* PWM mode */
>  		f71882fg_write8(data, F71882FG_REG_PWM(nr), val);
> @@ -1195,6 +1373,7 @@ static ssize_t store_pwm(struct device *
>  		data->fan_target[nr] = target;
>  		data->fan_full_speed[nr] = full_speed;
>  	}
> +leave:
>  	mutex_unlock(&data->update_lock);
>  
>  	return count;
> @@ -1203,14 +1382,25 @@ static ssize_t store_pwm(struct device *
>  static ssize_t show_pwm_enable(struct device *dev,
>  			       struct device_attribute *devattr, char *buf)
>  {
> -	int result;
> +	int result = 0;
>  	struct f71882fg_data *data = f71882fg_update_device(dev);
>  	int nr = to_sensor_dev_attr_2(devattr)->index;
>  
> -	if (data->pwm_enable & (2 << (2 * nr)))
> -		result = 1;
> -	else
> -		result = 2;
> +	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;
> +	}
>  
>  	return sprintf(buf, "%d\n", result);
>  }
> @@ -1221,20 +1411,37 @@ static ssize_t store_pwm_enable(struct d
>  	struct f71882fg_data *data = dev_get_drvdata(dev);
>  	int nr = to_sensor_dev_attr_2(devattr)->index;
>  	long val = simple_strtol(buf, NULL, 10);
> -	if (val < 1 || val > 2)
> -		return -EINVAL;
>  
>  	mutex_lock(&data->update_lock);
>  	data->pwm_enable = f71882fg_read8(data, F71882FG_REG_PWM_ENABLE);
> -	switch (val) {
> -	case 1:
> -		data->pwm_enable |= 2 << (2 * nr);
> -		break;		/* Manual */
> -	case 2:
> -		data->pwm_enable &= ~(2 << (2 * nr));
> -		break;		/* Temperature ctrl */
> +	/* 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:
> +			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;
> +		}
>  	}
>  	f71882fg_write8(data, F71882FG_REG_PWM_ENABLE, data->pwm_enable);
> +leave:
>  	mutex_unlock(&data->update_lock);
>  
>  	return count;
> @@ -1516,34 +1723,51 @@ static int __devinit f71882fg_probe(stru
>  		goto exit_unregister_sysfs;
>  
>  	if (start_reg & 0x01) {
> -		err = f71882fg_create_sysfs_files(pdev, f718x2fg_in_temp_attr,
> -					ARRAY_SIZE(f718x2fg_in_temp_attr));
> -		if (err)
> -			goto exit_unregister_sysfs;
> -
> -		if (data->type == f71882fg) {
> +		switch (data->type) {
> +		case f71882fg:
>  			err = f71882fg_create_sysfs_files(pdev,
>  					f71882fg_in_temp_attr,
>  					ARRAY_SIZE(f71882fg_in_temp_attr));
>  			if (err)
>  				goto exit_unregister_sysfs;
> +			/* fall through! */
> +		case f71862fg:
> +			err = f71882fg_create_sysfs_files(pdev,
> +					f718x2fg_in_temp_attr,
> +					ARRAY_SIZE(f718x2fg_in_temp_attr));
> +			break;
> +		case f8000:
> +			err = f71882fg_create_sysfs_files(pdev,
> +					f8000_in_temp_attr,
> +					ARRAY_SIZE(f8000_in_temp_attr));
> +			break;
>  		}
> +		if (err)
> +			goto exit_unregister_sysfs;
>  	}
>  
>  	if (start_reg & 0x02) {
> -		err = f71882fg_create_sysfs_files(pdev, f718x2fg_fan_attr,
> -					ARRAY_SIZE(f718x2fg_fan_attr));
> +		err = f71882fg_create_sysfs_files(pdev, fxxxx_fan_attr,
> +					ARRAY_SIZE(fxxxx_fan_attr));
>  		if (err)
>  			goto exit_unregister_sysfs;
>  
> -		if (data->type == f71862fg) {
> +		switch (data->type) {
> +		case f71862fg:
>  			err = f71882fg_create_sysfs_files(pdev,
>  					f71862fg_fan_attr,
>  					ARRAY_SIZE(f71862fg_fan_attr));
> -		} else {
> +			break;
> +		case f71882fg:
>  			err = f71882fg_create_sysfs_files(pdev,
>  					f71882fg_fan_attr,
>  					ARRAY_SIZE(f71882fg_fan_attr));
> +			break;
> +		case f8000:
> +			err = f71882fg_create_sysfs_files(pdev,
> +					f8000_fan_attr,
> +					ARRAY_SIZE(f8000_fan_attr));
> +			break;
>  		}
>  		if (err)
>  			goto exit_unregister_sysfs;
> @@ -1575,6 +1799,8 @@ static int f71882fg_remove(struct platfo
>  	if (data->hwmon_dev)
>  		hwmon_device_unregister(data->hwmon_dev);
>  
> +	/* Note we are not looping over all attr arrays we have as the ones
> +	   below are supersets of the ones skipped. */
>  	device_remove_file(&pdev->dev, &dev_attr_name);
>  
>  	for (i = 0; i < ARRAY_SIZE(f718x2fg_in_temp_attr); i++)
> @@ -1585,15 +1811,15 @@ static int f71882fg_remove(struct platfo
>  		device_remove_file(&pdev->dev,
>  					&f71882fg_in_temp_attr[i].dev_attr);
>  
> -	for (i = 0; i < ARRAY_SIZE(f718x2fg_fan_attr); i++)
> -		device_remove_file(&pdev->dev, &f718x2fg_fan_attr[i].dev_attr);
> -
> -	for (i = 0; i < ARRAY_SIZE(f71862fg_fan_attr); i++)
> -		device_remove_file(&pdev->dev, &f71862fg_fan_attr[i].dev_attr);
> +	for (i = 0; i < ARRAY_SIZE(fxxxx_fan_attr); i++)
> +		device_remove_file(&pdev->dev, &fxxxx_fan_attr[i].dev_attr);
>  
>  	for (i = 0; i < ARRAY_SIZE(f71882fg_fan_attr); i++)
>  		device_remove_file(&pdev->dev, &f71882fg_fan_attr[i].dev_attr);
>  
> +	for (i = 0; i < ARRAY_SIZE(f8000_fan_attr); i++)
> +		device_remove_file(&pdev->dev, &f8000_fan_attr[i].dev_attr);
> +
>  	kfree(data);
>  
>  	return 0;
> @@ -1621,6 +1847,9 @@ static int __init f71882fg_find(int sioa
>  	case SIO_F71882_ID:
>  		sio_data->type = f71882fg;
>  		break;
> +	case SIO_F8000_ID:
> +		sio_data->type = f8000;
> +		break;
>  	default:
>  		printk(KERN_INFO DRVNAME ": Unsupported Fintek device\n");
>  		goto exit;

Looks pretty good overall, so I've applied it already.

-- 
Jean Delvare




[Index of Archives]     [Linux Kernel]     [Linux Hardware Monitoring]     [Linux USB Devel]     [Linux Audio Users]     [Linux Kernel]     [Linux SCSI]     [Yosemite Backpacking]

  Powered by Linux