175a6faf6SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2b9d453a5SGwendal Grignou /*
3f225951dSHans de Goede * IIO driver for the MiraMEMS DA217 and DA280 3-axis accelerometer and
4f225951dSHans de Goede * IIO driver for the MiraMEMS DA226 2-axis accelerometer
5f225951dSHans de Goede *
6f225951dSHans de Goede * Copyright (c) 2016 Hans de Goede <hdegoede@redhat.com>
7f225951dSHans de Goede */
8f225951dSHans de Goede
9f225951dSHans de Goede #include <linux/module.h>
10f225951dSHans de Goede #include <linux/i2c.h>
114c42bef0SLuke Ross #include <linux/acpi.h>
12f225951dSHans de Goede #include <linux/iio/iio.h>
13f225951dSHans de Goede #include <linux/iio/sysfs.h>
14f225951dSHans de Goede #include <linux/byteorder/generic.h>
15f225951dSHans de Goede
16f225951dSHans de Goede #define DA280_REG_CHIP_ID 0x01
17f225951dSHans de Goede #define DA280_REG_ACC_X_LSB 0x02
18f225951dSHans de Goede #define DA280_REG_ACC_Y_LSB 0x04
19f225951dSHans de Goede #define DA280_REG_ACC_Z_LSB 0x06
20f225951dSHans de Goede #define DA280_REG_MODE_BW 0x11
21f225951dSHans de Goede
22f225951dSHans de Goede #define DA280_CHIP_ID 0x13
23f225951dSHans de Goede #define DA280_MODE_ENABLE 0x1e
24f225951dSHans de Goede #define DA280_MODE_DISABLE 0x9e
25f225951dSHans de Goede
264c42bef0SLuke Ross enum da280_chipset { da217, da226, da280 };
27f225951dSHans de Goede
28f225951dSHans de Goede /*
29f225951dSHans de Goede * a value of + or -4096 corresponds to + or - 1G
30f225951dSHans de Goede * scale = 9.81 / 4096 = 0.002395019
31f225951dSHans de Goede */
32f225951dSHans de Goede
33f225951dSHans de Goede static const int da280_nscale = 2395019;
34f225951dSHans de Goede
35f225951dSHans de Goede #define DA280_CHANNEL(reg, axis) { \
36f225951dSHans de Goede .type = IIO_ACCEL, \
37f225951dSHans de Goede .address = reg, \
38f225951dSHans de Goede .modified = 1, \
39f225951dSHans de Goede .channel2 = IIO_MOD_##axis, \
40f225951dSHans de Goede .info_mask_separate = BIT(IIO_CHAN_INFO_RAW), \
41f225951dSHans de Goede .info_mask_shared_by_type = BIT(IIO_CHAN_INFO_SCALE), \
42f225951dSHans de Goede }
43f225951dSHans de Goede
44f225951dSHans de Goede static const struct iio_chan_spec da280_channels[] = {
45f225951dSHans de Goede DA280_CHANNEL(DA280_REG_ACC_X_LSB, X),
46f225951dSHans de Goede DA280_CHANNEL(DA280_REG_ACC_Y_LSB, Y),
47f225951dSHans de Goede DA280_CHANNEL(DA280_REG_ACC_Z_LSB, Z),
48f225951dSHans de Goede };
49f225951dSHans de Goede
50f225951dSHans de Goede struct da280_data {
51f225951dSHans de Goede struct i2c_client *client;
52f225951dSHans de Goede };
53f225951dSHans de Goede
da280_enable(struct i2c_client * client,bool enable)54f225951dSHans de Goede static int da280_enable(struct i2c_client *client, bool enable)
55f225951dSHans de Goede {
56f225951dSHans de Goede u8 data = enable ? DA280_MODE_ENABLE : DA280_MODE_DISABLE;
57f225951dSHans de Goede
58f225951dSHans de Goede return i2c_smbus_write_byte_data(client, DA280_REG_MODE_BW, data);
59f225951dSHans de Goede }
60f225951dSHans de Goede
da280_read_raw(struct iio_dev * indio_dev,struct iio_chan_spec const * chan,int * val,int * val2,long mask)61f225951dSHans de Goede static int da280_read_raw(struct iio_dev *indio_dev,
62f225951dSHans de Goede struct iio_chan_spec const *chan,
63f225951dSHans de Goede int *val, int *val2, long mask)
64f225951dSHans de Goede {
65f225951dSHans de Goede struct da280_data *data = iio_priv(indio_dev);
66f225951dSHans de Goede int ret;
67f225951dSHans de Goede
68f225951dSHans de Goede switch (mask) {
69f225951dSHans de Goede case IIO_CHAN_INFO_RAW:
70f225951dSHans de Goede ret = i2c_smbus_read_word_data(data->client, chan->address);
71f225951dSHans de Goede if (ret < 0)
72f225951dSHans de Goede return ret;
73f225951dSHans de Goede /*
74f225951dSHans de Goede * Values are 14 bits, stored as 16 bits with the 2
75f225951dSHans de Goede * least significant bits always 0.
76f225951dSHans de Goede */
77f225951dSHans de Goede *val = (short)ret >> 2;
78f225951dSHans de Goede return IIO_VAL_INT;
79f225951dSHans de Goede case IIO_CHAN_INFO_SCALE:
80f225951dSHans de Goede *val = 0;
81f225951dSHans de Goede *val2 = da280_nscale;
82f225951dSHans de Goede return IIO_VAL_INT_PLUS_NANO;
83f225951dSHans de Goede default:
84f225951dSHans de Goede return -EINVAL;
85f225951dSHans de Goede }
86f225951dSHans de Goede }
87f225951dSHans de Goede
88f225951dSHans de Goede static const struct iio_info da280_info = {
89f225951dSHans de Goede .read_raw = da280_read_raw,
90f225951dSHans de Goede };
91f225951dSHans de Goede
da280_match_acpi_device(struct device * dev)924c42bef0SLuke Ross static enum da280_chipset da280_match_acpi_device(struct device *dev)
934c42bef0SLuke Ross {
944c42bef0SLuke Ross const struct acpi_device_id *id;
954c42bef0SLuke Ross
964c42bef0SLuke Ross id = acpi_match_device(dev->driver->acpi_match_table, dev);
974c42bef0SLuke Ross if (!id)
984c42bef0SLuke Ross return -EINVAL;
994c42bef0SLuke Ross
1004c42bef0SLuke Ross return (enum da280_chipset) id->driver_data;
1014c42bef0SLuke Ross }
1024c42bef0SLuke Ross
da280_disable(void * client)1033d9efa9bSAlexandru Ardelean static void da280_disable(void *client)
1043d9efa9bSAlexandru Ardelean {
1053d9efa9bSAlexandru Ardelean da280_enable(client, false);
1063d9efa9bSAlexandru Ardelean }
1073d9efa9bSAlexandru Ardelean
da280_probe(struct i2c_client * client)108*a8fab448SUwe Kleine-König static int da280_probe(struct i2c_client *client)
109f225951dSHans de Goede {
110*a8fab448SUwe Kleine-König const struct i2c_device_id *id = i2c_client_get_device_id(client);
111f225951dSHans de Goede int ret;
112f225951dSHans de Goede struct iio_dev *indio_dev;
113f225951dSHans de Goede struct da280_data *data;
1144c42bef0SLuke Ross enum da280_chipset chip;
115f225951dSHans de Goede
116f225951dSHans de Goede ret = i2c_smbus_read_byte_data(client, DA280_REG_CHIP_ID);
117f225951dSHans de Goede if (ret != DA280_CHIP_ID)
118f225951dSHans de Goede return (ret < 0) ? ret : -ENODEV;
119f225951dSHans de Goede
120f225951dSHans de Goede indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data));
121f225951dSHans de Goede if (!indio_dev)
122f225951dSHans de Goede return -ENOMEM;
123f225951dSHans de Goede
124f225951dSHans de Goede data = iio_priv(indio_dev);
125f225951dSHans de Goede data->client = client;
126f225951dSHans de Goede
127f225951dSHans de Goede indio_dev->info = &da280_info;
128f225951dSHans de Goede indio_dev->modes = INDIO_DIRECT_MODE;
129f225951dSHans de Goede indio_dev->channels = da280_channels;
1304c42bef0SLuke Ross
1314c42bef0SLuke Ross if (ACPI_HANDLE(&client->dev)) {
1324c42bef0SLuke Ross chip = da280_match_acpi_device(&client->dev);
1334c42bef0SLuke Ross } else {
1344c42bef0SLuke Ross chip = id->driver_data;
1354c42bef0SLuke Ross }
1364c42bef0SLuke Ross
1374c42bef0SLuke Ross if (chip == da217) {
138f225951dSHans de Goede indio_dev->name = "da217";
139f225951dSHans de Goede indio_dev->num_channels = 3;
140f225951dSHans de Goede } else if (chip == da226) {
141f225951dSHans de Goede indio_dev->name = "da226";
142f225951dSHans de Goede indio_dev->num_channels = 2;
143f225951dSHans de Goede } else {
144f225951dSHans de Goede indio_dev->name = "da280";
145f225951dSHans de Goede indio_dev->num_channels = 3;
146f225951dSHans de Goede }
147f225951dSHans de Goede
148f225951dSHans de Goede ret = da280_enable(client, true);
1493d9efa9bSAlexandru Ardelean if (ret < 0)
1503d9efa9bSAlexandru Ardelean return ret;
151f225951dSHans de Goede
152f225951dSHans de Goede ret = devm_add_action_or_reset(&client->dev, da280_disable, client);
1533d9efa9bSAlexandru Ardelean if (ret)
154f225951dSHans de Goede return ret;
155f225951dSHans de Goede
156f225951dSHans de Goede return devm_iio_device_register(&client->dev, indio_dev);
157f225951dSHans de Goede }
158f225951dSHans de Goede
da280_suspend(struct device * dev)159f225951dSHans de Goede static int da280_suspend(struct device *dev)
160f225951dSHans de Goede {
161f225951dSHans de Goede return da280_enable(to_i2c_client(dev), false);
162f225951dSHans de Goede }
163f225951dSHans de Goede
da280_resume(struct device * dev)164f225951dSHans de Goede static int da280_resume(struct device *dev)
165f225951dSHans de Goede {
166ff9231c7SJonathan Cameron return da280_enable(to_i2c_client(dev), true);
167f225951dSHans de Goede }
1684c42bef0SLuke Ross
1694c42bef0SLuke Ross static DEFINE_SIMPLE_DEV_PM_OPS(da280_pm_ops, da280_suspend, da280_resume);
1704c42bef0SLuke Ross
1714c42bef0SLuke Ross static const struct acpi_device_id da280_acpi_match[] = {
1724c42bef0SLuke Ross {"NSA2513", da217},
1734c42bef0SLuke Ross {"MIRAACC", da280},
174f225951dSHans de Goede {},
175f225951dSHans de Goede };
176f225951dSHans de Goede MODULE_DEVICE_TABLE(acpi, da280_acpi_match);
177f225951dSHans de Goede
178f225951dSHans de Goede static const struct i2c_device_id da280_i2c_id[] = {
179f225951dSHans de Goede { "da217", da217 },
180f225951dSHans de Goede { "da226", da226 },
181f225951dSHans de Goede { "da280", da280 },
182f225951dSHans de Goede {}
183f225951dSHans de Goede };
1844c42bef0SLuke Ross MODULE_DEVICE_TABLE(i2c, da280_i2c_id);
185ff9231c7SJonathan Cameron
186f225951dSHans de Goede static struct i2c_driver da280_driver = {
187*a8fab448SUwe Kleine-König .driver = {
188f225951dSHans de Goede .name = "da280",
189f225951dSHans de Goede .acpi_match_table = ACPI_PTR(da280_acpi_match),
190f225951dSHans de Goede .pm = pm_sleep_ptr(&da280_pm_ops),
191f225951dSHans de Goede },
192f225951dSHans de Goede .probe = da280_probe,
193f225951dSHans de Goede .id_table = da280_i2c_id,
194f225951dSHans de Goede };
195f225951dSHans de Goede
196 module_i2c_driver(da280_driver);
197
198 MODULE_AUTHOR("Hans de Goede <hdegoede@redhat.com>");
199 MODULE_DESCRIPTION("MiraMEMS DA280 3-Axis Accelerometer driver");
200 MODULE_LICENSE("GPL v2");
201