Pull sbs into release branch
[firefly-linux-kernel-4.4.55.git] / drivers / hwmon / w83627ehf.c
index 2157c3424de1e013cc620527f63c2a29dde4e6e0..c51ae2e17758591236721d5b99be8ce2ffea888c 100644 (file)
@@ -45,6 +45,7 @@
 #include <linux/platform_device.h>
 #include <linux/hwmon.h>
 #include <linux/hwmon-sysfs.h>
+#include <linux/hwmon-vid.h>
 #include <linux/err.h>
 #include <linux/mutex.h>
 #include <asm/io.h>
@@ -68,8 +69,11 @@ static const char * w83627ehf_device_names[] = {
 
 #define SIO_REG_LDSEL          0x07    /* Logical device select */
 #define SIO_REG_DEVID          0x20    /* Device ID (2 bytes) */
+#define SIO_REG_EN_VRM10       0x2C    /* GPIO3, GPIO4 selection */
 #define SIO_REG_ENABLE         0x30    /* Logical device enable */
 #define SIO_REG_ADDR           0x60    /* Logical device address (2 bytes) */
+#define SIO_REG_VID_CTRL       0xF0    /* VID control */
+#define SIO_REG_VID_DATA       0xF1    /* VID data */
 
 #define SIO_W83627EHF_ID       0x8850
 #define SIO_W83627EHG_ID       0x8860
@@ -268,6 +272,7 @@ struct w83627ehf_data {
        u8 fan_min[5];
        u8 fan_div[5];
        u8 has_fan;             /* some fan inputs can be disabled */
+       u8 temp_type[3];
        s8 temp1;
        s8 temp1_max;
        s8 temp1_max_hyst;
@@ -285,6 +290,9 @@ struct w83627ehf_data {
 
        u8 fan_min_output[4]; /* minimum fan speed */
        u8 fan_stop_time[4];
+
+       u8 vid;
+       u8 vrm;
 };
 
 struct w83627ehf_sio_data {
@@ -421,7 +429,7 @@ static struct w83627ehf_data *w83627ehf_update_device(struct device *dev)
 
        mutex_lock(&data->update_lock);
 
-       if (time_after(jiffies, data->last_updated + HZ)
+       if (time_after(jiffies, data->last_updated + HZ + HZ/2)
         || !data->valid) {
                /* Fan clock dividers */
                i = w83627ehf_read_value(data, W83627EHF_REG_FANDIV1);
@@ -727,6 +735,8 @@ store_fan_min(struct device *dev, struct device_attribute *attr,
                        div_from_reg(new_div));
                data->fan_div[nr] = new_div;
                w83627ehf_write_fan_div(data, nr);
+               /* Give the chip time to sample a new speed value */
+               data->last_updated = jiffies;
        }
        w83627ehf_write_value(data, W83627EHF_REG_FAN_MIN[nr],
                              data->fan_min[nr]);
@@ -837,6 +847,15 @@ store_##reg(struct device *dev, struct device_attribute *attr, \
 store_temp_reg(OVER, temp_max);
 store_temp_reg(HYST, temp_max_hyst);
 
+static ssize_t
+show_temp_type(struct device *dev, struct device_attribute *attr, char *buf)
+{
+       struct w83627ehf_data *data = w83627ehf_update_device(dev);
+       struct sensor_device_attribute *sensor_attr = to_sensor_dev_attr(attr);
+       int nr = sensor_attr->index;
+       return sprintf(buf, "%d\n", (int)data->temp_type[nr]);
+}
+
 static struct sensor_device_attribute sda_temp[] = {
        SENSOR_ATTR(temp1_input, S_IRUGO, show_temp1, NULL, 0),
        SENSOR_ATTR(temp2_input, S_IRUGO, show_temp, NULL, 0),
@@ -856,6 +875,9 @@ static struct sensor_device_attribute sda_temp[] = {
        SENSOR_ATTR(temp1_alarm, S_IRUGO, show_alarm, NULL, 4),
        SENSOR_ATTR(temp2_alarm, S_IRUGO, show_alarm, NULL, 5),
        SENSOR_ATTR(temp3_alarm, S_IRUGO, show_alarm, NULL, 13),
+       SENSOR_ATTR(temp1_type, S_IRUGO, show_temp_type, NULL, 0),
+       SENSOR_ATTR(temp2_type, S_IRUGO, show_temp_type, NULL, 1),
+       SENSOR_ATTR(temp3_type, S_IRUGO, show_temp_type, NULL, 2),
 };
 
 #define show_pwm_reg(reg) \
@@ -1125,6 +1147,14 @@ static struct sensor_device_attribute sda_sf3_arrays[] = {
                    store_fan_min_output, 2),
 };
 
+static ssize_t
+show_vid(struct device *dev, struct device_attribute *attr, char *buf)
+{
+       struct w83627ehf_data *data = dev_get_drvdata(dev);
+       return sprintf(buf, "%d\n", vid_from_reg(data->vid, data->vrm));
+}
+static DEVICE_ATTR(cpu0_vid, S_IRUGO, show_vid, NULL);
+
 /*
  * Driver and device management
  */
@@ -1163,13 +1193,15 @@ static void w83627ehf_device_remove_files(struct device *dev)
                device_remove_file(dev, &sda_temp[i].dev_attr);
 
        device_remove_file(dev, &dev_attr_name);
+       if (data->vid != 0x3f)
+               device_remove_file(dev, &dev_attr_cpu0_vid);
 }
 
 /* Get the monitoring functions started */
 static inline void __devinit w83627ehf_init_device(struct w83627ehf_data *data)
 {
        int i;
-       u8 tmp;
+       u8 tmp, diode;
 
        /* Start monitoring is needed */
        tmp = w83627ehf_read_value(data, W83627EHF_REG_CONFIG);
@@ -1186,6 +1218,20 @@ static inline void __devinit w83627ehf_init_device(struct w83627ehf_data *data)
                                              W83627EHF_REG_TEMP_CONFIG[i],
                                              tmp & 0xfe);
        }
+
+       /* Enable VBAT monitoring if needed */
+       tmp = w83627ehf_read_value(data, W83627EHF_REG_VBAT);
+       if (!(tmp & 0x01))
+               w83627ehf_write_value(data, W83627EHF_REG_VBAT, tmp | 0x01);
+
+       /* Get thermal sensor types */
+       diode = w83627ehf_read_value(data, W83627EHF_REG_DIODE);
+       for (i = 0; i < 3; i++) {
+               if ((tmp & (0x02 << i)))
+                       data->temp_type[i] = (diode & (0x10 << i)) ? 1 : 2;
+               else
+                       data->temp_type[i] = 4; /* thermistor */
+       }
 }
 
 static int __devinit w83627ehf_probe(struct platform_device *pdev)
@@ -1194,7 +1240,7 @@ static int __devinit w83627ehf_probe(struct platform_device *pdev)
        struct w83627ehf_sio_data *sio_data = dev->platform_data;
        struct w83627ehf_data *data;
        struct resource *res;
-       u8 fan4pin, fan5pin;
+       u8 fan4pin, fan5pin, en_vrm10;
        int i, err = 0;
 
        res = platform_get_resource(pdev, IORESOURCE_IO, 0);
@@ -1223,14 +1269,32 @@ static int __devinit w83627ehf_probe(struct platform_device *pdev)
        /* Initialize the chip */
        w83627ehf_init_device(data);
 
-       /* A few vars need to be filled upon startup */
-       for (i = 0; i < 5; i++)
-               data->fan_min[i] = w83627ehf_read_value(data,
-                                  W83627EHF_REG_FAN_MIN[i]);
+       data->vrm = vid_which_vrm();
+       superio_enter(sio_data->sioreg);
+       /* Set VID input sensibility if needed. In theory the BIOS should
+          have set it, but in practice it's not always the case. */
+       en_vrm10 = superio_inb(sio_data->sioreg, SIO_REG_EN_VRM10);
+       if ((en_vrm10 & 0x08) && data->vrm != 100) {
+               dev_warn(dev, "Setting VID input voltage to TTL\n");
+               superio_outb(sio_data->sioreg, SIO_REG_EN_VRM10,
+                            en_vrm10 & ~0x08);
+       } else if (!(en_vrm10 & 0x08) && data->vrm == 100) {
+               dev_warn(dev, "Setting VID input voltage to VRM10\n");
+               superio_outb(sio_data->sioreg, SIO_REG_EN_VRM10,
+                            en_vrm10 | 0x08);
+       }
+       /* Read VID value */
+       superio_select(sio_data->sioreg, W83627EHF_LD_HWM);
+       if (superio_inb(sio_data->sioreg, SIO_REG_VID_CTRL) & 0x80)
+               data->vid = superio_inb(sio_data->sioreg, SIO_REG_VID_DATA) & 0x3f;
+       else {
+               dev_info(dev, "VID pins in output mode, CPU VID not "
+                        "available\n");
+               data->vid = 0x3f;
+       }
 
        /* fan4 and fan5 share some pins with the GPIO and serial flash */
 
-       superio_enter(sio_data->sioreg);
        fan5pin = superio_inb(sio_data->sioreg, 0x24) & 0x2;
        fan4pin = superio_inb(sio_data->sioreg, 0x29) & 0x6;
        superio_exit(sio_data->sioreg);
@@ -1306,6 +1370,12 @@ static int __devinit w83627ehf_probe(struct platform_device *pdev)
        if (err)
                goto exit_remove;
 
+       if (data->vid != 0x3f) {
+               err = device_create_file(dev, &dev_attr_cpu0_vid);
+               if (err)
+                       goto exit_remove;
+       }
+
        data->class_dev = hwmon_device_register(dev);
        if (IS_ERR(data->class_dev)) {
                err = PTR_ERR(data->class_dev);
@@ -1375,8 +1445,9 @@ static int __init w83627ehf_find(int sioaddr, unsigned short *addr,
                sio_name = sio_name_W83627DHG;
                break;
        default:
-               pr_info(DRVNAME ": unsupported chip ID: 0x%04x\n",
-                       val);
+               if (val != 0xffff)
+                       pr_debug(DRVNAME ": unsupported chip ID: 0x%04x\n",
+                                val);
                superio_exit(sioaddr);
                return -ENODEV;
        }
@@ -1387,14 +1458,19 @@ static int __init w83627ehf_find(int sioaddr, unsigned short *addr,
            | superio_inb(sioaddr, SIO_REG_ADDR + 1);
        *addr = val & IOREGION_ALIGNMENT;
        if (*addr == 0) {
+               printk(KERN_ERR DRVNAME ": Refusing to enable a Super-I/O "
+                      "device with a base I/O port 0.\n");
                superio_exit(sioaddr);
                return -ENODEV;
        }
 
        /* Activate logical device if needed */
        val = superio_inb(sioaddr, SIO_REG_ENABLE);
-       if (!(val & 0x01))
+       if (!(val & 0x01)) {
+               printk(KERN_WARNING DRVNAME ": Forcibly enabling Super-I/O. "
+                      "Sensor is probably unusable.\n");
                superio_outb(sioaddr, SIO_REG_ENABLE, val | 0x01);
+       }
 
        superio_exit(sioaddr);
        pr_info(DRVNAME ": Found %s chip at %#x\n", sio_name, *addr);