1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Driver for SMM665 Power Controller / Monitor
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Copyright (C) 2010 Ericsson AB.
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * This driver should also work for SMM465, SMM764, and SMM766, but is untested
8*4882a593Smuzhiyun * for those chips. Only monitoring functionality is implemented.
9*4882a593Smuzhiyun *
10*4882a593Smuzhiyun * Datasheets:
11*4882a593Smuzhiyun * http://www.summitmicro.com/prod_select/summary/SMM665/SMM665B_2089_20.pdf
12*4882a593Smuzhiyun * http://www.summitmicro.com/prod_select/summary/SMM766B/SMM766B_2122.pdf
13*4882a593Smuzhiyun */
14*4882a593Smuzhiyun
15*4882a593Smuzhiyun #include <linux/kernel.h>
16*4882a593Smuzhiyun #include <linux/module.h>
17*4882a593Smuzhiyun #include <linux/init.h>
18*4882a593Smuzhiyun #include <linux/err.h>
19*4882a593Smuzhiyun #include <linux/slab.h>
20*4882a593Smuzhiyun #include <linux/i2c.h>
21*4882a593Smuzhiyun #include <linux/hwmon.h>
22*4882a593Smuzhiyun #include <linux/hwmon-sysfs.h>
23*4882a593Smuzhiyun #include <linux/delay.h>
24*4882a593Smuzhiyun #include <linux/jiffies.h>
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun /* Internal reference voltage (VREF, x 1000 */
27*4882a593Smuzhiyun #define SMM665_VREF_ADC_X1000 1250
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun /* module parameters */
30*4882a593Smuzhiyun static int vref = SMM665_VREF_ADC_X1000;
31*4882a593Smuzhiyun module_param(vref, int, 0);
32*4882a593Smuzhiyun MODULE_PARM_DESC(vref, "Reference voltage in mV");
33*4882a593Smuzhiyun
34*4882a593Smuzhiyun enum chips { smm465, smm665, smm665c, smm764, smm766 };
35*4882a593Smuzhiyun
36*4882a593Smuzhiyun /*
37*4882a593Smuzhiyun * ADC channel addresses
38*4882a593Smuzhiyun */
39*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_A 0x00
40*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_B 0x01
41*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_C 0x02
42*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_D 0x03
43*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_E 0x04
44*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_F 0x05
45*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_VDD 0x06
46*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_12V 0x07
47*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_INT_TEMP 0x08
48*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_AIN1 0x09
49*4882a593Smuzhiyun #define SMM665_MISC16_ADC_DATA_AIN2 0x0a
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun /*
52*4882a593Smuzhiyun * Command registers
53*4882a593Smuzhiyun */
54*4882a593Smuzhiyun #define SMM665_MISC8_CMD_STS 0x80
55*4882a593Smuzhiyun #define SMM665_MISC8_STATUS1 0x81
56*4882a593Smuzhiyun #define SMM665_MISC8_STATUSS2 0x82
57*4882a593Smuzhiyun #define SMM665_MISC8_IO_POLARITY 0x83
58*4882a593Smuzhiyun #define SMM665_MISC8_PUP_POLARITY 0x84
59*4882a593Smuzhiyun #define SMM665_MISC8_ADOC_STATUS1 0x85
60*4882a593Smuzhiyun #define SMM665_MISC8_ADOC_STATUS2 0x86
61*4882a593Smuzhiyun #define SMM665_MISC8_WRITE_PROT 0x87
62*4882a593Smuzhiyun #define SMM665_MISC8_STS_TRACK 0x88
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun /*
65*4882a593Smuzhiyun * Configuration registers and register groups
66*4882a593Smuzhiyun */
67*4882a593Smuzhiyun #define SMM665_ADOC_ENABLE 0x0d
68*4882a593Smuzhiyun #define SMM665_LIMIT_BASE 0x80 /* First limit register */
69*4882a593Smuzhiyun
70*4882a593Smuzhiyun /*
71*4882a593Smuzhiyun * Limit register bit masks
72*4882a593Smuzhiyun */
73*4882a593Smuzhiyun #define SMM665_TRIGGER_RST 0x8000
74*4882a593Smuzhiyun #define SMM665_TRIGGER_HEALTHY 0x4000
75*4882a593Smuzhiyun #define SMM665_TRIGGER_POWEROFF 0x2000
76*4882a593Smuzhiyun #define SMM665_TRIGGER_SHUTDOWN 0x1000
77*4882a593Smuzhiyun #define SMM665_ADC_MASK 0x03ff
78*4882a593Smuzhiyun
79*4882a593Smuzhiyun #define smm665_is_critical(lim) ((lim) & (SMM665_TRIGGER_RST \
80*4882a593Smuzhiyun | SMM665_TRIGGER_POWEROFF \
81*4882a593Smuzhiyun | SMM665_TRIGGER_SHUTDOWN))
82*4882a593Smuzhiyun /*
83*4882a593Smuzhiyun * Fault register bit definitions
84*4882a593Smuzhiyun * Values are merged from status registers 1/2,
85*4882a593Smuzhiyun * with status register 1 providing the upper 8 bits.
86*4882a593Smuzhiyun */
87*4882a593Smuzhiyun #define SMM665_FAULT_A 0x0001
88*4882a593Smuzhiyun #define SMM665_FAULT_B 0x0002
89*4882a593Smuzhiyun #define SMM665_FAULT_C 0x0004
90*4882a593Smuzhiyun #define SMM665_FAULT_D 0x0008
91*4882a593Smuzhiyun #define SMM665_FAULT_E 0x0010
92*4882a593Smuzhiyun #define SMM665_FAULT_F 0x0020
93*4882a593Smuzhiyun #define SMM665_FAULT_VDD 0x0040
94*4882a593Smuzhiyun #define SMM665_FAULT_12V 0x0080
95*4882a593Smuzhiyun #define SMM665_FAULT_TEMP 0x0100
96*4882a593Smuzhiyun #define SMM665_FAULT_AIN1 0x0200
97*4882a593Smuzhiyun #define SMM665_FAULT_AIN2 0x0400
98*4882a593Smuzhiyun
99*4882a593Smuzhiyun /*
100*4882a593Smuzhiyun * I2C Register addresses
101*4882a593Smuzhiyun *
102*4882a593Smuzhiyun * The configuration register needs to be the configured base register.
103*4882a593Smuzhiyun * The command/status register address is derived from it.
104*4882a593Smuzhiyun */
105*4882a593Smuzhiyun #define SMM665_REGMASK 0x78
106*4882a593Smuzhiyun #define SMM665_CMDREG_BASE 0x48
107*4882a593Smuzhiyun #define SMM665_CONFREG_BASE 0x50
108*4882a593Smuzhiyun
109*4882a593Smuzhiyun /*
110*4882a593Smuzhiyun * Equations given by chip manufacturer to calculate voltage/temperature values
111*4882a593Smuzhiyun * vref = Reference voltage on VREF_ADC pin (module parameter)
112*4882a593Smuzhiyun * adc = 10bit ADC value read back from registers
113*4882a593Smuzhiyun */
114*4882a593Smuzhiyun
115*4882a593Smuzhiyun /* Voltage A-F and VDD */
116*4882a593Smuzhiyun #define SMM665_VMON_ADC_TO_VOLTS(adc) ((adc) * vref / 256)
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun /* Voltage 12VIN */
119*4882a593Smuzhiyun #define SMM665_12VIN_ADC_TO_VOLTS(adc) ((adc) * vref * 3 / 256)
120*4882a593Smuzhiyun
121*4882a593Smuzhiyun /* Voltage AIN1, AIN2 */
122*4882a593Smuzhiyun #define SMM665_AIN_ADC_TO_VOLTS(adc) ((adc) * vref / 512)
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun /* Temp Sensor */
125*4882a593Smuzhiyun #define SMM665_TEMP_ADC_TO_CELSIUS(adc) (((adc) <= 511) ? \
126*4882a593Smuzhiyun ((int)(adc) * 1000 / 4) : \
127*4882a593Smuzhiyun (((int)(adc) - 0x400) * 1000 / 4))
128*4882a593Smuzhiyun
129*4882a593Smuzhiyun #define SMM665_NUM_ADC 11
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun /*
132*4882a593Smuzhiyun * Chip dependent ADC conversion time, in uS
133*4882a593Smuzhiyun */
134*4882a593Smuzhiyun #define SMM665_ADC_WAIT_SMM665 70
135*4882a593Smuzhiyun #define SMM665_ADC_WAIT_SMM766 185
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun struct smm665_data {
138*4882a593Smuzhiyun enum chips type;
139*4882a593Smuzhiyun int conversion_time; /* ADC conversion time */
140*4882a593Smuzhiyun struct i2c_client *client;
141*4882a593Smuzhiyun struct mutex update_lock;
142*4882a593Smuzhiyun bool valid;
143*4882a593Smuzhiyun unsigned long last_updated; /* in jiffies */
144*4882a593Smuzhiyun u16 adc[SMM665_NUM_ADC]; /* adc values (raw) */
145*4882a593Smuzhiyun u16 faults; /* fault status */
146*4882a593Smuzhiyun /* The following values are in mV */
147*4882a593Smuzhiyun int critical_min_limit[SMM665_NUM_ADC];
148*4882a593Smuzhiyun int alarm_min_limit[SMM665_NUM_ADC];
149*4882a593Smuzhiyun int critical_max_limit[SMM665_NUM_ADC];
150*4882a593Smuzhiyun int alarm_max_limit[SMM665_NUM_ADC];
151*4882a593Smuzhiyun struct i2c_client *cmdreg;
152*4882a593Smuzhiyun };
153*4882a593Smuzhiyun
154*4882a593Smuzhiyun /*
155*4882a593Smuzhiyun * smm665_read16()
156*4882a593Smuzhiyun *
157*4882a593Smuzhiyun * Read 16 bit value from <reg>, <reg+1>. Upper 8 bits are in <reg>.
158*4882a593Smuzhiyun */
smm665_read16(struct i2c_client * client,int reg)159*4882a593Smuzhiyun static int smm665_read16(struct i2c_client *client, int reg)
160*4882a593Smuzhiyun {
161*4882a593Smuzhiyun int rv, val;
162*4882a593Smuzhiyun
163*4882a593Smuzhiyun rv = i2c_smbus_read_byte_data(client, reg);
164*4882a593Smuzhiyun if (rv < 0)
165*4882a593Smuzhiyun return rv;
166*4882a593Smuzhiyun val = rv << 8;
167*4882a593Smuzhiyun rv = i2c_smbus_read_byte_data(client, reg + 1);
168*4882a593Smuzhiyun if (rv < 0)
169*4882a593Smuzhiyun return rv;
170*4882a593Smuzhiyun val |= rv;
171*4882a593Smuzhiyun return val;
172*4882a593Smuzhiyun }
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun /*
175*4882a593Smuzhiyun * Read adc value.
176*4882a593Smuzhiyun */
smm665_read_adc(struct smm665_data * data,int adc)177*4882a593Smuzhiyun static int smm665_read_adc(struct smm665_data *data, int adc)
178*4882a593Smuzhiyun {
179*4882a593Smuzhiyun struct i2c_client *client = data->cmdreg;
180*4882a593Smuzhiyun int rv;
181*4882a593Smuzhiyun int radc;
182*4882a593Smuzhiyun
183*4882a593Smuzhiyun /*
184*4882a593Smuzhiyun * Algorithm for reading ADC, per SMM665 datasheet
185*4882a593Smuzhiyun *
186*4882a593Smuzhiyun * {[S][addr][W][Ack]} {[offset][Ack]} {[S][addr][R][Nack]}
187*4882a593Smuzhiyun * [wait conversion time]
188*4882a593Smuzhiyun * {[S][addr][R][Ack]} {[datahi][Ack]} {[datalo][Ack][P]}
189*4882a593Smuzhiyun *
190*4882a593Smuzhiyun * To implement the first part of this exchange,
191*4882a593Smuzhiyun * do a full read transaction and expect a failure/Nack.
192*4882a593Smuzhiyun * This sets up the address pointer on the SMM665
193*4882a593Smuzhiyun * and starts the ADC conversion.
194*4882a593Smuzhiyun * Then do a two-byte read transaction.
195*4882a593Smuzhiyun */
196*4882a593Smuzhiyun rv = i2c_smbus_read_byte_data(client, adc << 3);
197*4882a593Smuzhiyun if (rv != -ENXIO) {
198*4882a593Smuzhiyun /*
199*4882a593Smuzhiyun * We expect ENXIO to reflect NACK
200*4882a593Smuzhiyun * (per Documentation/i2c/fault-codes.rst).
201*4882a593Smuzhiyun * Everything else is an error.
202*4882a593Smuzhiyun */
203*4882a593Smuzhiyun dev_dbg(&client->dev,
204*4882a593Smuzhiyun "Unexpected return code %d when setting ADC index", rv);
205*4882a593Smuzhiyun return (rv < 0) ? rv : -EIO;
206*4882a593Smuzhiyun }
207*4882a593Smuzhiyun
208*4882a593Smuzhiyun udelay(data->conversion_time);
209*4882a593Smuzhiyun
210*4882a593Smuzhiyun /*
211*4882a593Smuzhiyun * Now read two bytes.
212*4882a593Smuzhiyun *
213*4882a593Smuzhiyun * Neither i2c_smbus_read_byte() nor
214*4882a593Smuzhiyun * i2c_smbus_read_block_data() worked here,
215*4882a593Smuzhiyun * so use i2c_smbus_read_word_swapped() instead.
216*4882a593Smuzhiyun * We could also try to use i2c_master_recv(),
217*4882a593Smuzhiyun * but that is not always supported.
218*4882a593Smuzhiyun */
219*4882a593Smuzhiyun rv = i2c_smbus_read_word_swapped(client, 0);
220*4882a593Smuzhiyun if (rv < 0) {
221*4882a593Smuzhiyun dev_dbg(&client->dev, "Failed to read ADC value: error %d", rv);
222*4882a593Smuzhiyun return rv;
223*4882a593Smuzhiyun }
224*4882a593Smuzhiyun /*
225*4882a593Smuzhiyun * Validate/verify readback adc channel (in bit 11..14).
226*4882a593Smuzhiyun */
227*4882a593Smuzhiyun radc = (rv >> 11) & 0x0f;
228*4882a593Smuzhiyun if (radc != adc) {
229*4882a593Smuzhiyun dev_dbg(&client->dev, "Unexpected RADC: Expected %d got %d",
230*4882a593Smuzhiyun adc, radc);
231*4882a593Smuzhiyun return -EIO;
232*4882a593Smuzhiyun }
233*4882a593Smuzhiyun
234*4882a593Smuzhiyun return rv & SMM665_ADC_MASK;
235*4882a593Smuzhiyun }
236*4882a593Smuzhiyun
smm665_update_device(struct device * dev)237*4882a593Smuzhiyun static struct smm665_data *smm665_update_device(struct device *dev)
238*4882a593Smuzhiyun {
239*4882a593Smuzhiyun struct smm665_data *data = dev_get_drvdata(dev);
240*4882a593Smuzhiyun struct i2c_client *client = data->client;
241*4882a593Smuzhiyun struct smm665_data *ret = data;
242*4882a593Smuzhiyun
243*4882a593Smuzhiyun mutex_lock(&data->update_lock);
244*4882a593Smuzhiyun
245*4882a593Smuzhiyun if (time_after(jiffies, data->last_updated + HZ) || !data->valid) {
246*4882a593Smuzhiyun int i, val;
247*4882a593Smuzhiyun
248*4882a593Smuzhiyun /*
249*4882a593Smuzhiyun * read status registers
250*4882a593Smuzhiyun */
251*4882a593Smuzhiyun val = smm665_read16(client, SMM665_MISC8_STATUS1);
252*4882a593Smuzhiyun if (unlikely(val < 0)) {
253*4882a593Smuzhiyun ret = ERR_PTR(val);
254*4882a593Smuzhiyun goto abort;
255*4882a593Smuzhiyun }
256*4882a593Smuzhiyun data->faults = val;
257*4882a593Smuzhiyun
258*4882a593Smuzhiyun /* Read adc registers */
259*4882a593Smuzhiyun for (i = 0; i < SMM665_NUM_ADC; i++) {
260*4882a593Smuzhiyun val = smm665_read_adc(data, i);
261*4882a593Smuzhiyun if (unlikely(val < 0)) {
262*4882a593Smuzhiyun ret = ERR_PTR(val);
263*4882a593Smuzhiyun goto abort;
264*4882a593Smuzhiyun }
265*4882a593Smuzhiyun data->adc[i] = val;
266*4882a593Smuzhiyun }
267*4882a593Smuzhiyun data->last_updated = jiffies;
268*4882a593Smuzhiyun data->valid = 1;
269*4882a593Smuzhiyun }
270*4882a593Smuzhiyun abort:
271*4882a593Smuzhiyun mutex_unlock(&data->update_lock);
272*4882a593Smuzhiyun return ret;
273*4882a593Smuzhiyun }
274*4882a593Smuzhiyun
275*4882a593Smuzhiyun /* Return converted value from given adc */
smm665_convert(u16 adcval,int index)276*4882a593Smuzhiyun static int smm665_convert(u16 adcval, int index)
277*4882a593Smuzhiyun {
278*4882a593Smuzhiyun int val = 0;
279*4882a593Smuzhiyun
280*4882a593Smuzhiyun switch (index) {
281*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_12V:
282*4882a593Smuzhiyun val = SMM665_12VIN_ADC_TO_VOLTS(adcval & SMM665_ADC_MASK);
283*4882a593Smuzhiyun break;
284*4882a593Smuzhiyun
285*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_VDD:
286*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_A:
287*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_B:
288*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_C:
289*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_D:
290*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_E:
291*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_F:
292*4882a593Smuzhiyun val = SMM665_VMON_ADC_TO_VOLTS(adcval & SMM665_ADC_MASK);
293*4882a593Smuzhiyun break;
294*4882a593Smuzhiyun
295*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_AIN1:
296*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_AIN2:
297*4882a593Smuzhiyun val = SMM665_AIN_ADC_TO_VOLTS(adcval & SMM665_ADC_MASK);
298*4882a593Smuzhiyun break;
299*4882a593Smuzhiyun
300*4882a593Smuzhiyun case SMM665_MISC16_ADC_DATA_INT_TEMP:
301*4882a593Smuzhiyun val = SMM665_TEMP_ADC_TO_CELSIUS(adcval & SMM665_ADC_MASK);
302*4882a593Smuzhiyun break;
303*4882a593Smuzhiyun
304*4882a593Smuzhiyun default:
305*4882a593Smuzhiyun /* If we get here, the developer messed up */
306*4882a593Smuzhiyun WARN_ON_ONCE(1);
307*4882a593Smuzhiyun break;
308*4882a593Smuzhiyun }
309*4882a593Smuzhiyun
310*4882a593Smuzhiyun return val;
311*4882a593Smuzhiyun }
312*4882a593Smuzhiyun
smm665_get_min(struct device * dev,int index)313*4882a593Smuzhiyun static int smm665_get_min(struct device *dev, int index)
314*4882a593Smuzhiyun {
315*4882a593Smuzhiyun struct smm665_data *data = dev_get_drvdata(dev);
316*4882a593Smuzhiyun
317*4882a593Smuzhiyun return data->alarm_min_limit[index];
318*4882a593Smuzhiyun }
319*4882a593Smuzhiyun
smm665_get_max(struct device * dev,int index)320*4882a593Smuzhiyun static int smm665_get_max(struct device *dev, int index)
321*4882a593Smuzhiyun {
322*4882a593Smuzhiyun struct smm665_data *data = dev_get_drvdata(dev);
323*4882a593Smuzhiyun
324*4882a593Smuzhiyun return data->alarm_max_limit[index];
325*4882a593Smuzhiyun }
326*4882a593Smuzhiyun
smm665_get_lcrit(struct device * dev,int index)327*4882a593Smuzhiyun static int smm665_get_lcrit(struct device *dev, int index)
328*4882a593Smuzhiyun {
329*4882a593Smuzhiyun struct smm665_data *data = dev_get_drvdata(dev);
330*4882a593Smuzhiyun
331*4882a593Smuzhiyun return data->critical_min_limit[index];
332*4882a593Smuzhiyun }
333*4882a593Smuzhiyun
smm665_get_crit(struct device * dev,int index)334*4882a593Smuzhiyun static int smm665_get_crit(struct device *dev, int index)
335*4882a593Smuzhiyun {
336*4882a593Smuzhiyun struct smm665_data *data = dev_get_drvdata(dev);
337*4882a593Smuzhiyun
338*4882a593Smuzhiyun return data->critical_max_limit[index];
339*4882a593Smuzhiyun }
340*4882a593Smuzhiyun
smm665_show_crit_alarm(struct device * dev,struct device_attribute * da,char * buf)341*4882a593Smuzhiyun static ssize_t smm665_show_crit_alarm(struct device *dev,
342*4882a593Smuzhiyun struct device_attribute *da, char *buf)
343*4882a593Smuzhiyun {
344*4882a593Smuzhiyun struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
345*4882a593Smuzhiyun struct smm665_data *data = smm665_update_device(dev);
346*4882a593Smuzhiyun int val = 0;
347*4882a593Smuzhiyun
348*4882a593Smuzhiyun if (IS_ERR(data))
349*4882a593Smuzhiyun return PTR_ERR(data);
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun if (data->faults & (1 << attr->index))
352*4882a593Smuzhiyun val = 1;
353*4882a593Smuzhiyun
354*4882a593Smuzhiyun return snprintf(buf, PAGE_SIZE, "%d\n", val);
355*4882a593Smuzhiyun }
356*4882a593Smuzhiyun
smm665_show_input(struct device * dev,struct device_attribute * da,char * buf)357*4882a593Smuzhiyun static ssize_t smm665_show_input(struct device *dev,
358*4882a593Smuzhiyun struct device_attribute *da, char *buf)
359*4882a593Smuzhiyun {
360*4882a593Smuzhiyun struct sensor_device_attribute *attr = to_sensor_dev_attr(da);
361*4882a593Smuzhiyun struct smm665_data *data = smm665_update_device(dev);
362*4882a593Smuzhiyun int adc = attr->index;
363*4882a593Smuzhiyun int val;
364*4882a593Smuzhiyun
365*4882a593Smuzhiyun if (IS_ERR(data))
366*4882a593Smuzhiyun return PTR_ERR(data);
367*4882a593Smuzhiyun
368*4882a593Smuzhiyun val = smm665_convert(data->adc[adc], adc);
369*4882a593Smuzhiyun return snprintf(buf, PAGE_SIZE, "%d\n", val);
370*4882a593Smuzhiyun }
371*4882a593Smuzhiyun
372*4882a593Smuzhiyun #define SMM665_SHOW(what) \
373*4882a593Smuzhiyun static ssize_t smm665_show_##what(struct device *dev, \
374*4882a593Smuzhiyun struct device_attribute *da, char *buf) \
375*4882a593Smuzhiyun { \
376*4882a593Smuzhiyun struct sensor_device_attribute *attr = to_sensor_dev_attr(da); \
377*4882a593Smuzhiyun const int val = smm665_get_##what(dev, attr->index); \
378*4882a593Smuzhiyun return snprintf(buf, PAGE_SIZE, "%d\n", val); \
379*4882a593Smuzhiyun }
380*4882a593Smuzhiyun
381*4882a593Smuzhiyun SMM665_SHOW(min);
382*4882a593Smuzhiyun SMM665_SHOW(max);
383*4882a593Smuzhiyun SMM665_SHOW(lcrit);
384*4882a593Smuzhiyun SMM665_SHOW(crit);
385*4882a593Smuzhiyun
386*4882a593Smuzhiyun /*
387*4882a593Smuzhiyun * These macros are used below in constructing device attribute objects
388*4882a593Smuzhiyun * for use with sysfs_create_group() to make a sysfs device file
389*4882a593Smuzhiyun * for each register.
390*4882a593Smuzhiyun */
391*4882a593Smuzhiyun
392*4882a593Smuzhiyun #define SMM665_ATTR(name, type, cmd_idx) \
393*4882a593Smuzhiyun static SENSOR_DEVICE_ATTR(name##_##type, S_IRUGO, \
394*4882a593Smuzhiyun smm665_show_##type, NULL, cmd_idx)
395*4882a593Smuzhiyun
396*4882a593Smuzhiyun /* Construct a sensor_device_attribute structure for each register */
397*4882a593Smuzhiyun
398*4882a593Smuzhiyun /* Input voltages */
399*4882a593Smuzhiyun SMM665_ATTR(in1, input, SMM665_MISC16_ADC_DATA_12V);
400*4882a593Smuzhiyun SMM665_ATTR(in2, input, SMM665_MISC16_ADC_DATA_VDD);
401*4882a593Smuzhiyun SMM665_ATTR(in3, input, SMM665_MISC16_ADC_DATA_A);
402*4882a593Smuzhiyun SMM665_ATTR(in4, input, SMM665_MISC16_ADC_DATA_B);
403*4882a593Smuzhiyun SMM665_ATTR(in5, input, SMM665_MISC16_ADC_DATA_C);
404*4882a593Smuzhiyun SMM665_ATTR(in6, input, SMM665_MISC16_ADC_DATA_D);
405*4882a593Smuzhiyun SMM665_ATTR(in7, input, SMM665_MISC16_ADC_DATA_E);
406*4882a593Smuzhiyun SMM665_ATTR(in8, input, SMM665_MISC16_ADC_DATA_F);
407*4882a593Smuzhiyun SMM665_ATTR(in9, input, SMM665_MISC16_ADC_DATA_AIN1);
408*4882a593Smuzhiyun SMM665_ATTR(in10, input, SMM665_MISC16_ADC_DATA_AIN2);
409*4882a593Smuzhiyun
410*4882a593Smuzhiyun /* Input voltages min */
411*4882a593Smuzhiyun SMM665_ATTR(in1, min, SMM665_MISC16_ADC_DATA_12V);
412*4882a593Smuzhiyun SMM665_ATTR(in2, min, SMM665_MISC16_ADC_DATA_VDD);
413*4882a593Smuzhiyun SMM665_ATTR(in3, min, SMM665_MISC16_ADC_DATA_A);
414*4882a593Smuzhiyun SMM665_ATTR(in4, min, SMM665_MISC16_ADC_DATA_B);
415*4882a593Smuzhiyun SMM665_ATTR(in5, min, SMM665_MISC16_ADC_DATA_C);
416*4882a593Smuzhiyun SMM665_ATTR(in6, min, SMM665_MISC16_ADC_DATA_D);
417*4882a593Smuzhiyun SMM665_ATTR(in7, min, SMM665_MISC16_ADC_DATA_E);
418*4882a593Smuzhiyun SMM665_ATTR(in8, min, SMM665_MISC16_ADC_DATA_F);
419*4882a593Smuzhiyun SMM665_ATTR(in9, min, SMM665_MISC16_ADC_DATA_AIN1);
420*4882a593Smuzhiyun SMM665_ATTR(in10, min, SMM665_MISC16_ADC_DATA_AIN2);
421*4882a593Smuzhiyun
422*4882a593Smuzhiyun /* Input voltages max */
423*4882a593Smuzhiyun SMM665_ATTR(in1, max, SMM665_MISC16_ADC_DATA_12V);
424*4882a593Smuzhiyun SMM665_ATTR(in2, max, SMM665_MISC16_ADC_DATA_VDD);
425*4882a593Smuzhiyun SMM665_ATTR(in3, max, SMM665_MISC16_ADC_DATA_A);
426*4882a593Smuzhiyun SMM665_ATTR(in4, max, SMM665_MISC16_ADC_DATA_B);
427*4882a593Smuzhiyun SMM665_ATTR(in5, max, SMM665_MISC16_ADC_DATA_C);
428*4882a593Smuzhiyun SMM665_ATTR(in6, max, SMM665_MISC16_ADC_DATA_D);
429*4882a593Smuzhiyun SMM665_ATTR(in7, max, SMM665_MISC16_ADC_DATA_E);
430*4882a593Smuzhiyun SMM665_ATTR(in8, max, SMM665_MISC16_ADC_DATA_F);
431*4882a593Smuzhiyun SMM665_ATTR(in9, max, SMM665_MISC16_ADC_DATA_AIN1);
432*4882a593Smuzhiyun SMM665_ATTR(in10, max, SMM665_MISC16_ADC_DATA_AIN2);
433*4882a593Smuzhiyun
434*4882a593Smuzhiyun /* Input voltages lcrit */
435*4882a593Smuzhiyun SMM665_ATTR(in1, lcrit, SMM665_MISC16_ADC_DATA_12V);
436*4882a593Smuzhiyun SMM665_ATTR(in2, lcrit, SMM665_MISC16_ADC_DATA_VDD);
437*4882a593Smuzhiyun SMM665_ATTR(in3, lcrit, SMM665_MISC16_ADC_DATA_A);
438*4882a593Smuzhiyun SMM665_ATTR(in4, lcrit, SMM665_MISC16_ADC_DATA_B);
439*4882a593Smuzhiyun SMM665_ATTR(in5, lcrit, SMM665_MISC16_ADC_DATA_C);
440*4882a593Smuzhiyun SMM665_ATTR(in6, lcrit, SMM665_MISC16_ADC_DATA_D);
441*4882a593Smuzhiyun SMM665_ATTR(in7, lcrit, SMM665_MISC16_ADC_DATA_E);
442*4882a593Smuzhiyun SMM665_ATTR(in8, lcrit, SMM665_MISC16_ADC_DATA_F);
443*4882a593Smuzhiyun SMM665_ATTR(in9, lcrit, SMM665_MISC16_ADC_DATA_AIN1);
444*4882a593Smuzhiyun SMM665_ATTR(in10, lcrit, SMM665_MISC16_ADC_DATA_AIN2);
445*4882a593Smuzhiyun
446*4882a593Smuzhiyun /* Input voltages crit */
447*4882a593Smuzhiyun SMM665_ATTR(in1, crit, SMM665_MISC16_ADC_DATA_12V);
448*4882a593Smuzhiyun SMM665_ATTR(in2, crit, SMM665_MISC16_ADC_DATA_VDD);
449*4882a593Smuzhiyun SMM665_ATTR(in3, crit, SMM665_MISC16_ADC_DATA_A);
450*4882a593Smuzhiyun SMM665_ATTR(in4, crit, SMM665_MISC16_ADC_DATA_B);
451*4882a593Smuzhiyun SMM665_ATTR(in5, crit, SMM665_MISC16_ADC_DATA_C);
452*4882a593Smuzhiyun SMM665_ATTR(in6, crit, SMM665_MISC16_ADC_DATA_D);
453*4882a593Smuzhiyun SMM665_ATTR(in7, crit, SMM665_MISC16_ADC_DATA_E);
454*4882a593Smuzhiyun SMM665_ATTR(in8, crit, SMM665_MISC16_ADC_DATA_F);
455*4882a593Smuzhiyun SMM665_ATTR(in9, crit, SMM665_MISC16_ADC_DATA_AIN1);
456*4882a593Smuzhiyun SMM665_ATTR(in10, crit, SMM665_MISC16_ADC_DATA_AIN2);
457*4882a593Smuzhiyun
458*4882a593Smuzhiyun /* critical alarms */
459*4882a593Smuzhiyun SMM665_ATTR(in1, crit_alarm, SMM665_FAULT_12V);
460*4882a593Smuzhiyun SMM665_ATTR(in2, crit_alarm, SMM665_FAULT_VDD);
461*4882a593Smuzhiyun SMM665_ATTR(in3, crit_alarm, SMM665_FAULT_A);
462*4882a593Smuzhiyun SMM665_ATTR(in4, crit_alarm, SMM665_FAULT_B);
463*4882a593Smuzhiyun SMM665_ATTR(in5, crit_alarm, SMM665_FAULT_C);
464*4882a593Smuzhiyun SMM665_ATTR(in6, crit_alarm, SMM665_FAULT_D);
465*4882a593Smuzhiyun SMM665_ATTR(in7, crit_alarm, SMM665_FAULT_E);
466*4882a593Smuzhiyun SMM665_ATTR(in8, crit_alarm, SMM665_FAULT_F);
467*4882a593Smuzhiyun SMM665_ATTR(in9, crit_alarm, SMM665_FAULT_AIN1);
468*4882a593Smuzhiyun SMM665_ATTR(in10, crit_alarm, SMM665_FAULT_AIN2);
469*4882a593Smuzhiyun
470*4882a593Smuzhiyun /* Temperature */
471*4882a593Smuzhiyun SMM665_ATTR(temp1, input, SMM665_MISC16_ADC_DATA_INT_TEMP);
472*4882a593Smuzhiyun SMM665_ATTR(temp1, min, SMM665_MISC16_ADC_DATA_INT_TEMP);
473*4882a593Smuzhiyun SMM665_ATTR(temp1, max, SMM665_MISC16_ADC_DATA_INT_TEMP);
474*4882a593Smuzhiyun SMM665_ATTR(temp1, lcrit, SMM665_MISC16_ADC_DATA_INT_TEMP);
475*4882a593Smuzhiyun SMM665_ATTR(temp1, crit, SMM665_MISC16_ADC_DATA_INT_TEMP);
476*4882a593Smuzhiyun SMM665_ATTR(temp1, crit_alarm, SMM665_FAULT_TEMP);
477*4882a593Smuzhiyun
478*4882a593Smuzhiyun /*
479*4882a593Smuzhiyun * Finally, construct an array of pointers to members of the above objects,
480*4882a593Smuzhiyun * as required for sysfs_create_group()
481*4882a593Smuzhiyun */
482*4882a593Smuzhiyun static struct attribute *smm665_attrs[] = {
483*4882a593Smuzhiyun &sensor_dev_attr_in1_input.dev_attr.attr,
484*4882a593Smuzhiyun &sensor_dev_attr_in1_min.dev_attr.attr,
485*4882a593Smuzhiyun &sensor_dev_attr_in1_max.dev_attr.attr,
486*4882a593Smuzhiyun &sensor_dev_attr_in1_lcrit.dev_attr.attr,
487*4882a593Smuzhiyun &sensor_dev_attr_in1_crit.dev_attr.attr,
488*4882a593Smuzhiyun &sensor_dev_attr_in1_crit_alarm.dev_attr.attr,
489*4882a593Smuzhiyun
490*4882a593Smuzhiyun &sensor_dev_attr_in2_input.dev_attr.attr,
491*4882a593Smuzhiyun &sensor_dev_attr_in2_min.dev_attr.attr,
492*4882a593Smuzhiyun &sensor_dev_attr_in2_max.dev_attr.attr,
493*4882a593Smuzhiyun &sensor_dev_attr_in2_lcrit.dev_attr.attr,
494*4882a593Smuzhiyun &sensor_dev_attr_in2_crit.dev_attr.attr,
495*4882a593Smuzhiyun &sensor_dev_attr_in2_crit_alarm.dev_attr.attr,
496*4882a593Smuzhiyun
497*4882a593Smuzhiyun &sensor_dev_attr_in3_input.dev_attr.attr,
498*4882a593Smuzhiyun &sensor_dev_attr_in3_min.dev_attr.attr,
499*4882a593Smuzhiyun &sensor_dev_attr_in3_max.dev_attr.attr,
500*4882a593Smuzhiyun &sensor_dev_attr_in3_lcrit.dev_attr.attr,
501*4882a593Smuzhiyun &sensor_dev_attr_in3_crit.dev_attr.attr,
502*4882a593Smuzhiyun &sensor_dev_attr_in3_crit_alarm.dev_attr.attr,
503*4882a593Smuzhiyun
504*4882a593Smuzhiyun &sensor_dev_attr_in4_input.dev_attr.attr,
505*4882a593Smuzhiyun &sensor_dev_attr_in4_min.dev_attr.attr,
506*4882a593Smuzhiyun &sensor_dev_attr_in4_max.dev_attr.attr,
507*4882a593Smuzhiyun &sensor_dev_attr_in4_lcrit.dev_attr.attr,
508*4882a593Smuzhiyun &sensor_dev_attr_in4_crit.dev_attr.attr,
509*4882a593Smuzhiyun &sensor_dev_attr_in4_crit_alarm.dev_attr.attr,
510*4882a593Smuzhiyun
511*4882a593Smuzhiyun &sensor_dev_attr_in5_input.dev_attr.attr,
512*4882a593Smuzhiyun &sensor_dev_attr_in5_min.dev_attr.attr,
513*4882a593Smuzhiyun &sensor_dev_attr_in5_max.dev_attr.attr,
514*4882a593Smuzhiyun &sensor_dev_attr_in5_lcrit.dev_attr.attr,
515*4882a593Smuzhiyun &sensor_dev_attr_in5_crit.dev_attr.attr,
516*4882a593Smuzhiyun &sensor_dev_attr_in5_crit_alarm.dev_attr.attr,
517*4882a593Smuzhiyun
518*4882a593Smuzhiyun &sensor_dev_attr_in6_input.dev_attr.attr,
519*4882a593Smuzhiyun &sensor_dev_attr_in6_min.dev_attr.attr,
520*4882a593Smuzhiyun &sensor_dev_attr_in6_max.dev_attr.attr,
521*4882a593Smuzhiyun &sensor_dev_attr_in6_lcrit.dev_attr.attr,
522*4882a593Smuzhiyun &sensor_dev_attr_in6_crit.dev_attr.attr,
523*4882a593Smuzhiyun &sensor_dev_attr_in6_crit_alarm.dev_attr.attr,
524*4882a593Smuzhiyun
525*4882a593Smuzhiyun &sensor_dev_attr_in7_input.dev_attr.attr,
526*4882a593Smuzhiyun &sensor_dev_attr_in7_min.dev_attr.attr,
527*4882a593Smuzhiyun &sensor_dev_attr_in7_max.dev_attr.attr,
528*4882a593Smuzhiyun &sensor_dev_attr_in7_lcrit.dev_attr.attr,
529*4882a593Smuzhiyun &sensor_dev_attr_in7_crit.dev_attr.attr,
530*4882a593Smuzhiyun &sensor_dev_attr_in7_crit_alarm.dev_attr.attr,
531*4882a593Smuzhiyun
532*4882a593Smuzhiyun &sensor_dev_attr_in8_input.dev_attr.attr,
533*4882a593Smuzhiyun &sensor_dev_attr_in8_min.dev_attr.attr,
534*4882a593Smuzhiyun &sensor_dev_attr_in8_max.dev_attr.attr,
535*4882a593Smuzhiyun &sensor_dev_attr_in8_lcrit.dev_attr.attr,
536*4882a593Smuzhiyun &sensor_dev_attr_in8_crit.dev_attr.attr,
537*4882a593Smuzhiyun &sensor_dev_attr_in8_crit_alarm.dev_attr.attr,
538*4882a593Smuzhiyun
539*4882a593Smuzhiyun &sensor_dev_attr_in9_input.dev_attr.attr,
540*4882a593Smuzhiyun &sensor_dev_attr_in9_min.dev_attr.attr,
541*4882a593Smuzhiyun &sensor_dev_attr_in9_max.dev_attr.attr,
542*4882a593Smuzhiyun &sensor_dev_attr_in9_lcrit.dev_attr.attr,
543*4882a593Smuzhiyun &sensor_dev_attr_in9_crit.dev_attr.attr,
544*4882a593Smuzhiyun &sensor_dev_attr_in9_crit_alarm.dev_attr.attr,
545*4882a593Smuzhiyun
546*4882a593Smuzhiyun &sensor_dev_attr_in10_input.dev_attr.attr,
547*4882a593Smuzhiyun &sensor_dev_attr_in10_min.dev_attr.attr,
548*4882a593Smuzhiyun &sensor_dev_attr_in10_max.dev_attr.attr,
549*4882a593Smuzhiyun &sensor_dev_attr_in10_lcrit.dev_attr.attr,
550*4882a593Smuzhiyun &sensor_dev_attr_in10_crit.dev_attr.attr,
551*4882a593Smuzhiyun &sensor_dev_attr_in10_crit_alarm.dev_attr.attr,
552*4882a593Smuzhiyun
553*4882a593Smuzhiyun &sensor_dev_attr_temp1_input.dev_attr.attr,
554*4882a593Smuzhiyun &sensor_dev_attr_temp1_min.dev_attr.attr,
555*4882a593Smuzhiyun &sensor_dev_attr_temp1_max.dev_attr.attr,
556*4882a593Smuzhiyun &sensor_dev_attr_temp1_lcrit.dev_attr.attr,
557*4882a593Smuzhiyun &sensor_dev_attr_temp1_crit.dev_attr.attr,
558*4882a593Smuzhiyun &sensor_dev_attr_temp1_crit_alarm.dev_attr.attr,
559*4882a593Smuzhiyun
560*4882a593Smuzhiyun NULL,
561*4882a593Smuzhiyun };
562*4882a593Smuzhiyun
563*4882a593Smuzhiyun ATTRIBUTE_GROUPS(smm665);
564*4882a593Smuzhiyun
565*4882a593Smuzhiyun static const struct i2c_device_id smm665_id[];
566*4882a593Smuzhiyun
smm665_probe(struct i2c_client * client)567*4882a593Smuzhiyun static int smm665_probe(struct i2c_client *client)
568*4882a593Smuzhiyun {
569*4882a593Smuzhiyun struct i2c_adapter *adapter = client->adapter;
570*4882a593Smuzhiyun struct smm665_data *data;
571*4882a593Smuzhiyun struct device *hwmon_dev;
572*4882a593Smuzhiyun int i, ret;
573*4882a593Smuzhiyun
574*4882a593Smuzhiyun if (!i2c_check_functionality(adapter, I2C_FUNC_SMBUS_BYTE_DATA
575*4882a593Smuzhiyun | I2C_FUNC_SMBUS_WORD_DATA))
576*4882a593Smuzhiyun return -ENODEV;
577*4882a593Smuzhiyun
578*4882a593Smuzhiyun if (i2c_smbus_read_byte_data(client, SMM665_ADOC_ENABLE) < 0)
579*4882a593Smuzhiyun return -ENODEV;
580*4882a593Smuzhiyun
581*4882a593Smuzhiyun data = devm_kzalloc(&client->dev, sizeof(*data), GFP_KERNEL);
582*4882a593Smuzhiyun if (!data)
583*4882a593Smuzhiyun return -ENOMEM;
584*4882a593Smuzhiyun
585*4882a593Smuzhiyun i2c_set_clientdata(client, data);
586*4882a593Smuzhiyun mutex_init(&data->update_lock);
587*4882a593Smuzhiyun
588*4882a593Smuzhiyun data->client = client;
589*4882a593Smuzhiyun data->type = i2c_match_id(smm665_id, client)->driver_data;
590*4882a593Smuzhiyun data->cmdreg = i2c_new_dummy_device(adapter, (client->addr & ~SMM665_REGMASK)
591*4882a593Smuzhiyun | SMM665_CMDREG_BASE);
592*4882a593Smuzhiyun if (IS_ERR(data->cmdreg))
593*4882a593Smuzhiyun return PTR_ERR(data->cmdreg);
594*4882a593Smuzhiyun
595*4882a593Smuzhiyun switch (data->type) {
596*4882a593Smuzhiyun case smm465:
597*4882a593Smuzhiyun case smm665:
598*4882a593Smuzhiyun data->conversion_time = SMM665_ADC_WAIT_SMM665;
599*4882a593Smuzhiyun break;
600*4882a593Smuzhiyun case smm665c:
601*4882a593Smuzhiyun case smm764:
602*4882a593Smuzhiyun case smm766:
603*4882a593Smuzhiyun data->conversion_time = SMM665_ADC_WAIT_SMM766;
604*4882a593Smuzhiyun break;
605*4882a593Smuzhiyun }
606*4882a593Smuzhiyun
607*4882a593Smuzhiyun ret = -ENODEV;
608*4882a593Smuzhiyun if (i2c_smbus_read_byte_data(data->cmdreg, SMM665_MISC8_CMD_STS) < 0)
609*4882a593Smuzhiyun goto out_unregister;
610*4882a593Smuzhiyun
611*4882a593Smuzhiyun /*
612*4882a593Smuzhiyun * Read limits.
613*4882a593Smuzhiyun *
614*4882a593Smuzhiyun * Limit registers start with register SMM665_LIMIT_BASE.
615*4882a593Smuzhiyun * Each channel uses 8 registers, providing four limit values
616*4882a593Smuzhiyun * per channel. Each limit value requires two registers, with the
617*4882a593Smuzhiyun * high byte in the first register and the low byte in the second
618*4882a593Smuzhiyun * register. The first two limits are under limit values, followed
619*4882a593Smuzhiyun * by two over limit values.
620*4882a593Smuzhiyun *
621*4882a593Smuzhiyun * Limit register order matches the ADC register order, so we use
622*4882a593Smuzhiyun * ADC register defines throughout the code to index limit registers.
623*4882a593Smuzhiyun *
624*4882a593Smuzhiyun * We save the first retrieved value both as "critical" and "alarm"
625*4882a593Smuzhiyun * value. The second value overwrites either the critical or the
626*4882a593Smuzhiyun * alarm value, depending on its configuration. This ensures that both
627*4882a593Smuzhiyun * critical and alarm values are initialized, even if both registers are
628*4882a593Smuzhiyun * configured as critical or non-critical.
629*4882a593Smuzhiyun */
630*4882a593Smuzhiyun for (i = 0; i < SMM665_NUM_ADC; i++) {
631*4882a593Smuzhiyun int val;
632*4882a593Smuzhiyun
633*4882a593Smuzhiyun val = smm665_read16(client, SMM665_LIMIT_BASE + i * 8);
634*4882a593Smuzhiyun if (unlikely(val < 0))
635*4882a593Smuzhiyun goto out_unregister;
636*4882a593Smuzhiyun data->critical_min_limit[i] = data->alarm_min_limit[i]
637*4882a593Smuzhiyun = smm665_convert(val, i);
638*4882a593Smuzhiyun val = smm665_read16(client, SMM665_LIMIT_BASE + i * 8 + 2);
639*4882a593Smuzhiyun if (unlikely(val < 0))
640*4882a593Smuzhiyun goto out_unregister;
641*4882a593Smuzhiyun if (smm665_is_critical(val))
642*4882a593Smuzhiyun data->critical_min_limit[i] = smm665_convert(val, i);
643*4882a593Smuzhiyun else
644*4882a593Smuzhiyun data->alarm_min_limit[i] = smm665_convert(val, i);
645*4882a593Smuzhiyun val = smm665_read16(client, SMM665_LIMIT_BASE + i * 8 + 4);
646*4882a593Smuzhiyun if (unlikely(val < 0))
647*4882a593Smuzhiyun goto out_unregister;
648*4882a593Smuzhiyun data->critical_max_limit[i] = data->alarm_max_limit[i]
649*4882a593Smuzhiyun = smm665_convert(val, i);
650*4882a593Smuzhiyun val = smm665_read16(client, SMM665_LIMIT_BASE + i * 8 + 6);
651*4882a593Smuzhiyun if (unlikely(val < 0))
652*4882a593Smuzhiyun goto out_unregister;
653*4882a593Smuzhiyun if (smm665_is_critical(val))
654*4882a593Smuzhiyun data->critical_max_limit[i] = smm665_convert(val, i);
655*4882a593Smuzhiyun else
656*4882a593Smuzhiyun data->alarm_max_limit[i] = smm665_convert(val, i);
657*4882a593Smuzhiyun }
658*4882a593Smuzhiyun
659*4882a593Smuzhiyun hwmon_dev = devm_hwmon_device_register_with_groups(&client->dev,
660*4882a593Smuzhiyun client->name, data,
661*4882a593Smuzhiyun smm665_groups);
662*4882a593Smuzhiyun if (IS_ERR(hwmon_dev)) {
663*4882a593Smuzhiyun ret = PTR_ERR(hwmon_dev);
664*4882a593Smuzhiyun goto out_unregister;
665*4882a593Smuzhiyun }
666*4882a593Smuzhiyun
667*4882a593Smuzhiyun return 0;
668*4882a593Smuzhiyun
669*4882a593Smuzhiyun out_unregister:
670*4882a593Smuzhiyun i2c_unregister_device(data->cmdreg);
671*4882a593Smuzhiyun return ret;
672*4882a593Smuzhiyun }
673*4882a593Smuzhiyun
smm665_remove(struct i2c_client * client)674*4882a593Smuzhiyun static int smm665_remove(struct i2c_client *client)
675*4882a593Smuzhiyun {
676*4882a593Smuzhiyun struct smm665_data *data = i2c_get_clientdata(client);
677*4882a593Smuzhiyun
678*4882a593Smuzhiyun i2c_unregister_device(data->cmdreg);
679*4882a593Smuzhiyun return 0;
680*4882a593Smuzhiyun }
681*4882a593Smuzhiyun
682*4882a593Smuzhiyun static const struct i2c_device_id smm665_id[] = {
683*4882a593Smuzhiyun {"smm465", smm465},
684*4882a593Smuzhiyun {"smm665", smm665},
685*4882a593Smuzhiyun {"smm665c", smm665c},
686*4882a593Smuzhiyun {"smm764", smm764},
687*4882a593Smuzhiyun {"smm766", smm766},
688*4882a593Smuzhiyun {}
689*4882a593Smuzhiyun };
690*4882a593Smuzhiyun
691*4882a593Smuzhiyun MODULE_DEVICE_TABLE(i2c, smm665_id);
692*4882a593Smuzhiyun
693*4882a593Smuzhiyun /* This is the driver that will be inserted */
694*4882a593Smuzhiyun static struct i2c_driver smm665_driver = {
695*4882a593Smuzhiyun .driver = {
696*4882a593Smuzhiyun .name = "smm665",
697*4882a593Smuzhiyun },
698*4882a593Smuzhiyun .probe_new = smm665_probe,
699*4882a593Smuzhiyun .remove = smm665_remove,
700*4882a593Smuzhiyun .id_table = smm665_id,
701*4882a593Smuzhiyun };
702*4882a593Smuzhiyun
703*4882a593Smuzhiyun module_i2c_driver(smm665_driver);
704*4882a593Smuzhiyun
705*4882a593Smuzhiyun MODULE_AUTHOR("Guenter Roeck");
706*4882a593Smuzhiyun MODULE_DESCRIPTION("SMM665 driver");
707*4882a593Smuzhiyun MODULE_LICENSE("GPL");
708