net: phy: marvell: mv88e6390 temperature sensor reading
authorAndrew Lunn <andrew@lunn.ch>
Tue, 9 Jan 2018 21:42:09 +0000 (22:42 +0100)
committerDavid S. Miller <davem@davemloft.net>
Wed, 10 Jan 2018 20:35:05 +0000 (15:35 -0500)
The internal PHYs in the mv88e6390 switch have a temperature sensor.
It uses a different register layout to other PHY currently supported.
It also has an errata, in that some reads of the sensor result in bad
values. So a number of reads need to be made, and the average taken.

Signed-off-by: Andrew Lunn <andrew@lunn.ch>
Reviewed-by: Florian Fainelli <f.fainelli@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
drivers/net/phy/marvell.c

index fd66304f18b7fbe1aff9ed94744550b8f5bbf495..22d9bc9c33a4bce864505babe6b3060348542a62 100644 (file)
 #define MII_88E1510_TEMP_SENSOR                0x1b
 #define MII_88E1510_TEMP_SENSOR_MASK   0xff
 
+#define MII_88E6390_MISC_TEST          0x1b
+#define MII_88E6390_MISC_TEST_SAMPLE_1S                0
+#define MII_88E6390_MISC_TEST_SAMPLE_10MS      BIT(14)
+#define MII_88E6390_MISC_TEST_SAMPLE_DISABLE   BIT(15)
+#define MII_88E6390_MISC_TEST_SAMPLE_ENABLE    0
+#define MII_88E6390_MISC_TEST_SAMPLE_MASK      (0x3 << 14)
+
+#define MII_88E6390_TEMP_SENSOR                0x1c
+#define MII_88E6390_TEMP_SENSOR_MASK   0xff
+#define MII_88E6390_TEMP_SENSOR_SAMPLES 10
+
 #define MII_88E1318S_PHY_MSCR1_REG     16
 #define MII_88E1318S_PHY_MSCR1_PAD_ODD BIT(6)
 
@@ -1738,6 +1749,123 @@ static const struct hwmon_chip_info m88e1510_hwmon_chip_info = {
        .info = m88e1510_hwmon_info,
 };
 
+static int m88e6390_get_temp(struct phy_device *phydev, long *temp)
+{
+       int sum = 0;
+       int oldpage;
+       int ret = 0;
+       int i;
+
+       *temp = 0;
+
+       oldpage = phy_select_page(phydev, MII_MARVELL_MISC_TEST_PAGE);
+       if (oldpage < 0)
+               goto error;
+
+       /* Enable temperature sensor */
+       ret = __phy_read(phydev, MII_88E6390_MISC_TEST);
+       if (ret < 0)
+               goto error;
+
+       ret = ret & ~MII_88E6390_MISC_TEST_SAMPLE_MASK;
+       ret |= MII_88E6390_MISC_TEST_SAMPLE_ENABLE |
+               MII_88E6390_MISC_TEST_SAMPLE_1S;
+
+       ret = __phy_write(phydev, MII_88E6390_MISC_TEST, ret);
+       if (ret < 0)
+               goto error;
+
+       /* Wait for temperature to stabilize */
+       usleep_range(10000, 12000);
+
+       /* Reading the temperature sense has an errata. You need to read
+        * a number of times and take an average.
+        */
+       for (i = 0; i < MII_88E6390_TEMP_SENSOR_SAMPLES; i++) {
+               ret = __phy_read(phydev, MII_88E6390_TEMP_SENSOR);
+               if (ret < 0)
+                       goto error;
+               sum += ret & MII_88E6390_TEMP_SENSOR_MASK;
+       }
+
+       sum /= MII_88E6390_TEMP_SENSOR_SAMPLES;
+       *temp = (sum  - 75) * 1000;
+
+       /* Disable temperature sensor */
+       ret = __phy_read(phydev, MII_88E6390_MISC_TEST);
+       if (ret < 0)
+               goto error;
+
+       ret = ret & ~MII_88E6390_MISC_TEST_SAMPLE_MASK;
+       ret |= MII_88E6390_MISC_TEST_SAMPLE_DISABLE;
+
+       ret = __phy_write(phydev, MII_88E6390_MISC_TEST, ret);
+
+error:
+       phy_restore_page(phydev, oldpage, ret);
+
+       return ret;
+}
+
+static int m88e6390_hwmon_read(struct device *dev,
+                              enum hwmon_sensor_types type,
+                              u32 attr, int channel, long *temp)
+{
+       struct phy_device *phydev = dev_get_drvdata(dev);
+       int err;
+
+       switch (attr) {
+       case hwmon_temp_input:
+               err = m88e6390_get_temp(phydev, temp);
+               break;
+       default:
+               return -EOPNOTSUPP;
+       }
+
+       return err;
+}
+
+static umode_t m88e6390_hwmon_is_visible(const void *data,
+                                        enum hwmon_sensor_types type,
+                                        u32 attr, int channel)
+{
+       if (type != hwmon_temp)
+               return 0;
+
+       switch (attr) {
+       case hwmon_temp_input:
+               return 0444;
+       default:
+               return 0;
+       }
+}
+
+static u32 m88e6390_hwmon_temp_config[] = {
+       HWMON_T_INPUT,
+       0
+};
+
+static const struct hwmon_channel_info m88e6390_hwmon_temp = {
+       .type = hwmon_temp,
+       .config = m88e6390_hwmon_temp_config,
+};
+
+static const struct hwmon_channel_info *m88e6390_hwmon_info[] = {
+       &m88e1121_hwmon_chip,
+       &m88e6390_hwmon_temp,
+       NULL
+};
+
+static const struct hwmon_ops m88e6390_hwmon_hwmon_ops = {
+       .is_visible = m88e6390_hwmon_is_visible,
+       .read = m88e6390_hwmon_read,
+};
+
+static const struct hwmon_chip_info m88e6390_hwmon_chip_info = {
+       .ops = &m88e6390_hwmon_hwmon_ops,
+       .info = m88e6390_hwmon_info,
+};
+
 static int marvell_hwmon_name(struct phy_device *phydev)
 {
        struct marvell_priv *priv = phydev->priv;
@@ -1784,6 +1912,11 @@ static int m88e1510_hwmon_probe(struct phy_device *phydev)
 {
        return marvell_hwmon_probe(phydev, &m88e1510_hwmon_chip_info);
 }
+
+static int m88e6390_hwmon_probe(struct phy_device *phydev)
+{
+       return marvell_hwmon_probe(phydev, &m88e6390_hwmon_chip_info);
+}
 #else
 static int m88e1121_hwmon_probe(struct phy_device *phydev)
 {
@@ -1794,6 +1927,11 @@ static int m88e1510_hwmon_probe(struct phy_device *phydev)
 {
        return 0;
 }
+
+static int m88e6390_hwmon_probe(struct phy_device *phydev)
+{
+       return 0;
+}
 #endif
 
 static int marvell_probe(struct phy_device *phydev)
@@ -1831,6 +1969,17 @@ static int m88e1510_probe(struct phy_device *phydev)
        return m88e1510_hwmon_probe(phydev);
 }
 
+static int m88e6390_probe(struct phy_device *phydev)
+{
+       int err;
+
+       err = marvell_probe(phydev);
+       if (err)
+               return err;
+
+       return m88e6390_hwmon_probe(phydev);
+}
+
 static struct phy_driver marvell_drivers[] = {
        {
                .phy_id = MARVELL_PHY_ID_88E1101,
@@ -2122,7 +2271,7 @@ static struct phy_driver marvell_drivers[] = {
                .name = "Marvell 88E6390",
                .features = PHY_GBIT_FEATURES,
                .flags = PHY_HAS_INTERRUPT,
-               .probe = m88e1510_probe,
+               .probe = m88e6390_probe,
                .config_init = &marvell_config_init,
                .config_aneg = &m88e1510_config_aneg,
                .read_status = &marvell_read_status,