1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Industrial I/O driver for Microchip digital potentiometers
4*4882a593Smuzhiyun * Copyright (c) 2018 Axentia Technologies AB
5*4882a593Smuzhiyun * Author: Peter Rosin <peda@axentia.se>
6*4882a593Smuzhiyun *
7*4882a593Smuzhiyun * Datasheet: http://www.microchip.com/downloads/en/DeviceDoc/22147a.pdf
8*4882a593Smuzhiyun *
9*4882a593Smuzhiyun * DEVID #Wipers #Positions Resistor Opts (kOhm)
10*4882a593Smuzhiyun * mcp4017 1 128 5, 10, 50, 100
11*4882a593Smuzhiyun * mcp4018 1 128 5, 10, 50, 100
12*4882a593Smuzhiyun * mcp4019 1 128 5, 10, 50, 100
13*4882a593Smuzhiyun */
14*4882a593Smuzhiyun
15*4882a593Smuzhiyun #include <linux/err.h>
16*4882a593Smuzhiyun #include <linux/i2c.h>
17*4882a593Smuzhiyun #include <linux/iio/iio.h>
18*4882a593Smuzhiyun #include <linux/module.h>
19*4882a593Smuzhiyun #include <linux/mod_devicetable.h>
20*4882a593Smuzhiyun #include <linux/property.h>
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun #define MCP4018_WIPER_MAX 127
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun struct mcp4018_cfg {
25*4882a593Smuzhiyun int kohms;
26*4882a593Smuzhiyun };
27*4882a593Smuzhiyun
28*4882a593Smuzhiyun enum mcp4018_type {
29*4882a593Smuzhiyun MCP4018_502,
30*4882a593Smuzhiyun MCP4018_103,
31*4882a593Smuzhiyun MCP4018_503,
32*4882a593Smuzhiyun MCP4018_104,
33*4882a593Smuzhiyun };
34*4882a593Smuzhiyun
35*4882a593Smuzhiyun static const struct mcp4018_cfg mcp4018_cfg[] = {
36*4882a593Smuzhiyun [MCP4018_502] = { .kohms = 5, },
37*4882a593Smuzhiyun [MCP4018_103] = { .kohms = 10, },
38*4882a593Smuzhiyun [MCP4018_503] = { .kohms = 50, },
39*4882a593Smuzhiyun [MCP4018_104] = { .kohms = 100, },
40*4882a593Smuzhiyun };
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun struct mcp4018_data {
43*4882a593Smuzhiyun struct i2c_client *client;
44*4882a593Smuzhiyun const struct mcp4018_cfg *cfg;
45*4882a593Smuzhiyun };
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun static const struct iio_chan_spec mcp4018_channel = {
48*4882a593Smuzhiyun .type = IIO_RESISTANCE,
49*4882a593Smuzhiyun .indexed = 1,
50*4882a593Smuzhiyun .output = 1,
51*4882a593Smuzhiyun .channel = 0,
52*4882a593Smuzhiyun .info_mask_separate = BIT(IIO_CHAN_INFO_RAW),
53*4882a593Smuzhiyun .info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE),
54*4882a593Smuzhiyun };
55*4882a593Smuzhiyun
mcp4018_read_raw(struct iio_dev * indio_dev,struct iio_chan_spec const * chan,int * val,int * val2,long mask)56*4882a593Smuzhiyun static int mcp4018_read_raw(struct iio_dev *indio_dev,
57*4882a593Smuzhiyun struct iio_chan_spec const *chan,
58*4882a593Smuzhiyun int *val, int *val2, long mask)
59*4882a593Smuzhiyun {
60*4882a593Smuzhiyun struct mcp4018_data *data = iio_priv(indio_dev);
61*4882a593Smuzhiyun s32 ret;
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun switch (mask) {
64*4882a593Smuzhiyun case IIO_CHAN_INFO_RAW:
65*4882a593Smuzhiyun ret = i2c_smbus_read_byte(data->client);
66*4882a593Smuzhiyun if (ret < 0)
67*4882a593Smuzhiyun return ret;
68*4882a593Smuzhiyun *val = ret;
69*4882a593Smuzhiyun return IIO_VAL_INT;
70*4882a593Smuzhiyun case IIO_CHAN_INFO_SCALE:
71*4882a593Smuzhiyun *val = 1000 * data->cfg->kohms;
72*4882a593Smuzhiyun *val2 = MCP4018_WIPER_MAX;
73*4882a593Smuzhiyun return IIO_VAL_FRACTIONAL;
74*4882a593Smuzhiyun }
75*4882a593Smuzhiyun
76*4882a593Smuzhiyun return -EINVAL;
77*4882a593Smuzhiyun }
78*4882a593Smuzhiyun
mcp4018_write_raw(struct iio_dev * indio_dev,struct iio_chan_spec const * chan,int val,int val2,long mask)79*4882a593Smuzhiyun static int mcp4018_write_raw(struct iio_dev *indio_dev,
80*4882a593Smuzhiyun struct iio_chan_spec const *chan,
81*4882a593Smuzhiyun int val, int val2, long mask)
82*4882a593Smuzhiyun {
83*4882a593Smuzhiyun struct mcp4018_data *data = iio_priv(indio_dev);
84*4882a593Smuzhiyun
85*4882a593Smuzhiyun switch (mask) {
86*4882a593Smuzhiyun case IIO_CHAN_INFO_RAW:
87*4882a593Smuzhiyun if (val > MCP4018_WIPER_MAX || val < 0)
88*4882a593Smuzhiyun return -EINVAL;
89*4882a593Smuzhiyun break;
90*4882a593Smuzhiyun default:
91*4882a593Smuzhiyun return -EINVAL;
92*4882a593Smuzhiyun }
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun return i2c_smbus_write_byte(data->client, val);
95*4882a593Smuzhiyun }
96*4882a593Smuzhiyun
97*4882a593Smuzhiyun static const struct iio_info mcp4018_info = {
98*4882a593Smuzhiyun .read_raw = mcp4018_read_raw,
99*4882a593Smuzhiyun .write_raw = mcp4018_write_raw,
100*4882a593Smuzhiyun };
101*4882a593Smuzhiyun
102*4882a593Smuzhiyun static const struct i2c_device_id mcp4018_id[] = {
103*4882a593Smuzhiyun { "mcp4017-502", MCP4018_502 },
104*4882a593Smuzhiyun { "mcp4017-103", MCP4018_103 },
105*4882a593Smuzhiyun { "mcp4017-503", MCP4018_503 },
106*4882a593Smuzhiyun { "mcp4017-104", MCP4018_104 },
107*4882a593Smuzhiyun { "mcp4018-502", MCP4018_502 },
108*4882a593Smuzhiyun { "mcp4018-103", MCP4018_103 },
109*4882a593Smuzhiyun { "mcp4018-503", MCP4018_503 },
110*4882a593Smuzhiyun { "mcp4018-104", MCP4018_104 },
111*4882a593Smuzhiyun { "mcp4019-502", MCP4018_502 },
112*4882a593Smuzhiyun { "mcp4019-103", MCP4018_103 },
113*4882a593Smuzhiyun { "mcp4019-503", MCP4018_503 },
114*4882a593Smuzhiyun { "mcp4019-104", MCP4018_104 },
115*4882a593Smuzhiyun {}
116*4882a593Smuzhiyun };
117*4882a593Smuzhiyun MODULE_DEVICE_TABLE(i2c, mcp4018_id);
118*4882a593Smuzhiyun
119*4882a593Smuzhiyun #define MCP4018_COMPATIBLE(of_compatible, cfg) { \
120*4882a593Smuzhiyun .compatible = of_compatible, \
121*4882a593Smuzhiyun .data = &mcp4018_cfg[cfg], \
122*4882a593Smuzhiyun }
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun static const struct of_device_id mcp4018_of_match[] = {
125*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4017-502", MCP4018_502),
126*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4017-103", MCP4018_103),
127*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4017-503", MCP4018_503),
128*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4017-104", MCP4018_104),
129*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4018-502", MCP4018_502),
130*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4018-103", MCP4018_103),
131*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4018-503", MCP4018_503),
132*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4018-104", MCP4018_104),
133*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4019-502", MCP4018_502),
134*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4019-103", MCP4018_103),
135*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4019-503", MCP4018_503),
136*4882a593Smuzhiyun MCP4018_COMPATIBLE("microchip,mcp4019-104", MCP4018_104),
137*4882a593Smuzhiyun { /* sentinel */ }
138*4882a593Smuzhiyun };
139*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, mcp4018_of_match);
140*4882a593Smuzhiyun
mcp4018_probe(struct i2c_client * client)141*4882a593Smuzhiyun static int mcp4018_probe(struct i2c_client *client)
142*4882a593Smuzhiyun {
143*4882a593Smuzhiyun struct device *dev = &client->dev;
144*4882a593Smuzhiyun struct mcp4018_data *data;
145*4882a593Smuzhiyun struct iio_dev *indio_dev;
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun if (!i2c_check_functionality(client->adapter,
148*4882a593Smuzhiyun I2C_FUNC_SMBUS_BYTE)) {
149*4882a593Smuzhiyun dev_err(dev, "SMBUS Byte transfers not supported\n");
150*4882a593Smuzhiyun return -EOPNOTSUPP;
151*4882a593Smuzhiyun }
152*4882a593Smuzhiyun
153*4882a593Smuzhiyun indio_dev = devm_iio_device_alloc(dev, sizeof(*data));
154*4882a593Smuzhiyun if (!indio_dev)
155*4882a593Smuzhiyun return -ENOMEM;
156*4882a593Smuzhiyun data = iio_priv(indio_dev);
157*4882a593Smuzhiyun i2c_set_clientdata(client, indio_dev);
158*4882a593Smuzhiyun data->client = client;
159*4882a593Smuzhiyun
160*4882a593Smuzhiyun data->cfg = device_get_match_data(dev);
161*4882a593Smuzhiyun if (!data->cfg)
162*4882a593Smuzhiyun data->cfg = &mcp4018_cfg[i2c_match_id(mcp4018_id, client)->driver_data];
163*4882a593Smuzhiyun
164*4882a593Smuzhiyun indio_dev->info = &mcp4018_info;
165*4882a593Smuzhiyun indio_dev->channels = &mcp4018_channel;
166*4882a593Smuzhiyun indio_dev->num_channels = 1;
167*4882a593Smuzhiyun indio_dev->name = client->name;
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun return devm_iio_device_register(dev, indio_dev);
170*4882a593Smuzhiyun }
171*4882a593Smuzhiyun
172*4882a593Smuzhiyun static struct i2c_driver mcp4018_driver = {
173*4882a593Smuzhiyun .driver = {
174*4882a593Smuzhiyun .name = "mcp4018",
175*4882a593Smuzhiyun .of_match_table = mcp4018_of_match,
176*4882a593Smuzhiyun },
177*4882a593Smuzhiyun .probe_new = mcp4018_probe,
178*4882a593Smuzhiyun .id_table = mcp4018_id,
179*4882a593Smuzhiyun };
180*4882a593Smuzhiyun
181*4882a593Smuzhiyun module_i2c_driver(mcp4018_driver);
182*4882a593Smuzhiyun
183*4882a593Smuzhiyun MODULE_AUTHOR("Peter Rosin <peda@axentia.se>");
184*4882a593Smuzhiyun MODULE_DESCRIPTION("MCP4018 digital potentiometer");
185*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
186