xref: /openbmc/linux/drivers/iio/accel/da280.c (revision ba54ff1fb662215de683777f815b9e96276d55cf)
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