On Thu, 07 Aug 2014, Wei-Chun Pan wrote: > Sorry, I sent wrong patch just now. Please ignore the mail "[PATCH 1/3] > imanager2: rename io functions and remove no used functions". > > This mail include 2 patches: > The first patch is shown I rename some functions. > The second patch is shown I moditfy code according to your comment. > > Signed-off-by: Wei-Chun Pan <weichun....@advantech.com.tw>
Please use Git to send patches. All of the above is in the wrong order and is missing lots of context. Please read: Documentation/SubmittingPatches Documentation/email-clients.txt > --- > diff --git a/drivers/hwmon/imanager2_hwm.c b/drivers/hwmon/imanager2_hwm.c > index 335bffb..ab63296 100644 > --- a/drivers/hwmon/imanager2_hwm.c > +++ b/drivers/hwmon/imanager2_hwm.c > @@ -119,17 +119,17 @@ static int imanager2_volt_get_value_by_io(struct > imanager2 *ec, int index, > u8 portnum; > int ret; > > - ret = imanager2_mbox_io_read(EC_CMD_ADC_INDEX, pin, &portnum, 1); > + ret = imanager2_io_read_data(EC_CMD_ADC_INDEX, pin, &portnum, 1); > if (ret) > return ret; > if (portnum == EC_ERROR) > return -ENXIO; > > - ret = imanager2_mbox_io_simple_read(EC_CMD_ADC_READ_LSB, &buf[1]); > + ret = imanager2_io_nooffset_readbyte(EC_CMD_ADC_READ_LSB, &buf[1]); > if (ret) > return ret; > > - return imanager2_mbox_io_simple_read(EC_CMD_ADC_READ_MSB, &buf[0]); > + return imanager2_io_nooffset_readbyte(EC_CMD_ADC_READ_MSB, &buf[0]); > } [...] > --- > > > --- > > > drivers/hwmon/Kconfig | 5 + > > > drivers/hwmon/Makefile | 1 + > > > drivers/hwmon/imanager2_hwm.c | 768 > > ++++++++++++++++++++++++++++++++++++++++++ > > > > Documentation/hwmon/imanager2_hwm missing. WTF? I'm confused my this and even more so with the stuff below this. What's the idea of this email? Are you sending a patch or replying to a code review? You can't do both in one submission. > diff --git a/Documentation/hwmon/imanager2_hwm > b/Documentation/hwmon/imanager2_hwm > new file mode 100644 > index 0000000..bf3d0b5 > --- /dev/null > +++ b/Documentation/hwmon/imanager2_hwm > @@ -0,0 +1,42 @@ > +Kernel driver imanager2_hwm > +=========================== > + > +Supported chips: > + * ITE IT8516 > + Prefix: 'it8516' > + Addresses scanned: I/O chennel 0x029A/0x0299, > + IET mailbox chennel 0x029E/0x029F > + Datasheet: Not publicly available > + * ITE IT8518 > + Prefix: 'it8518' > + Addresses scanned: I/O chennel 0x029A/0x0299, > + IET mailbox chennel 0x029E/0x029F > + Datasheet: Not publicly available > + * ITE IT8528 > + Prefix: 'it8528' > + Addresses scanned: I/O chennel 0x029A/0x0299 > + Datasheet: Not publicly available > + > +Authors: > + Richard Vidal-Dorsch <richard.dor...@advantech.com> > + > + > +Description > +----------- > + > +This driver supports the hardware monitoring features of the IT8516, IT8518, > and > +IT8528 chips. These features include 11 voltage sensors, 1 current sensor, 4 > +temperature sensors, and 3 fan rotation speed sensors. > + > +ITE IT8516, IT8518, and IT8528 are are 'EC chips'. These chips are like > Super I/O control boards. For IT8516 and IT 8518 The control chennel can be > I/O or ITE mailbox chennel. I/O chennel is a common way but ITE mailbox > chennel performance faster since it does not need to wait IBF (input buffer > full) and OBF (output buffer full) to before send or get data. > + > +ITE IT8528 use an I/O chennel way to access mailbox, called I/O mailbox for > cost down. Its performance the between pure I/O controller and ITE mailbox > controller. > + > + > +sysfs-Interface > +--------------- > + > +in[0-11]_input - adc voltage input > +curr1_input - adc current input > +temp[1-4]_input - temperature input > +fan[1-3]_input - fan speed input > > > > > > 3 files changed, 774 insertions(+) > > > create mode 100644 drivers/hwmon/imanager2_hwm.c > > > > > > diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig > > > index bc196f4..7524fc3 100644 > > > --- a/drivers/hwmon/Kconfig > > > +++ b/drivers/hwmon/Kconfig > > > @@ -39,6 +39,11 @@ config HWMON_DEBUG_CHIP > > > > > > comment "Native drivers" > > > > > > +config SENSORS_IMANAGER2 > > > + tristate "Support for Advantech iManager2 EC H.W. Monitor" > > > + select MFD_CORE > > > + depends on MFD_IMANAGER2 > > > + > > Alphabetic order please. > > > > > config SENSORS_AB8500 > > > tristate "AB8500 thermal monitoring" > > > depends on AB8500_GPADC && AB8500_BM > > > diff --git a/drivers/hwmon/Makefile b/drivers/hwmon/Makefile > > > index c48f987..a2c8f07 100644 > > > --- a/drivers/hwmon/Makefile > > > +++ b/drivers/hwmon/Makefile > > > @@ -146,6 +146,7 @@ obj-$(CONFIG_SENSORS_W83L785TS) += w83l785ts.o > > > obj-$(CONFIG_SENSORS_W83L786NG) += w83l786ng.o > > > obj-$(CONFIG_SENSORS_WM831X) += wm831x-hwmon.o > > > obj-$(CONFIG_SENSORS_WM8350) += wm8350-hwmon.o > > > +obj-$(CONFIG_SENSORS_IMANAGER2) += imanager2_hwm.o > > > > > Alphabetic order please. > > > drivers/hwmon/Kconfig | 10 +++++----- > drivers/hwmon/Makefile | 2 +- > 2 files changed, 6 insertions(+), 6 deletions(-) > > diff --git a/drivers/hwmon/Kconfig b/drivers/hwmon/Kconfig > index 7524fc3..e39f8e0 100644 > --- a/drivers/hwmon/Kconfig > +++ b/drivers/hwmon/Kconfig > @@ -39,11 +39,6 @@ config HWMON_DEBUG_CHIP > > comment "Native drivers" > > -config SENSORS_IMANAGER2 > - tristate "Support for Advantech iManager2 EC H.W. Monitor" > - select MFD_CORE > - depends on MFD_IMANAGER2 > - > config SENSORS_AB8500 > tristate "AB8500 thermal monitoring" > depends on AB8500_GPADC && AB8500_BM > @@ -576,6 +571,11 @@ config SENSORS_CORETEMP > sensor inside your CPU. Most of the family 6 CPUs > are supported. Check Documentation/hwmon/coretemp for details. > > +config SENSORS_IMANAGER2 > + tristate "Support for Advantech iManager2 EC H.W. Monitor" > + select MFD_CORE > + depends on MFD_IMANAGER2 > + > config SENSORS_IT87 > tristate "ITE IT87xx and compatibles" > depends on !PPC > diff --git a/drivers/hwmon/Makefile b/drivers/hwmon/Makefile > index a2c8f07..564b6fe 100644 > --- a/drivers/hwmon/Makefile > +++ b/drivers/hwmon/Makefile > @@ -72,6 +72,7 @@ obj-$(CONFIG_SENSORS_I5K_AMB) += i5k_amb.o > obj-$(CONFIG_SENSORS_IBMAEM) += ibmaem.o > obj-$(CONFIG_SENSORS_IBMPEX) += ibmpex.o > obj-$(CONFIG_SENSORS_IIO_HWMON) += iio_hwmon.o > +obj-$(CONFIG_SENSORS_IMANAGER2) += imanager2_hwm.o > obj-$(CONFIG_SENSORS_INA209) += ina209.o > obj-$(CONFIG_SENSORS_INA2XX) += ina2xx.o > obj-$(CONFIG_SENSORS_IT87) += it87.o > @@ -146,7 +147,6 @@ obj-$(CONFIG_SENSORS_W83L785TS) += w83l785ts.o > obj-$(CONFIG_SENSORS_W83L786NG) += w83l786ng.o > obj-$(CONFIG_SENSORS_WM831X) += wm831x-hwmon.o > obj-$(CONFIG_SENSORS_WM8350) += wm8350-hwmon.o > -obj-$(CONFIG_SENSORS_IMANAGER2) += imanager2_hwm.o > > obj-$(CONFIG_PMBUS) += pmbus/ > > > > obj-$(CONFIG_PMBUS) += pmbus/ > > > > > > diff --git a/drivers/hwmon/imanager2_hwm.c > > [...] > > > b/drivers/hwmon/imanager2_hwm.c > > > new file mode 100644 > > > index 0000000..335bffb > > > --- /dev/null > > > +++ b/drivers/hwmon/imanager2_hwm.c > > > @@ -0,0 +1,768 @@ > > > +/* > > > + * imanager2_hwm.c - HW Monitoring interface for Advantech EC > > IT8516/18/28 > > > + * Copyright (C) 2014 Richard Vidal-Dorsch > > > <richard.dor...@advantech.com> > > > + * > > > + * This program is free software: you can redistribute it and/or modify > > > + * it under the terms of the GNU General Public License as published by > > > + * the Free Software Foundation, either version 3 of the License, or > > > + * (at your option) any later version. > > > + * > > > + * This program is distributed in the hope that it will be useful, > > > + * but WITHOUT ANY WARRANTY; without even the implied warranty of > > > + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the > > > + * GNU General Public License for more details. > > > + * > > > + * You should have received a copy of the GNU General Public License > > > + * along with this program. If not, see <http://www.gnu.org/licenses/>. > > > + */ > > > + > > > +#include <linux/module.h> > > > +#include <linux/platform_device.h> > > > +#include <linux/hwmon.h> > > > +#include <linux/hwmon-sysfs.h> > > > +#include <linux/mfd/imanager2_ec.h> > > > + > > > +#define DRV_NAME "imanager2_hwm" > > > +#define DRV_VERSION "4.0.1" > > > + > > > +/* ADC */ > > > +#define EC_ADC_RESOLUTION_MAX 0x03FF /* 10-bit */ > > > +#define EC_ADC_VALUE_MAX 3000 /* max: 3000 mV or mA */ > > > + > > > +/* Thermal */ > > > +#define EC_THERMAL_ZONE_MAX 4 > > > + > > > +enum thermaltype { > > > + none, > > > + sys1, > > > + cpu, > > > + sys3, > > > + sys2, > > > +}; > > > + > > > +struct ec_thermalzone { > > > + u8 channel, > > > + addr, > > > + cmd, > > > + status, > > > + fancode, > > > + temp; > > > +}; > > > + > > > +/* Tacho */ > > > +#define EC_MAX_IO_FAN 3 > > > + > > > +/* Voltage */ > > > +struct volt_item { > > > + u8 did; > > > + const char *name; > > > + int factor; > > > + bool visible; > > > +}; > > > + > > > +static struct volt_item ec_volt_table[] = { > > > + { > > > + .did = adcmosbat, > > > + .name = "BAT CMOS", > > > + }, > > > + { > > > + .did = adcbat, > > > + .name = "BAT", > > > + }, > > > + { > > > + .did = adc5vs0, > > > + .name = "5V S0", > > > + }, > > > + { > > > + .did = adv5vs5, > > > + .name = "5V S5", > > > + }, > > > + { > > > + .did = adc33vs0, > > > + .name = "3V3 S0", > > > + }, > > > + { > > > + .did = adc33vs5, > > > + .name = "3V3 S5", > > > + }, > > > + { > > > + .did = adv12vs0, > > > + .name = "12V S0", > > > + }, > > > + { > > > + .did = adcvcorea, > > > + .name = "Vcore A", > > > + }, > > > + { > > > + .did = adcvcoreb, > > > + .name = "Vcore B", > > > + }, > > > + { > > > + .did = adcdc, > > > + .name = "DC", > > > + }, > > > + { > > > + .did = adcdcstby, > > > + .name = "DC Standby", > > > + }, > > > + { > > > + .did = adcdcother, > > > + .name = "DC Other", > > > + }, > > > +}; > > > + > > > +static int imanager2_volt_get_value_by_io(struct imanager2 *ec, int > > > index, > > > + u8 *buf) > > > +{ > > > + u8 item = ec->table.devid2itemnum[ec_volt_table[index].did]; > > > + u8 pin = ec->table.pinnum[item]; > > > + u8 portnum; > > > + int ret; > > > + > > > + ret = imanager2_mbox_io_read(EC_CMD_ADC_INDEX, pin, &portnum, 1); > > > + if (ret) > > > + return ret; > > > + if (portnum == EC_ERROR) > > > + return -ENXIO; > > > + > > > + ret = imanager2_mbox_io_simple_read(EC_CMD_ADC_READ_LSB, &buf[1]); > > > + if (ret) > > > + return ret; > > > + > > > + return imanager2_mbox_io_simple_read(EC_CMD_ADC_READ_MSB, > > &buf[0]); > > > +} > > > + > > > +static int imanager2_volt_get_value(struct imanager2 *ec, int index, > > > + u32 *volt_mvolt) > > > +{ > > > + int ret; > > > + u8 tmp[2]; > > > + > > > + mutex_lock(&ec->lock); > > > + > > > + if (ec->flag & EC_FLAG_MAILBOX) > > > + ret = imanager2_mbox_read_data(ec->flag, > > > + EC_CMD_MAILBOX_READ_HW_PIN, > > > + ec_volt_table[index].did, > > > + tmp, 2); > > > + else > > > + ret = imanager2_volt_get_value_by_io(ec, index, tmp); > > > + > > > + mutex_unlock(&ec->lock); > > > + > > > + if (ret) > > > + return ret; > > > + > > > + *volt_mvolt = (((tmp[0] << 8) | tmp[1]) & EC_ADC_RESOLUTION_MAX) * > > > + ec_volt_table[index].factor * > > > + DIV_ROUND_CLOSEST(EC_ADC_VALUE_MAX, > > > + EC_ADC_RESOLUTION_MAX); > > > + > > > + return 0; > > > +} > > > + > > > +static void imanager2_volt_init(struct imanager2 *ec) > > > +{ > > > + u8 did; > > > + int i; > > > + > > > + for (i = 0; i < ARRAY_SIZE(ec_volt_table); i++) { > > > + did = ec_volt_table[i].did; > > > + > > > + if (ec->table.devid2itemnum[did] != EC_TABLE_ITEM_UNUSED) { > > > + ec_volt_table[i].factor = 1; > > > + ec_volt_table[i].visible = true; > > > + } else if (ec->table.devid2itemnum[did + 1] != > > > + EC_TABLE_ITEM_UNUSED) { > > > + ec_volt_table[i].did += 1; > > > + ec_volt_table[i].factor = 2; > > > + ec_volt_table[i].visible = true; > > > + } else if (ec->table.devid2itemnum[did + 2] != > > > + EC_TABLE_ITEM_UNUSED) { > > > + ec_volt_table[i].did += 2; > > > + ec_volt_table[i].factor = 10; > > > + ec_volt_table[i].visible = true; > > > + } else { > > > + ec_volt_table[i].visible = false; > > > + } > > > + } > > > +} > > > + > > > +static ssize_t show_in(struct device *dev, struct device_attribute > > > *dev_attr, > > > + char *buf) > > > +{ > > > + struct imanager2 *ec = dev_get_drvdata(dev); > > > + u32 val; > > > + int ret = imanager2_volt_get_value(ec, > > > + to_sensor_dev_attr(dev_attr)->index, > > > + &val); > > > + if (ret) > > > + return ret; > > > + > > > + return sprintf(buf, "%u\n", val); > > > +} > > > + > > > +static ssize_t show_in_label(struct device *dev, > > > + struct device_attribute *dev_attr, char *buf) > > > +{ > > > + return sprintf(buf, "%s\n", > > > + ec_volt_table[to_sensor_dev_attr(dev_attr)->index].name); > > > +} > > > + > > > +static SENSOR_DEVICE_ATTR(in0_input, S_IRUGO, show_in, NULL, 0); > > > +static SENSOR_DEVICE_ATTR(in1_input, S_IRUGO, show_in, NULL, 1); > > > +static SENSOR_DEVICE_ATTR(in2_input, S_IRUGO, show_in, NULL, 2); > > > +static SENSOR_DEVICE_ATTR(in3_input, S_IRUGO, show_in, NULL, 3); > > > +static SENSOR_DEVICE_ATTR(in4_input, S_IRUGO, show_in, NULL, 4); > > > +static SENSOR_DEVICE_ATTR(in5_input, S_IRUGO, show_in, NULL, 5); > > > +static SENSOR_DEVICE_ATTR(in6_input, S_IRUGO, show_in, NULL, 6); > > > +static SENSOR_DEVICE_ATTR(in7_input, S_IRUGO, show_in, NULL, 7); > > > +static SENSOR_DEVICE_ATTR(in8_input, S_IRUGO, show_in, NULL, 8); > > > +static SENSOR_DEVICE_ATTR(in9_input, S_IRUGO, show_in, NULL, 9); > > > +static SENSOR_DEVICE_ATTR(in10_input, S_IRUGO, show_in, NULL, 10); > > > +static SENSOR_DEVICE_ATTR(in11_input, S_IRUGO, show_in, NULL, 11); > > > + > > > +static SENSOR_DEVICE_ATTR(in0_label, S_IRUGO, show_in_label, NULL, 0); > > > +static SENSOR_DEVICE_ATTR(in1_label, S_IRUGO, show_in_label, NULL, 1); > > > +static SENSOR_DEVICE_ATTR(in2_label, S_IRUGO, show_in_label, NULL, 2); > > > +static SENSOR_DEVICE_ATTR(in3_label, S_IRUGO, show_in_label, NULL, 3); > > > +static SENSOR_DEVICE_ATTR(in4_label, S_IRUGO, show_in_label, NULL, 4); > > > +static SENSOR_DEVICE_ATTR(in5_label, S_IRUGO, show_in_label, NULL, 5); > > > +static SENSOR_DEVICE_ATTR(in6_label, S_IRUGO, show_in_label, NULL, 6); > > > +static SENSOR_DEVICE_ATTR(in7_label, S_IRUGO, show_in_label, NULL, 7); > > > +static SENSOR_DEVICE_ATTR(in8_label, S_IRUGO, show_in_label, NULL, 8); > > > +static SENSOR_DEVICE_ATTR(in9_label, S_IRUGO, show_in_label, NULL, 9); > > > +static SENSOR_DEVICE_ATTR(in10_label, S_IRUGO, show_in_label, NULL, 10); > > > +static SENSOR_DEVICE_ATTR(in11_label, S_IRUGO, show_in_label, NULL, 11); > > > + > > > +static struct attribute *imanager2_volt_attrs[] = { > > > + &sensor_dev_attr_in0_label.dev_attr.attr, > > > + &sensor_dev_attr_in0_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in1_label.dev_attr.attr, > > > + &sensor_dev_attr_in1_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in2_label.dev_attr.attr, > > > + &sensor_dev_attr_in2_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in3_label.dev_attr.attr, > > > + &sensor_dev_attr_in3_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in4_label.dev_attr.attr, > > > + &sensor_dev_attr_in4_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in5_label.dev_attr.attr, > > > + &sensor_dev_attr_in5_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in6_label.dev_attr.attr, > > > + &sensor_dev_attr_in6_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in7_label.dev_attr.attr, > > > + &sensor_dev_attr_in7_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in8_label.dev_attr.attr, > > > + &sensor_dev_attr_in8_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in9_label.dev_attr.attr, > > > + &sensor_dev_attr_in9_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in10_label.dev_attr.attr, > > > + &sensor_dev_attr_in10_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_in11_label.dev_attr.attr, > > > + &sensor_dev_attr_in11_input.dev_attr.attr, > > > + > > > + NULL > > > +}; > > > + > > > +static umode_t imanager2_volt_mode(struct kobject *kobj, struct attribute > > *attr, > > > + int index) > > > +{ > > > + struct device_attribute *dev_attr; > > > + > > > + dev_attr = container_of(attr, struct device_attribute, attr); > > > + if (!ec_volt_table[to_sensor_dev_attr(dev_attr)->index].visible) > > > + return 0; > > > + > > > + return attr->mode; > > > +} > > > + > > > +static const struct attribute_group imanager2_volt_group = { > > > + .attrs = imanager2_volt_attrs, > > > + .is_visible = imanager2_volt_mode, > > > +}; > > > + > > > +/* Current */ > > > +struct curr_item { > > > + const u8 did; > > > + const char *name; > > > + bool visible; > > > +}; > > > + > > > +static struct curr_item ec_curr_table[] = { > > > + { > > > + .did = adccurrent, > > > + .name = "IMON" > > > + }, > > > +}; > > > + > > > +static int imanager2_curr_get_value(struct imanager2 *ec, int index, > > > + u32 *curr_mamp) > > > +{ > > > + int ret; > > > + u8 tmp[5]; > > > + u16 value, factor; > > > + u32 baseunit = 1; > > > + > > > + mutex_lock(&ec->lock); > > > + ret = imanager2_mbox_read_data(ec->flag, > > EC_CMD_MAILBOX_READ_HW_PIN, > > > + ec_curr_table[index].did, tmp, > > > + ARRAY_SIZE(tmp)); > > > + mutex_unlock(&ec->lock); > > > + > > > + if (ret) > > > + return ret; > > > + > > > + value = ((tmp[0] << 8) | tmp[1]) & EC_ADC_RESOLUTION_MAX; > > > + factor = (tmp[2] << 8) | tmp[3]; > > > > Is it guaranteed that factor is never 0 ? If not, this may result > > in a division by 0 error. > > > > @@ -326,6 +318,11 @@ static int imanager2_curr_get_value(struct imanager2 > *ec, int index, > value = ((tmp[0] << 8) | tmp[1]) & EC_ADC_RESOLUTION_MAX; > factor = (tmp[2] << 8) | tmp[3]; > > + if (!factor) { > + *curr_mamp = 0; > + return 0; > + } > + > while (tmp[4]++) > baseunit *= 10; > > > > + > > > + while (tmp[4]++) > > > + baseunit *= 10; > > > + > > [...] > > > > +static void imanager2_temp_init(struct imanager2 *ec) > > > +{ > > > + int i, thm, ret; > > > + u8 tmltype, smbid, fanid; > > > + struct ec_thermalzone zone; > > > + > > > + for (i = 0; i < ARRAY_SIZE(ec_temp_table); i++) > > > + ec_temp_table[i].visible = false; > > > + > > > + for (thm = 0; thm < EC_THERMAL_ZONE_MAX; thm++) { > > > + mutex_lock(&ec->lock); > > > + > > > + if (ec->flag & EC_FLAG_MAILBOX) { > > > + int len = sizeof(struct ec_thermalzone); > > > > Checkpatch warning. > > I use "./scripts/checkpatch.pl --no-tree --strict -f > drivers/hwmon/imanager2_hwm" > but no warning shown out. > Can you help me to find out the warning? > > > > > > + ret = imanager2_mbox_read_thermalzone(ec->flag, thm, > > > + &smbid, &fanid, > > > + (u8 *)&zone, > > > + &len); > > [...] > > > > +static struct fan_item ec_fan_table[] = { > > > + { > > > + .did = tacho0, > > > + .name = "Fan CPU", > > > + .fspeed_acpireg = 0, > > > > It is not necessary to set such constants to 0. > > @@ -566,20 +566,17 @@ static struct fan_item ec_fan_table[] = { > { > .did = tacho0, > .name = "Fan CPU", > - .fspeed_acpireg = 0, > - .visible = false > + .visible = false, > }, > { > .did = tacho1, > .name = "Fan SYS", > - .fspeed_acpireg = 0, > - .visible = false > + .visible = false, > }, > { > .did = tacho2, > .name = "Fan SYS2", > - .fspeed_acpireg = 0, > - .visible = false > + .visible = false, > }, > }; > > > > > > + .visible = false > > > + }, > > > + { > > > + .did = tacho1, > > > + .name = "Fan SYS", > > > + .fspeed_acpireg = 0, > > > + .visible = false > > > + }, > > > + { > > > + .did = tacho2, > > > + .name = "Fan SYS2", > > > + .fspeed_acpireg = 0, > > > + .visible = false > > > + }, > > > +}; > > > + > > > +static int imanager2_fan_get_value(struct imanager2 *ec, int index, > > > + u32 *speed_rpm) > > > +{ > > > + int ret; > > > + u8 tmp[2]; > > > + > > > + mutex_lock(&ec->lock); > > > + > > > + if (ec->flag & EC_FLAG_MAILBOX) > > > + ret = imanager2_mbox_read_data(ec->flag, > > > + EC_CMD_MAILBOX_READ_HW_PIN, > > > + ec_fan_table[index].did, tmp, 2); > > > + else > > > + ret = imanager2_mbox_io_read(EC_CMD_ACPIRAM_READ, > > > + ec_fan_table[index].fspeed_acpireg, > > > + tmp, 2); > > > + > > > + mutex_unlock(&ec->lock); > > > + > > > + if (ret) > > > + return ret; > > > + > > > + if (tmp[0] == 0xFF && tmp[1] == 0xFF) > > > + return -ENODEV; > > > > That is a bit late for detecting that there is no such device. > > > > @@ -605,11 +602,11 @@ static int imanager2_fan_get_value(struct imanager2 > *ec, int index, > if (ret) > return ret; > > - if (tmp[0] == 0xFF && tmp[1] == 0xFF) > - return -ENODEV; > - > *speed_rpm = (tmp[0] << 8) | tmp[1]; > > + if (*speed_rpm == 0xFFFF) > + *speed_rpm = 0; > + > return 0; > } > > > > + > > > + *speed_rpm = (tmp[0] << 8) | tmp[1]; > > > + > > > + return 0; > > > +} > > > + > > > +#define EC_FANCTROL_SPEEDSRC_MASK 0x30 > > > + > > > +static int imanager2_fan_item_init_by_io(struct imanager2 *ec, int fnum) > > > +{ > > > + int i, ret; > > > + u8 tmp; > > > + > > > + mutex_lock(&ec->lock); > > > + ret = imanager2_mbox_io_read(EC_CMD_HWRAM_READ, > > > + EC_HWRAM_ADDR_FAN_CONTROL(fnum), &tmp, 1); > > > + mutex_unlock(&ec->lock); > > > + > > > + if (ret) > > > + return ret; > > > + > > > + i = ((tmp & EC_FANCTROL_SPEEDSRC_MASK) >> 4) - 1; > > > + if (i < 0) > > > + return -ENODEV; > > > + > > > > This error return is quite pointless. See below. > > > > @@ -627,7 +624,7 @@ static int imanager2_fan_item_init_by_io(struct imanager2 > *ec, int fnum) > > i = ((tmp & EC_FANCTROL_SPEEDSRC_MASK) >> 4) - 1; > if (i < 0) > - return -ENODEV; > + return 0; > > /* Unnecessary set again if it has been set. */ > if (ec_fan_table[i].visible) > > > > + /* Unnecessary set again if it has been set. */ > > > + if (ec_fan_table[i].visible) > > > + return 0; > > > + > > > + ec_fan_table[i].fspeed_acpireg = EC_ACPIRAM_ADDR_FAN_SPEED_BASE(i); > > > + ec_fan_table[i].visible = true; > > > + > > > + return 0; > > > +} > > > + > > > +static void imanager2_fan_init(struct imanager2 *ec) > > > +{ > > > + int i; > > > + > > > + if (ec->flag & EC_FLAG_MAILBOX) { > > > + for (i = 0; i < ARRAY_SIZE(ec_fan_table); i++) > > > + ec_fan_table[i].visible = > > > + ec->table.devid2itemnum[ec_fan_table[i].did] != > > > + EC_TABLE_ITEM_UNUSED; > > > + } else { > > > + int fnum; > > > + > > > + for (i = 0; i < ARRAY_SIZE(ec_fan_table); i++) > > > + ec_fan_table[i].visible = false; > > > + > > > + for (fnum = 0; fnum < EC_MAX_IO_FAN; fnum++) { > > > + if (imanager2_fan_item_init_by_io(ec, fnum)) > > > + continue; > > > > This continue is quite pointless. > > > > @@ -654,10 +651,8 @@ static void imanager2_fan_init(struct imanager2 *ec) > for (i = 0; i < ARRAY_SIZE(ec_fan_table); i++) > ec_fan_table[i].visible = false; > > - for (fnum = 0; fnum < EC_MAX_IO_FAN; fnum++) { > - if (imanager2_fan_item_init_by_io(ec, fnum)) > - continue; > - } > + for (fnum = 0; fnum < EC_MAX_IO_FAN; fnum++) > + imanager2_fan_item_init_by_io(ec, fnum); > } > } > > > > + } > > > + } > > > +} > > > + > > > +static ssize_t show_fan(struct device *dev, struct device_attribute > > > *dev_attr, > > > + char *buf) > > > +{ > > > + struct imanager2 *ec = dev_get_drvdata(dev); > > > + u32 val; > > > + int ret = imanager2_fan_get_value(ec, > > > + to_sensor_dev_attr(dev_attr)->index, > > > + &val); > > > + if (ret) > > > + return ret; > > > + > > > + return sprintf(buf, "%u\n", val); > > > +} > > > + > > > +static ssize_t show_fan_label(struct device *dev, > > > + struct device_attribute *dev_attr, char *buf) > > > +{ > > > + return sprintf(buf, "%s\n", > > > + ec_fan_table[to_sensor_dev_attr(dev_attr)->index].name); > > > +} > > > + > > > +static SENSOR_DEVICE_ATTR(fan1_input, S_IRUGO, show_fan, NULL, 0); > > > +static SENSOR_DEVICE_ATTR(fan2_input, S_IRUGO, show_fan, NULL, 1); > > > +static SENSOR_DEVICE_ATTR(fan3_input, S_IRUGO, show_fan, NULL, 2); > > > + > > > +static SENSOR_DEVICE_ATTR(fan1_label, S_IRUGO, show_fan_label, NULL, 0); > > > +static SENSOR_DEVICE_ATTR(fan2_label, S_IRUGO, show_fan_label, NULL, 1); > > > +static SENSOR_DEVICE_ATTR(fan3_label, S_IRUGO, show_fan_label, NULL, 2); > > > + > > > +static struct attribute *imanager2_fan_attrs[] = { > > > + &sensor_dev_attr_fan1_label.dev_attr.attr, > > > + &sensor_dev_attr_fan1_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_fan2_label.dev_attr.attr, > > > + &sensor_dev_attr_fan2_input.dev_attr.attr, > > > + > > > + &sensor_dev_attr_fan3_label.dev_attr.attr, > > > + &sensor_dev_attr_fan3_input.dev_attr.attr, > > > + > > > + NULL > > > +}; > > > + > > > +static umode_t imanager2_fan_mode(struct kobject *kobj, struct attribute > > > *attr, > > > + int index) > > > +{ > > > + struct device_attribute *dev_attr; > > > + > > > + dev_attr = container_of(attr, struct device_attribute, attr); > > > + if (!ec_fan_table[to_sensor_dev_attr(dev_attr)->index].visible) > > > + return 0; > > > + > > > + return attr->mode; > > > +} > > > + > > > +static const struct attribute_group imanager2_fan_group = { > > > + .attrs = imanager2_fan_attrs, > > > + .is_visible = imanager2_fan_mode, > > > +}; > > > + > > > +/* Module */ > > > +static const struct attribute_group *imanager2_hwmon_groups[] = { > > > + &imanager2_volt_group, > > > + &imanager2_curr_group, > > > + &imanager2_temp_group, > > > + &imanager2_fan_group, > > > + NULL > > > +}; > > > + > > > +static int imanager2_hwmon_probe(struct platform_device *pdev) > > > +{ > > > + struct device *hwmon_dev; > > > + struct imanager2 *hwmon_ec = pdev->dev.parent->platform_data; > > > + > > > + imanager2_volt_init(hwmon_ec); > > > + imanager2_curr_init(hwmon_ec); > > > + imanager2_temp_init(hwmon_ec); > > > + imanager2_fan_init(hwmon_ec); > > > + > > > + hwmon_dev = devm_hwmon_device_register_with_groups( > > > + &pdev->dev, "imanager2", hwmon_ec, > > > + imanager2_hwmon_groups); > > > + > > > + if (IS_ERR(hwmon_dev)) > > > + return PTR_ERR(hwmon_dev); > > > + > > > + return 0; > > > > return PTR_ERR_OR_ZERO(hwmon_dev); > > > > @@ -743,10 +738,7 @@ static int imanager2_hwmon_probe(struct platform_device > *pdev) > &pdev->dev, "imanager2", hwmon_ec, > imanager2_hwmon_groups); > > - if (IS_ERR(hwmon_dev)) > - return PTR_ERR(hwmon_dev); > - > - return 0; > + return PTR_ERR_OR_ZERO(hwmon_dev); > } > > > > +} > > > + > > > +static struct platform_driver imanager2_hwmon_driver = { > > > + .probe = imanager2_hwmon_probe, > > > + .driver = { > > > + .owner = THIS_MODULE, > > > + .name = DRV_NAME, > > > + }, > > > +}; > > > + > > > +module_platform_driver(imanager2_hwmon_driver); > > > + > > > +MODULE_AUTHOR("Richard Vidal-Dorsch <richard.dorsch at > > advantech.com>"); > > > +MODULE_DESCRIPTION("HW Monitoring interface for Advantech EC > > IT8516/18/28"); > > > +MODULE_LICENSE("GPL"); > > > +MODULE_VERSION(DRV_VERSION); > > > > > > -- Lee Jones Linaro STMicroelectronics Landing Team Lead Linaro.org │ Open source software for ARM SoCs Follow Linaro: Facebook | Twitter | Blog -- To unsubscribe from this list: send the line "unsubscribe linux-kernel" in the body of a message to majord...@vger.kernel.org More majordomo info at http://vger.kernel.org/majordomo-info.html Please read the FAQ at http://www.tux.org/lkml/