1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * STMicroelectronics sensors spi library driver
4  *
5  * Copyright 2012-2013 STMicroelectronics Inc.
6  *
7  * Denis Ciocca <denis.ciocca@st.com>
8  */
9 
10 #include <linux/kernel.h>
11 #include <linux/module.h>
12 #include <linux/slab.h>
13 #include <linux/iio/iio.h>
14 #include <linux/property.h>
15 #include <linux/regmap.h>
16 
17 #include <linux/iio/common/st_sensors_spi.h>
18 #include "st_sensors_core.h"
19 
20 #define ST_SENSORS_SPI_MULTIREAD	0xc0
21 
22 static const struct regmap_config st_sensors_spi_regmap_config = {
23 	.reg_bits = 8,
24 	.val_bits = 8,
25 };
26 
27 static const struct regmap_config st_sensors_spi_regmap_multiread_bit_config = {
28 	.reg_bits = 8,
29 	.val_bits = 8,
30 	.read_flag_mask = ST_SENSORS_SPI_MULTIREAD,
31 };
32 
33 /*
34  * st_sensors_is_spi_3_wire() - check if SPI 3-wire mode has been selected
35  * @spi: spi device reference.
36  *
37  * Return: true if SPI 3-wire mode is selected, false otherwise.
38  */
st_sensors_is_spi_3_wire(struct spi_device * spi)39 static bool st_sensors_is_spi_3_wire(struct spi_device *spi)
40 {
41 	struct st_sensors_platform_data *pdata;
42 	struct device *dev = &spi->dev;
43 
44 	if (device_property_read_bool(dev, "spi-3wire"))
45 		return true;
46 
47 	pdata = dev_get_platdata(dev);
48 	if (pdata && pdata->spi_3wire)
49 		return true;
50 
51 	return false;
52 }
53 
54 /*
55  * st_sensors_configure_spi_3_wire() - configure SPI 3-wire if needed
56  * @spi: spi device reference.
57  * @settings: sensor specific settings reference.
58  *
59  * Return: 0 on success, else a negative error code.
60  */
st_sensors_configure_spi_3_wire(struct spi_device * spi,struct st_sensor_settings * settings)61 static int st_sensors_configure_spi_3_wire(struct spi_device *spi,
62 					   struct st_sensor_settings *settings)
63 {
64 	if (settings->sim.addr) {
65 		u8 buffer[] = {
66 			settings->sim.addr,
67 			settings->sim.value
68 		};
69 
70 		return spi_write(spi, buffer, 2);
71 	}
72 
73 	return 0;
74 }
75 
76 /*
77  * st_sensors_spi_configure() - configure SPI interface
78  * @indio_dev: IIO device reference.
79  * @spi: spi device reference.
80  *
81  * Return: 0 on success, else a negative error code.
82  */
st_sensors_spi_configure(struct iio_dev * indio_dev,struct spi_device * spi)83 int st_sensors_spi_configure(struct iio_dev *indio_dev,
84 			     struct spi_device *spi)
85 {
86 	struct st_sensor_data *sdata = iio_priv(indio_dev);
87 	const struct regmap_config *config;
88 	int err;
89 
90 	if (st_sensors_is_spi_3_wire(spi)) {
91 		err = st_sensors_configure_spi_3_wire(spi,
92 						      sdata->sensor_settings);
93 		if (err < 0)
94 			return err;
95 	}
96 
97 	if (sdata->sensor_settings->multi_read_bit)
98 		config = &st_sensors_spi_regmap_multiread_bit_config;
99 	else
100 		config = &st_sensors_spi_regmap_config;
101 
102 	sdata->regmap = devm_regmap_init_spi(spi, config);
103 	if (IS_ERR(sdata->regmap)) {
104 		dev_err(&spi->dev, "Failed to register spi regmap (%ld)\n",
105 			PTR_ERR(sdata->regmap));
106 		return PTR_ERR(sdata->regmap);
107 	}
108 
109 	spi_set_drvdata(spi, indio_dev);
110 
111 	indio_dev->name = spi->modalias;
112 
113 	sdata->dev = &spi->dev;
114 	sdata->irq = spi->irq;
115 
116 	return 0;
117 }
118 EXPORT_SYMBOL(st_sensors_spi_configure);
119 
120 MODULE_AUTHOR("Denis Ciocca <denis.ciocca@st.com>");
121 MODULE_DESCRIPTION("STMicroelectronics ST-sensors spi driver");
122 MODULE_LICENSE("GPL v2");
123