1*128e5ebeSSiratul Islam // SPDX-License-Identifier: GPL-2.0 OR BSD-2-Clause
2*128e5ebeSSiratul Islam /*
3*128e5ebeSSiratul Islam * Support for ST VL53L1X FlightSense ToF Ranging Sensor on a i2c bus.
4*128e5ebeSSiratul Islam *
5*128e5ebeSSiratul Islam * Copyright (C) 2026 Siratul Islam <email@sirat.me>
6*128e5ebeSSiratul Islam *
7*128e5ebeSSiratul Islam * Datasheet available at
8*128e5ebeSSiratul Islam * <https://www.st.com/resource/en/datasheet/vl53l1x.pdf>
9*128e5ebeSSiratul Islam *
10*128e5ebeSSiratul Islam * Default 7-bit i2c slave address 0x29.
11*128e5ebeSSiratul Islam *
12*128e5ebeSSiratul Islam * The VL53L1X requires a firmware configuration blob to be loaded at boot.
13*128e5ebeSSiratul Islam * Register values for the default configuration are taken from
14*128e5ebeSSiratul Islam * ST's VL53L1X Ultra Lite Driver (STSW-IMG009).
15*128e5ebeSSiratul Islam */
16*128e5ebeSSiratul Islam
17*128e5ebeSSiratul Islam #include <linux/array_size.h>
18*128e5ebeSSiratul Islam #include <linux/bits.h>
19*128e5ebeSSiratul Islam #include <linux/bitfield.h>
20*128e5ebeSSiratul Islam #include <linux/completion.h>
21*128e5ebeSSiratul Islam #include <linux/delay.h>
22*128e5ebeSSiratul Islam #include <linux/dev_printk.h>
23*128e5ebeSSiratul Islam #include <linux/err.h>
24*128e5ebeSSiratul Islam #include <linux/i2c.h>
25*128e5ebeSSiratul Islam #include <linux/interrupt.h>
26*128e5ebeSSiratul Islam #include <linux/math.h>
27*128e5ebeSSiratul Islam #include <linux/mod_devicetable.h>
28*128e5ebeSSiratul Islam #include <linux/module.h>
29*128e5ebeSSiratul Islam #include <linux/regmap.h>
30*128e5ebeSSiratul Islam #include <linux/regulator/consumer.h>
31*128e5ebeSSiratul Islam #include <linux/reset.h>
32*128e5ebeSSiratul Islam #include <linux/time.h>
33*128e5ebeSSiratul Islam #include <linux/types.h>
34*128e5ebeSSiratul Islam
35*128e5ebeSSiratul Islam #include <asm/byteorder.h>
36*128e5ebeSSiratul Islam
37*128e5ebeSSiratul Islam #include <linux/iio/buffer.h>
38*128e5ebeSSiratul Islam #include <linux/iio/iio.h>
39*128e5ebeSSiratul Islam #include <linux/iio/trigger.h>
40*128e5ebeSSiratul Islam #include <linux/iio/trigger_consumer.h>
41*128e5ebeSSiratul Islam #include <linux/iio/triggered_buffer.h>
42*128e5ebeSSiratul Islam
43*128e5ebeSSiratul Islam #define VL53L1X_REG_SOFT_RESET 0x0000
44*128e5ebeSSiratul Islam #define VL53L1X_REG_VHV_CONFIG__TIMEOUT_MACROP_LOOP_BOUND 0x0008
45*128e5ebeSSiratul Islam #define VL53L1X_REG_VHV_CONFIG__INIT 0x000B
46*128e5ebeSSiratul Islam #define VL53L1X_REG_GPIO_HV_MUX__CTRL 0x0030
47*128e5ebeSSiratul Islam #define VL53L1X_REG_GPIO__TIO_HV_STATUS 0x0031
48*128e5ebeSSiratul Islam #define VL53L1X_REG_SYSTEM__INTERRUPT_CONFIG_GPIO 0x0046
49*128e5ebeSSiratul Islam #define VL53L1X_REG_PHASECAL_CONFIG__TIMEOUT_MACROP 0x004B
50*128e5ebeSSiratul Islam #define VL53L1X_REG_RANGE_CONFIG__TIMEOUT_MACROP_A 0x005E
51*128e5ebeSSiratul Islam #define VL53L1X_REG_RANGE_CONFIG__VCSEL_PERIOD_A 0x0060
52*128e5ebeSSiratul Islam #define VL53L1X_REG_RANGE_CONFIG__TIMEOUT_MACROP_B 0x0061
53*128e5ebeSSiratul Islam #define VL53L1X_REG_RANGE_CONFIG__VCSEL_PERIOD_B 0x0063
54*128e5ebeSSiratul Islam #define VL53L1X_REG_RANGE_CONFIG__VALID_PHASE_HIGH 0x0069
55*128e5ebeSSiratul Islam #define VL53L1X_REG_SYSTEM__INTERMEASUREMENT_PERIOD 0x006C
56*128e5ebeSSiratul Islam #define VL53L1X_REG_SD_CONFIG__WOI_SD0 0x0078
57*128e5ebeSSiratul Islam #define VL53L1X_REG_SD_CONFIG__WOI_SD1 0x0079
58*128e5ebeSSiratul Islam #define VL53L1X_REG_SD_CONFIG__INITIAL_PHASE_SD0 0x007A
59*128e5ebeSSiratul Islam #define VL53L1X_REG_SD_CONFIG__INITIAL_PHASE_SD1 0x007B
60*128e5ebeSSiratul Islam #define VL53L1X_REG_SYSTEM__INTERRUPT_CLEAR 0x0086
61*128e5ebeSSiratul Islam #define VL53L1X_REG_SYSTEM__MODE_START 0x0087
62*128e5ebeSSiratul Islam #define VL53L1X_REG_RESULT__RANGE_STATUS 0x0089
63*128e5ebeSSiratul Islam #define VL53L1X_REG_RESULT__FINAL_CROSSTALK_CORRECTED_RANGE_MM_SD0 0x0096
64*128e5ebeSSiratul Islam #define VL53L1X_REG_RESULT__OSC_CALIBRATE_VAL 0x00DE
65*128e5ebeSSiratul Islam #define VL53L1X_REG_FIRMWARE__SYSTEM_STATUS 0x00E5
66*128e5ebeSSiratul Islam #define VL53L1X_REG_IDENTIFICATION__MODEL_ID 0x010F
67*128e5ebeSSiratul Islam #define VL53L1X_REG_DEFAULT_CONFIG 0x002D
68*128e5ebeSSiratul Islam
69*128e5ebeSSiratul Islam #define VL53L1X_MODEL_ID_VAL 0xEACC
70*128e5ebeSSiratul Islam
71*128e5ebeSSiratul Islam #define VL53L1X_MODE_START_TIMED 0x40
72*128e5ebeSSiratul Islam #define VL53L1X_MODE_START_STOP 0x00
73*128e5ebeSSiratul Islam
74*128e5ebeSSiratul Islam #define VL53L1X_INT_NEW_SAMPLE_READY 0x02
75*128e5ebeSSiratul Islam
76*128e5ebeSSiratul Islam #define VL53L1X_GPIO_HV_MUX_POLARITY BIT(4)
77*128e5ebeSSiratul Islam
78*128e5ebeSSiratul Islam #define VL53L1X_VHV_LOOP_BOUND_TWO 0x09
79*128e5ebeSSiratul Islam
80*128e5ebeSSiratul Islam #define VL53L1X_RANGE_STATUS_MASK GENMASK(4, 0)
81*128e5ebeSSiratul Islam #define VL53L1X_RANGE_STATUS_VALID 9
82*128e5ebeSSiratul Islam
83*128e5ebeSSiratul Islam #define VL53L1X_OSC_CALIBRATE_MASK GENMASK(9, 0)
84*128e5ebeSSiratul Islam
85*128e5ebeSSiratul Islam /* Inter-measurement period uses PLL divider with 1.075 oscillator correction */
86*128e5ebeSSiratul Islam static const struct u32_fract vl53l1x_osc_correction = {
87*128e5ebeSSiratul Islam .numerator = 1075,
88*128e5ebeSSiratul Islam .denominator = 1000,
89*128e5ebeSSiratul Islam };
90*128e5ebeSSiratul Islam
91*128e5ebeSSiratul Islam enum vl53l1x_distance_mode {
92*128e5ebeSSiratul Islam VL53L1X_SHORT,
93*128e5ebeSSiratul Islam VL53L1X_LONG,
94*128e5ebeSSiratul Islam };
95*128e5ebeSSiratul Islam
96*128e5ebeSSiratul Islam struct vl53l1x_data {
97*128e5ebeSSiratul Islam struct regmap *regmap;
98*128e5ebeSSiratul Islam struct completion completion;
99*128e5ebeSSiratul Islam struct reset_control *xshut_reset;
100*128e5ebeSSiratul Islam enum vl53l1x_distance_mode distance_mode;
101*128e5ebeSSiratul Islam u8 gpio_polarity;
102*128e5ebeSSiratul Islam int irq;
103*128e5ebeSSiratul Islam };
104*128e5ebeSSiratul Islam
105*128e5ebeSSiratul Islam static const struct regmap_range vl53l1x_volatile_ranges[] = {
106*128e5ebeSSiratul Islam regmap_reg_range(VL53L1X_REG_GPIO__TIO_HV_STATUS,
107*128e5ebeSSiratul Islam VL53L1X_REG_GPIO__TIO_HV_STATUS),
108*128e5ebeSSiratul Islam regmap_reg_range(VL53L1X_REG_RESULT__RANGE_STATUS,
109*128e5ebeSSiratul Islam VL53L1X_REG_RESULT__RANGE_STATUS),
110*128e5ebeSSiratul Islam regmap_reg_range(VL53L1X_REG_RESULT__FINAL_CROSSTALK_CORRECTED_RANGE_MM_SD0,
111*128e5ebeSSiratul Islam VL53L1X_REG_RESULT__FINAL_CROSSTALK_CORRECTED_RANGE_MM_SD0 + 1),
112*128e5ebeSSiratul Islam regmap_reg_range(VL53L1X_REG_RESULT__OSC_CALIBRATE_VAL,
113*128e5ebeSSiratul Islam VL53L1X_REG_RESULT__OSC_CALIBRATE_VAL + 1),
114*128e5ebeSSiratul Islam regmap_reg_range(VL53L1X_REG_FIRMWARE__SYSTEM_STATUS,
115*128e5ebeSSiratul Islam VL53L1X_REG_FIRMWARE__SYSTEM_STATUS),
116*128e5ebeSSiratul Islam };
117*128e5ebeSSiratul Islam
118*128e5ebeSSiratul Islam static const struct regmap_access_table vl53l1x_volatile_table = {
119*128e5ebeSSiratul Islam .yes_ranges = vl53l1x_volatile_ranges,
120*128e5ebeSSiratul Islam .n_yes_ranges = ARRAY_SIZE(vl53l1x_volatile_ranges),
121*128e5ebeSSiratul Islam };
122*128e5ebeSSiratul Islam
123*128e5ebeSSiratul Islam static const struct regmap_range vl53l1x_write_only_ranges[] = {
124*128e5ebeSSiratul Islam regmap_reg_range(VL53L1X_REG_SOFT_RESET, VL53L1X_REG_SOFT_RESET),
125*128e5ebeSSiratul Islam regmap_reg_range(VL53L1X_REG_SYSTEM__INTERRUPT_CLEAR,
126*128e5ebeSSiratul Islam VL53L1X_REG_SYSTEM__MODE_START),
127*128e5ebeSSiratul Islam };
128*128e5ebeSSiratul Islam
129*128e5ebeSSiratul Islam static const struct regmap_access_table vl53l1x_readable_table = {
130*128e5ebeSSiratul Islam .no_ranges = vl53l1x_write_only_ranges,
131*128e5ebeSSiratul Islam .n_no_ranges = ARRAY_SIZE(vl53l1x_write_only_ranges),
132*128e5ebeSSiratul Islam };
133*128e5ebeSSiratul Islam
134*128e5ebeSSiratul Islam static const struct regmap_config vl53l1x_regmap_config = {
135*128e5ebeSSiratul Islam .reg_bits = 16,
136*128e5ebeSSiratul Islam .val_bits = 8,
137*128e5ebeSSiratul Islam /* MODEL_ID is 16-bit. +1 covers the second byte at 0x0110 */
138*128e5ebeSSiratul Islam .max_register = VL53L1X_REG_IDENTIFICATION__MODEL_ID + 1,
139*128e5ebeSSiratul Islam .cache_type = REGCACHE_MAPLE,
140*128e5ebeSSiratul Islam .volatile_table = &vl53l1x_volatile_table,
141*128e5ebeSSiratul Islam .rd_table = &vl53l1x_readable_table,
142*128e5ebeSSiratul Islam };
143*128e5ebeSSiratul Islam
vl53l1x_read_u16(struct vl53l1x_data * data,u16 reg,u16 * val)144*128e5ebeSSiratul Islam static int vl53l1x_read_u16(struct vl53l1x_data *data, u16 reg, u16 *val)
145*128e5ebeSSiratul Islam {
146*128e5ebeSSiratul Islam __be16 buf;
147*128e5ebeSSiratul Islam int ret;
148*128e5ebeSSiratul Islam
149*128e5ebeSSiratul Islam ret = regmap_bulk_read(data->regmap, reg, &buf, sizeof(buf));
150*128e5ebeSSiratul Islam if (ret)
151*128e5ebeSSiratul Islam return ret;
152*128e5ebeSSiratul Islam
153*128e5ebeSSiratul Islam *val = be16_to_cpu(buf);
154*128e5ebeSSiratul Islam return 0;
155*128e5ebeSSiratul Islam }
156*128e5ebeSSiratul Islam
vl53l1x_write_u16(struct vl53l1x_data * data,u16 reg,u16 val)157*128e5ebeSSiratul Islam static int vl53l1x_write_u16(struct vl53l1x_data *data, u16 reg, u16 val)
158*128e5ebeSSiratul Islam {
159*128e5ebeSSiratul Islam __be16 buf = cpu_to_be16(val);
160*128e5ebeSSiratul Islam
161*128e5ebeSSiratul Islam return regmap_bulk_write(data->regmap, reg, &buf, sizeof(buf));
162*128e5ebeSSiratul Islam }
163*128e5ebeSSiratul Islam
vl53l1x_write_u32(struct vl53l1x_data * data,u16 reg,u32 val)164*128e5ebeSSiratul Islam static int vl53l1x_write_u32(struct vl53l1x_data *data, u16 reg, u32 val)
165*128e5ebeSSiratul Islam {
166*128e5ebeSSiratul Islam __be32 buf = cpu_to_be32(val);
167*128e5ebeSSiratul Islam
168*128e5ebeSSiratul Islam return regmap_bulk_write(data->regmap, reg, &buf, sizeof(buf));
169*128e5ebeSSiratul Islam }
170*128e5ebeSSiratul Islam
vl53l1x_clear_irq(struct vl53l1x_data * data)171*128e5ebeSSiratul Islam static int vl53l1x_clear_irq(struct vl53l1x_data *data)
172*128e5ebeSSiratul Islam {
173*128e5ebeSSiratul Islam return regmap_write(data->regmap, VL53L1X_REG_SYSTEM__INTERRUPT_CLEAR, 0x01);
174*128e5ebeSSiratul Islam }
175*128e5ebeSSiratul Islam
vl53l1x_start_ranging(struct vl53l1x_data * data)176*128e5ebeSSiratul Islam static int vl53l1x_start_ranging(struct vl53l1x_data *data)
177*128e5ebeSSiratul Islam {
178*128e5ebeSSiratul Islam int ret;
179*128e5ebeSSiratul Islam
180*128e5ebeSSiratul Islam ret = vl53l1x_clear_irq(data);
181*128e5ebeSSiratul Islam if (ret)
182*128e5ebeSSiratul Islam return ret;
183*128e5ebeSSiratul Islam
184*128e5ebeSSiratul Islam return regmap_write(data->regmap, VL53L1X_REG_SYSTEM__MODE_START,
185*128e5ebeSSiratul Islam VL53L1X_MODE_START_TIMED);
186*128e5ebeSSiratul Islam }
187*128e5ebeSSiratul Islam
vl53l1x_stop_ranging(struct vl53l1x_data * data)188*128e5ebeSSiratul Islam static int vl53l1x_stop_ranging(struct vl53l1x_data *data)
189*128e5ebeSSiratul Islam {
190*128e5ebeSSiratul Islam return regmap_write(data->regmap, VL53L1X_REG_SYSTEM__MODE_START,
191*128e5ebeSSiratul Islam VL53L1X_MODE_START_STOP);
192*128e5ebeSSiratul Islam }
193*128e5ebeSSiratul Islam
194*128e5ebeSSiratul Islam /*
195*128e5ebeSSiratul Islam * Default configuration blob from ST's VL53L1X Ultra Lite Driver
196*128e5ebeSSiratul Islam * (STSW-IMG009).
197*128e5ebeSSiratul Islam */
198*128e5ebeSSiratul Islam static const u8 vl53l1x_default_config[] = {
199*128e5ebeSSiratul Islam 0x00, 0x00, 0x00, 0x01, 0x02, 0x00, 0x02, 0x08, /* reg 0x2d..0x34 */
200*128e5ebeSSiratul Islam 0x00, 0x08, 0x10, 0x01, 0x01, 0x00, 0x00, 0x00, /* reg 0x35..0x3c */
201*128e5ebeSSiratul Islam 0x00, 0xFF, 0x00, 0x0F, 0x00, 0x00, 0x00, 0x00, /* reg 0x3d..0x44 */
202*128e5ebeSSiratul Islam 0x00, 0x20, 0x0B, 0x00, 0x00, 0x02, 0x0A, 0x21, /* reg 0x45..0x4c */
203*128e5ebeSSiratul Islam 0x00, 0x00, 0x05, 0x00, 0x00, 0x00, 0x00, 0xC8, /* reg 0x4d..0x54 */
204*128e5ebeSSiratul Islam 0x00, 0x00, 0x38, 0xFF, 0x01, 0x00, 0x08, 0x00, /* reg 0x55..0x5c */
205*128e5ebeSSiratul Islam 0x00, 0x01, 0xCC, 0x0F, 0x01, 0xF1, 0x0D, 0x01, /* reg 0x5d..0x64 */
206*128e5ebeSSiratul Islam 0x68, 0x00, 0x80, 0x08, 0xB8, 0x00, 0x00, 0x00, /* reg 0x65..0x6c */
207*128e5ebeSSiratul Islam 0x00, 0x0F, 0x89, 0x00, 0x00, 0x00, 0x00, 0x00, /* reg 0x6d..0x74 */
208*128e5ebeSSiratul Islam 0x00, 0x00, 0x01, 0x0F, 0x0D, 0x0E, 0x0E, 0x00, /* reg 0x75..0x7c */
209*128e5ebeSSiratul Islam 0x00, 0x02, 0xC7, 0xFF, 0x9B, 0x00, 0x00, 0x00, /* reg 0x7d..0x84 */
210*128e5ebeSSiratul Islam 0x01, 0x00, 0x00, /* reg 0x85..0x87 */
211*128e5ebeSSiratul Islam };
212*128e5ebeSSiratul Islam
vl53l1x_chip_init(struct vl53l1x_data * data)213*128e5ebeSSiratul Islam static int vl53l1x_chip_init(struct vl53l1x_data *data)
214*128e5ebeSSiratul Islam {
215*128e5ebeSSiratul Islam struct device *dev = regmap_get_device(data->regmap);
216*128e5ebeSSiratul Islam unsigned int val;
217*128e5ebeSSiratul Islam u16 model_id;
218*128e5ebeSSiratul Islam int ret;
219*128e5ebeSSiratul Islam
220*128e5ebeSSiratul Islam if (!data->xshut_reset) {
221*128e5ebeSSiratul Islam ret = regmap_write(data->regmap, VL53L1X_REG_SOFT_RESET, 0x00);
222*128e5ebeSSiratul Islam if (ret)
223*128e5ebeSSiratul Islam return ret;
224*128e5ebeSSiratul Islam fsleep(100); /* conservative reset pulse, no spec */
225*128e5ebeSSiratul Islam
226*128e5ebeSSiratul Islam ret = regmap_write(data->regmap, VL53L1X_REG_SOFT_RESET, 0x01);
227*128e5ebeSSiratul Islam if (ret)
228*128e5ebeSSiratul Islam return ret;
229*128e5ebeSSiratul Islam fsleep(1000); /* conservative boot wait, no spec */
230*128e5ebeSSiratul Islam }
231*128e5ebeSSiratul Islam
232*128e5ebeSSiratul Islam ret = regmap_read_poll_timeout(data->regmap,
233*128e5ebeSSiratul Islam VL53L1X_REG_FIRMWARE__SYSTEM_STATUS, val,
234*128e5ebeSSiratul Islam val & BIT(0),
235*128e5ebeSSiratul Islam 1 * USEC_PER_MSEC,
236*128e5ebeSSiratul Islam 100 * USEC_PER_MSEC);
237*128e5ebeSSiratul Islam if (ret)
238*128e5ebeSSiratul Islam return dev_err_probe(dev, ret, "firmware boot timeout\n");
239*128e5ebeSSiratul Islam
240*128e5ebeSSiratul Islam ret = vl53l1x_read_u16(data, VL53L1X_REG_IDENTIFICATION__MODEL_ID,
241*128e5ebeSSiratul Islam &model_id);
242*128e5ebeSSiratul Islam if (ret)
243*128e5ebeSSiratul Islam return ret;
244*128e5ebeSSiratul Islam
245*128e5ebeSSiratul Islam if (model_id != VL53L1X_MODEL_ID_VAL)
246*128e5ebeSSiratul Islam dev_info(dev, "unknown model id: 0x%04x, continuing\n", model_id);
247*128e5ebeSSiratul Islam
248*128e5ebeSSiratul Islam ret = regmap_bulk_write(data->regmap, VL53L1X_REG_DEFAULT_CONFIG,
249*128e5ebeSSiratul Islam vl53l1x_default_config,
250*128e5ebeSSiratul Islam sizeof(vl53l1x_default_config));
251*128e5ebeSSiratul Islam if (ret)
252*128e5ebeSSiratul Islam return ret;
253*128e5ebeSSiratul Islam
254*128e5ebeSSiratul Islam ret = regmap_read(data->regmap, VL53L1X_REG_GPIO_HV_MUX__CTRL, &val);
255*128e5ebeSSiratul Islam if (ret)
256*128e5ebeSSiratul Islam return ret;
257*128e5ebeSSiratul Islam data->gpio_polarity = !!(val & VL53L1X_GPIO_HV_MUX_POLARITY);
258*128e5ebeSSiratul Islam
259*128e5ebeSSiratul Islam /* Initial ranging cycle for VHV calibration */
260*128e5ebeSSiratul Islam ret = vl53l1x_start_ranging(data);
261*128e5ebeSSiratul Islam if (ret)
262*128e5ebeSSiratul Islam return ret;
263*128e5ebeSSiratul Islam
264*128e5ebeSSiratul Islam /* 1ms poll, 1s timeout covers max timing budgets (per ST Ultra Lite Driver) */
265*128e5ebeSSiratul Islam ret = regmap_read_poll_timeout(data->regmap,
266*128e5ebeSSiratul Islam VL53L1X_REG_GPIO__TIO_HV_STATUS, val,
267*128e5ebeSSiratul Islam (val & 1) != data->gpio_polarity,
268*128e5ebeSSiratul Islam 1 * USEC_PER_MSEC,
269*128e5ebeSSiratul Islam 1000 * USEC_PER_MSEC);
270*128e5ebeSSiratul Islam if (ret)
271*128e5ebeSSiratul Islam return ret;
272*128e5ebeSSiratul Islam
273*128e5ebeSSiratul Islam ret = vl53l1x_clear_irq(data);
274*128e5ebeSSiratul Islam if (ret)
275*128e5ebeSSiratul Islam return ret;
276*128e5ebeSSiratul Islam
277*128e5ebeSSiratul Islam ret = vl53l1x_stop_ranging(data);
278*128e5ebeSSiratul Islam if (ret)
279*128e5ebeSSiratul Islam return ret;
280*128e5ebeSSiratul Islam
281*128e5ebeSSiratul Islam ret = regmap_write(data->regmap,
282*128e5ebeSSiratul Islam VL53L1X_REG_VHV_CONFIG__TIMEOUT_MACROP_LOOP_BOUND,
283*128e5ebeSSiratul Islam VL53L1X_VHV_LOOP_BOUND_TWO);
284*128e5ebeSSiratul Islam if (ret)
285*128e5ebeSSiratul Islam return ret;
286*128e5ebeSSiratul Islam
287*128e5ebeSSiratul Islam return regmap_write(data->regmap, VL53L1X_REG_VHV_CONFIG__INIT, 0x00);
288*128e5ebeSSiratul Islam }
289*128e5ebeSSiratul Islam
290*128e5ebeSSiratul Islam static const struct reg_sequence vl53l1x_mode_short[] = {
291*128e5ebeSSiratul Islam { VL53L1X_REG_PHASECAL_CONFIG__TIMEOUT_MACROP, 0x14 },
292*128e5ebeSSiratul Islam { VL53L1X_REG_RANGE_CONFIG__VCSEL_PERIOD_A, 0x07 },
293*128e5ebeSSiratul Islam { VL53L1X_REG_RANGE_CONFIG__VCSEL_PERIOD_B, 0x05 },
294*128e5ebeSSiratul Islam { VL53L1X_REG_RANGE_CONFIG__VALID_PHASE_HIGH, 0x38 },
295*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__WOI_SD0, 0x07 },
296*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__WOI_SD1, 0x05 },
297*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__INITIAL_PHASE_SD0, 0x06 },
298*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__INITIAL_PHASE_SD1, 0x06 },
299*128e5ebeSSiratul Islam };
300*128e5ebeSSiratul Islam
301*128e5ebeSSiratul Islam static const struct reg_sequence vl53l1x_mode_long[] = {
302*128e5ebeSSiratul Islam { VL53L1X_REG_PHASECAL_CONFIG__TIMEOUT_MACROP, 0x0A },
303*128e5ebeSSiratul Islam { VL53L1X_REG_RANGE_CONFIG__VCSEL_PERIOD_A, 0x0F },
304*128e5ebeSSiratul Islam { VL53L1X_REG_RANGE_CONFIG__VCSEL_PERIOD_B, 0x0D },
305*128e5ebeSSiratul Islam { VL53L1X_REG_RANGE_CONFIG__VALID_PHASE_HIGH, 0xB8 },
306*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__WOI_SD0, 0x0F },
307*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__WOI_SD1, 0x0D },
308*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__INITIAL_PHASE_SD0, 0x0E },
309*128e5ebeSSiratul Islam { VL53L1X_REG_SD_CONFIG__INITIAL_PHASE_SD1, 0x0E },
310*128e5ebeSSiratul Islam };
311*128e5ebeSSiratul Islam
312*128e5ebeSSiratul Islam static const struct {
313*128e5ebeSSiratul Islam const struct reg_sequence *regs;
314*128e5ebeSSiratul Islam size_t num_regs;
315*128e5ebeSSiratul Islam } vl53l1x_mode_configs[] = {
316*128e5ebeSSiratul Islam [VL53L1X_SHORT] = { vl53l1x_mode_short, ARRAY_SIZE(vl53l1x_mode_short) },
317*128e5ebeSSiratul Islam [VL53L1X_LONG] = { vl53l1x_mode_long, ARRAY_SIZE(vl53l1x_mode_long) },
318*128e5ebeSSiratul Islam };
319*128e5ebeSSiratul Islam
vl53l1x_set_distance_mode(struct vl53l1x_data * data,enum vl53l1x_distance_mode mode)320*128e5ebeSSiratul Islam static int vl53l1x_set_distance_mode(struct vl53l1x_data *data,
321*128e5ebeSSiratul Islam enum vl53l1x_distance_mode mode)
322*128e5ebeSSiratul Islam {
323*128e5ebeSSiratul Islam int ret;
324*128e5ebeSSiratul Islam
325*128e5ebeSSiratul Islam if (mode >= ARRAY_SIZE(vl53l1x_mode_configs))
326*128e5ebeSSiratul Islam return -EINVAL;
327*128e5ebeSSiratul Islam
328*128e5ebeSSiratul Islam ret = regmap_multi_reg_write(data->regmap,
329*128e5ebeSSiratul Islam vl53l1x_mode_configs[mode].regs,
330*128e5ebeSSiratul Islam vl53l1x_mode_configs[mode].num_regs);
331*128e5ebeSSiratul Islam if (ret)
332*128e5ebeSSiratul Islam return ret;
333*128e5ebeSSiratul Islam
334*128e5ebeSSiratul Islam data->distance_mode = mode;
335*128e5ebeSSiratul Islam return 0;
336*128e5ebeSSiratul Islam }
337*128e5ebeSSiratul Islam
338*128e5ebeSSiratul Islam /*
339*128e5ebeSSiratul Islam * The timing budget controls how long the sensor spends collecting
340*128e5ebeSSiratul Islam * a single range measurement. Pre-computed TIMEOUT_MACROP register
341*128e5ebeSSiratul Islam * values from ST's VL53L1X Ultra Lite Driver.
342*128e5ebeSSiratul Islam */
vl53l1x_set_timing_budget(struct vl53l1x_data * data,u16 budget_ms)343*128e5ebeSSiratul Islam static int vl53l1x_set_timing_budget(struct vl53l1x_data *data, u16 budget_ms)
344*128e5ebeSSiratul Islam {
345*128e5ebeSSiratul Islam u16 timeout_a, timeout_b;
346*128e5ebeSSiratul Islam int ret;
347*128e5ebeSSiratul Islam
348*128e5ebeSSiratul Islam switch (data->distance_mode) {
349*128e5ebeSSiratul Islam case VL53L1X_SHORT:
350*128e5ebeSSiratul Islam switch (budget_ms) {
351*128e5ebeSSiratul Islam case 15:
352*128e5ebeSSiratul Islam timeout_a = 0x001D;
353*128e5ebeSSiratul Islam timeout_b = 0x0027;
354*128e5ebeSSiratul Islam break;
355*128e5ebeSSiratul Islam case 20:
356*128e5ebeSSiratul Islam timeout_a = 0x0051;
357*128e5ebeSSiratul Islam timeout_b = 0x006E;
358*128e5ebeSSiratul Islam break;
359*128e5ebeSSiratul Islam case 33:
360*128e5ebeSSiratul Islam timeout_a = 0x00D6;
361*128e5ebeSSiratul Islam timeout_b = 0x006E;
362*128e5ebeSSiratul Islam break;
363*128e5ebeSSiratul Islam case 50:
364*128e5ebeSSiratul Islam timeout_a = 0x01AE;
365*128e5ebeSSiratul Islam timeout_b = 0x01E8;
366*128e5ebeSSiratul Islam break;
367*128e5ebeSSiratul Islam case 100:
368*128e5ebeSSiratul Islam timeout_a = 0x02E1;
369*128e5ebeSSiratul Islam timeout_b = 0x0388;
370*128e5ebeSSiratul Islam break;
371*128e5ebeSSiratul Islam case 200:
372*128e5ebeSSiratul Islam timeout_a = 0x03E1;
373*128e5ebeSSiratul Islam timeout_b = 0x0496;
374*128e5ebeSSiratul Islam break;
375*128e5ebeSSiratul Islam case 500:
376*128e5ebeSSiratul Islam timeout_a = 0x0591;
377*128e5ebeSSiratul Islam timeout_b = 0x05C1;
378*128e5ebeSSiratul Islam break;
379*128e5ebeSSiratul Islam default:
380*128e5ebeSSiratul Islam return -EINVAL;
381*128e5ebeSSiratul Islam }
382*128e5ebeSSiratul Islam break;
383*128e5ebeSSiratul Islam case VL53L1X_LONG:
384*128e5ebeSSiratul Islam switch (budget_ms) {
385*128e5ebeSSiratul Islam case 20:
386*128e5ebeSSiratul Islam timeout_a = 0x001E;
387*128e5ebeSSiratul Islam timeout_b = 0x0022;
388*128e5ebeSSiratul Islam break;
389*128e5ebeSSiratul Islam case 33:
390*128e5ebeSSiratul Islam timeout_a = 0x0060;
391*128e5ebeSSiratul Islam timeout_b = 0x006E;
392*128e5ebeSSiratul Islam break;
393*128e5ebeSSiratul Islam case 50:
394*128e5ebeSSiratul Islam timeout_a = 0x00AD;
395*128e5ebeSSiratul Islam timeout_b = 0x00C6;
396*128e5ebeSSiratul Islam break;
397*128e5ebeSSiratul Islam case 100:
398*128e5ebeSSiratul Islam timeout_a = 0x01CC;
399*128e5ebeSSiratul Islam timeout_b = 0x01EA;
400*128e5ebeSSiratul Islam break;
401*128e5ebeSSiratul Islam case 200:
402*128e5ebeSSiratul Islam timeout_a = 0x02D9;
403*128e5ebeSSiratul Islam timeout_b = 0x02F8;
404*128e5ebeSSiratul Islam break;
405*128e5ebeSSiratul Islam case 500:
406*128e5ebeSSiratul Islam timeout_a = 0x048F;
407*128e5ebeSSiratul Islam timeout_b = 0x04A4;
408*128e5ebeSSiratul Islam break;
409*128e5ebeSSiratul Islam default:
410*128e5ebeSSiratul Islam return -EINVAL;
411*128e5ebeSSiratul Islam }
412*128e5ebeSSiratul Islam break;
413*128e5ebeSSiratul Islam default:
414*128e5ebeSSiratul Islam return -EINVAL;
415*128e5ebeSSiratul Islam }
416*128e5ebeSSiratul Islam
417*128e5ebeSSiratul Islam ret = vl53l1x_write_u16(data, VL53L1X_REG_RANGE_CONFIG__TIMEOUT_MACROP_A,
418*128e5ebeSSiratul Islam timeout_a);
419*128e5ebeSSiratul Islam if (ret)
420*128e5ebeSSiratul Islam return ret;
421*128e5ebeSSiratul Islam
422*128e5ebeSSiratul Islam return vl53l1x_write_u16(data, VL53L1X_REG_RANGE_CONFIG__TIMEOUT_MACROP_B,
423*128e5ebeSSiratul Islam timeout_b);
424*128e5ebeSSiratul Islam }
425*128e5ebeSSiratul Islam
vl53l1x_set_inter_measurement_ms(struct vl53l1x_data * data,u16 period_ms)426*128e5ebeSSiratul Islam static int vl53l1x_set_inter_measurement_ms(struct vl53l1x_data *data,
427*128e5ebeSSiratul Islam u16 period_ms)
428*128e5ebeSSiratul Islam {
429*128e5ebeSSiratul Islam u16 osc_calibrate_val;
430*128e5ebeSSiratul Islam u16 clock_pll;
431*128e5ebeSSiratul Islam u32 inter_meas;
432*128e5ebeSSiratul Islam int ret;
433*128e5ebeSSiratul Islam
434*128e5ebeSSiratul Islam ret = vl53l1x_read_u16(data, VL53L1X_REG_RESULT__OSC_CALIBRATE_VAL,
435*128e5ebeSSiratul Islam &osc_calibrate_val);
436*128e5ebeSSiratul Islam if (ret)
437*128e5ebeSSiratul Islam return ret;
438*128e5ebeSSiratul Islam
439*128e5ebeSSiratul Islam clock_pll = osc_calibrate_val & VL53L1X_OSC_CALIBRATE_MASK;
440*128e5ebeSSiratul Islam inter_meas = (clock_pll * period_ms * vl53l1x_osc_correction.numerator) /
441*128e5ebeSSiratul Islam vl53l1x_osc_correction.denominator;
442*128e5ebeSSiratul Islam
443*128e5ebeSSiratul Islam return vl53l1x_write_u32(data,
444*128e5ebeSSiratul Islam VL53L1X_REG_SYSTEM__INTERMEASUREMENT_PERIOD,
445*128e5ebeSSiratul Islam inter_meas);
446*128e5ebeSSiratul Islam }
447*128e5ebeSSiratul Islam
vl53l1x_read_proximity(struct vl53l1x_data * data,int * val)448*128e5ebeSSiratul Islam static int vl53l1x_read_proximity(struct vl53l1x_data *data, int *val)
449*128e5ebeSSiratul Islam {
450*128e5ebeSSiratul Islam unsigned int range_status;
451*128e5ebeSSiratul Islam u16 distance;
452*128e5ebeSSiratul Islam int ret;
453*128e5ebeSSiratul Islam
454*128e5ebeSSiratul Islam if (data->irq) {
455*128e5ebeSSiratul Islam reinit_completion(&data->completion);
456*128e5ebeSSiratul Islam
457*128e5ebeSSiratul Islam ret = vl53l1x_clear_irq(data);
458*128e5ebeSSiratul Islam if (ret)
459*128e5ebeSSiratul Islam return ret;
460*128e5ebeSSiratul Islam
461*128e5ebeSSiratul Islam if (!wait_for_completion_timeout(&data->completion, HZ))
462*128e5ebeSSiratul Islam return -ETIMEDOUT;
463*128e5ebeSSiratul Islam } else {
464*128e5ebeSSiratul Islam unsigned int rdy;
465*128e5ebeSSiratul Islam
466*128e5ebeSSiratul Islam /* 1ms poll, 1s timeout covers max timing budgets (per ST Ultra Lite Driver) */
467*128e5ebeSSiratul Islam ret = regmap_read_poll_timeout(data->regmap,
468*128e5ebeSSiratul Islam VL53L1X_REG_GPIO__TIO_HV_STATUS, rdy,
469*128e5ebeSSiratul Islam (rdy & 1) != data->gpio_polarity,
470*128e5ebeSSiratul Islam 1 * USEC_PER_MSEC,
471*128e5ebeSSiratul Islam 1000 * USEC_PER_MSEC);
472*128e5ebeSSiratul Islam if (ret)
473*128e5ebeSSiratul Islam return ret;
474*128e5ebeSSiratul Islam }
475*128e5ebeSSiratul Islam
476*128e5ebeSSiratul Islam ret = regmap_read(data->regmap, VL53L1X_REG_RESULT__RANGE_STATUS,
477*128e5ebeSSiratul Islam &range_status);
478*128e5ebeSSiratul Islam if (ret)
479*128e5ebeSSiratul Islam goto clear_irq;
480*128e5ebeSSiratul Islam
481*128e5ebeSSiratul Islam if (FIELD_GET(VL53L1X_RANGE_STATUS_MASK, range_status) !=
482*128e5ebeSSiratul Islam VL53L1X_RANGE_STATUS_VALID) {
483*128e5ebeSSiratul Islam ret = -EIO;
484*128e5ebeSSiratul Islam goto clear_irq;
485*128e5ebeSSiratul Islam }
486*128e5ebeSSiratul Islam
487*128e5ebeSSiratul Islam ret = vl53l1x_read_u16(data,
488*128e5ebeSSiratul Islam VL53L1X_REG_RESULT__FINAL_CROSSTALK_CORRECTED_RANGE_MM_SD0,
489*128e5ebeSSiratul Islam &distance);
490*128e5ebeSSiratul Islam if (ret)
491*128e5ebeSSiratul Islam goto clear_irq;
492*128e5ebeSSiratul Islam
493*128e5ebeSSiratul Islam *val = distance;
494*128e5ebeSSiratul Islam
495*128e5ebeSSiratul Islam clear_irq:
496*128e5ebeSSiratul Islam vl53l1x_clear_irq(data);
497*128e5ebeSSiratul Islam return ret;
498*128e5ebeSSiratul Islam }
499*128e5ebeSSiratul Islam
500*128e5ebeSSiratul Islam static const struct iio_chan_spec vl53l1x_channels[] = {
501*128e5ebeSSiratul Islam {
502*128e5ebeSSiratul Islam .type = IIO_DISTANCE,
503*128e5ebeSSiratul Islam .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) |
504*128e5ebeSSiratul Islam BIT(IIO_CHAN_INFO_SCALE),
505*128e5ebeSSiratul Islam .scan_index = 0,
506*128e5ebeSSiratul Islam .scan_type = {
507*128e5ebeSSiratul Islam .sign = 'u',
508*128e5ebeSSiratul Islam .realbits = 16,
509*128e5ebeSSiratul Islam .storagebits = 16,
510*128e5ebeSSiratul Islam },
511*128e5ebeSSiratul Islam },
512*128e5ebeSSiratul Islam IIO_CHAN_SOFT_TIMESTAMP(1),
513*128e5ebeSSiratul Islam };
514*128e5ebeSSiratul Islam
vl53l1x_read_raw(struct iio_dev * indio_dev,const struct iio_chan_spec * chan,int * val,int * val2,long mask)515*128e5ebeSSiratul Islam static int vl53l1x_read_raw(struct iio_dev *indio_dev,
516*128e5ebeSSiratul Islam const struct iio_chan_spec *chan,
517*128e5ebeSSiratul Islam int *val, int *val2, long mask)
518*128e5ebeSSiratul Islam {
519*128e5ebeSSiratul Islam struct vl53l1x_data *data = iio_priv(indio_dev);
520*128e5ebeSSiratul Islam int ret;
521*128e5ebeSSiratul Islam
522*128e5ebeSSiratul Islam if (chan->type != IIO_DISTANCE)
523*128e5ebeSSiratul Islam return -EINVAL;
524*128e5ebeSSiratul Islam
525*128e5ebeSSiratul Islam switch (mask) {
526*128e5ebeSSiratul Islam case IIO_CHAN_INFO_RAW:
527*128e5ebeSSiratul Islam if (!iio_device_claim_direct(indio_dev))
528*128e5ebeSSiratul Islam return -EBUSY;
529*128e5ebeSSiratul Islam ret = vl53l1x_read_proximity(data, val);
530*128e5ebeSSiratul Islam iio_device_release_direct(indio_dev);
531*128e5ebeSSiratul Islam if (ret)
532*128e5ebeSSiratul Islam return ret;
533*128e5ebeSSiratul Islam return IIO_VAL_INT;
534*128e5ebeSSiratul Islam case IIO_CHAN_INFO_SCALE:
535*128e5ebeSSiratul Islam *val = 0;
536*128e5ebeSSiratul Islam *val2 = 1000;
537*128e5ebeSSiratul Islam return IIO_VAL_INT_PLUS_MICRO;
538*128e5ebeSSiratul Islam default:
539*128e5ebeSSiratul Islam return -EINVAL;
540*128e5ebeSSiratul Islam }
541*128e5ebeSSiratul Islam }
542*128e5ebeSSiratul Islam
543*128e5ebeSSiratul Islam static const struct iio_info vl53l1x_info = {
544*128e5ebeSSiratul Islam .read_raw = vl53l1x_read_raw,
545*128e5ebeSSiratul Islam .validate_trigger = iio_validate_own_trigger,
546*128e5ebeSSiratul Islam };
547*128e5ebeSSiratul Islam
vl53l1x_trigger_handler(int irq,void * priv)548*128e5ebeSSiratul Islam static irqreturn_t vl53l1x_trigger_handler(int irq, void *priv)
549*128e5ebeSSiratul Islam {
550*128e5ebeSSiratul Islam struct iio_poll_func *pf = priv;
551*128e5ebeSSiratul Islam struct iio_dev *indio_dev = pf->indio_dev;
552*128e5ebeSSiratul Islam struct vl53l1x_data *data = iio_priv(indio_dev);
553*128e5ebeSSiratul Islam struct {
554*128e5ebeSSiratul Islam u16 distance;
555*128e5ebeSSiratul Islam aligned_s64 timestamp;
556*128e5ebeSSiratul Islam } scan = { };
557*128e5ebeSSiratul Islam unsigned int range_status;
558*128e5ebeSSiratul Islam int ret;
559*128e5ebeSSiratul Islam
560*128e5ebeSSiratul Islam ret = regmap_read(data->regmap, VL53L1X_REG_RESULT__RANGE_STATUS,
561*128e5ebeSSiratul Islam &range_status);
562*128e5ebeSSiratul Islam if (ret)
563*128e5ebeSSiratul Islam goto notify_and_clear_irq;
564*128e5ebeSSiratul Islam if (FIELD_GET(VL53L1X_RANGE_STATUS_MASK, range_status) !=
565*128e5ebeSSiratul Islam VL53L1X_RANGE_STATUS_VALID)
566*128e5ebeSSiratul Islam goto notify_and_clear_irq;
567*128e5ebeSSiratul Islam
568*128e5ebeSSiratul Islam ret = vl53l1x_read_u16(data,
569*128e5ebeSSiratul Islam VL53L1X_REG_RESULT__FINAL_CROSSTALK_CORRECTED_RANGE_MM_SD0,
570*128e5ebeSSiratul Islam &scan.distance);
571*128e5ebeSSiratul Islam if (ret)
572*128e5ebeSSiratul Islam goto notify_and_clear_irq;
573*128e5ebeSSiratul Islam
574*128e5ebeSSiratul Islam iio_push_to_buffers_with_ts(indio_dev, &scan, sizeof(scan),
575*128e5ebeSSiratul Islam iio_get_time_ns(indio_dev));
576*128e5ebeSSiratul Islam
577*128e5ebeSSiratul Islam notify_and_clear_irq:
578*128e5ebeSSiratul Islam iio_trigger_notify_done(indio_dev->trig);
579*128e5ebeSSiratul Islam vl53l1x_clear_irq(data);
580*128e5ebeSSiratul Islam
581*128e5ebeSSiratul Islam return IRQ_HANDLED;
582*128e5ebeSSiratul Islam }
583*128e5ebeSSiratul Islam
vl53l1x_irq_handler(int irq,void * priv)584*128e5ebeSSiratul Islam static irqreturn_t vl53l1x_irq_handler(int irq, void *priv)
585*128e5ebeSSiratul Islam {
586*128e5ebeSSiratul Islam struct iio_dev *indio_dev = priv;
587*128e5ebeSSiratul Islam struct vl53l1x_data *data = iio_priv(indio_dev);
588*128e5ebeSSiratul Islam
589*128e5ebeSSiratul Islam if (iio_buffer_enabled(indio_dev))
590*128e5ebeSSiratul Islam iio_trigger_poll(indio_dev->trig);
591*128e5ebeSSiratul Islam else
592*128e5ebeSSiratul Islam complete(&data->completion);
593*128e5ebeSSiratul Islam
594*128e5ebeSSiratul Islam return IRQ_HANDLED;
595*128e5ebeSSiratul Islam }
596*128e5ebeSSiratul Islam
597*128e5ebeSSiratul Islam static const struct iio_trigger_ops vl53l1x_trigger_ops = {
598*128e5ebeSSiratul Islam .validate_device = iio_trigger_validate_own_device,
599*128e5ebeSSiratul Islam };
600*128e5ebeSSiratul Islam
vl53l1x_stop_ranging_action(void * priv)601*128e5ebeSSiratul Islam static void vl53l1x_stop_ranging_action(void *priv)
602*128e5ebeSSiratul Islam {
603*128e5ebeSSiratul Islam vl53l1x_stop_ranging(priv);
604*128e5ebeSSiratul Islam }
605*128e5ebeSSiratul Islam
vl53l1x_configure_irq(struct device * dev,int irq,struct iio_dev * indio_dev)606*128e5ebeSSiratul Islam static int vl53l1x_configure_irq(struct device *dev, int irq,
607*128e5ebeSSiratul Islam struct iio_dev *indio_dev)
608*128e5ebeSSiratul Islam {
609*128e5ebeSSiratul Islam struct vl53l1x_data *data = iio_priv(indio_dev);
610*128e5ebeSSiratul Islam int ret;
611*128e5ebeSSiratul Islam
612*128e5ebeSSiratul Islam ret = devm_request_irq(dev, irq, vl53l1x_irq_handler, IRQF_NO_THREAD,
613*128e5ebeSSiratul Islam indio_dev->name, indio_dev);
614*128e5ebeSSiratul Islam if (ret)
615*128e5ebeSSiratul Islam return ret;
616*128e5ebeSSiratul Islam
617*128e5ebeSSiratul Islam ret = regmap_write(data->regmap, VL53L1X_REG_SYSTEM__INTERRUPT_CONFIG_GPIO,
618*128e5ebeSSiratul Islam VL53L1X_INT_NEW_SAMPLE_READY);
619*128e5ebeSSiratul Islam if (ret)
620*128e5ebeSSiratul Islam return dev_err_probe(dev, ret, "failed to configure IRQ\n");
621*128e5ebeSSiratul Islam
622*128e5ebeSSiratul Islam return 0;
623*128e5ebeSSiratul Islam }
624*128e5ebeSSiratul Islam
vl53l1x_probe(struct i2c_client * client)625*128e5ebeSSiratul Islam static int vl53l1x_probe(struct i2c_client *client)
626*128e5ebeSSiratul Islam {
627*128e5ebeSSiratul Islam struct device *dev = &client->dev;
628*128e5ebeSSiratul Islam struct vl53l1x_data *data;
629*128e5ebeSSiratul Islam struct iio_dev *indio_dev;
630*128e5ebeSSiratul Islam int ret;
631*128e5ebeSSiratul Islam
632*128e5ebeSSiratul Islam indio_dev = devm_iio_device_alloc(dev, sizeof(*data));
633*128e5ebeSSiratul Islam if (!indio_dev)
634*128e5ebeSSiratul Islam return -ENOMEM;
635*128e5ebeSSiratul Islam
636*128e5ebeSSiratul Islam data = iio_priv(indio_dev);
637*128e5ebeSSiratul Islam data->irq = client->irq;
638*128e5ebeSSiratul Islam
639*128e5ebeSSiratul Islam data->regmap = devm_regmap_init_i2c(client, &vl53l1x_regmap_config);
640*128e5ebeSSiratul Islam if (IS_ERR(data->regmap))
641*128e5ebeSSiratul Islam return dev_err_probe(dev, PTR_ERR(data->regmap),
642*128e5ebeSSiratul Islam "regmap initialization failed\n");
643*128e5ebeSSiratul Islam
644*128e5ebeSSiratul Islam ret = devm_regulator_get_enable(dev, "vdd");
645*128e5ebeSSiratul Islam if (ret)
646*128e5ebeSSiratul Islam return dev_err_probe(dev, ret, "Failed to enable VDD regulator\n");
647*128e5ebeSSiratul Islam
648*128e5ebeSSiratul Islam /*
649*128e5ebeSSiratul Islam * XSHUT held low puts the chip in hardware standby. All register
650*128e5ebeSSiratul Islam * state is lost on de-assert so this is functionally a reset.
651*128e5ebeSSiratul Islam */
652*128e5ebeSSiratul Islam data->xshut_reset = devm_reset_control_get_optional_exclusive_deasserted(dev, NULL);
653*128e5ebeSSiratul Islam if (IS_ERR(data->xshut_reset))
654*128e5ebeSSiratul Islam return dev_err_probe(dev, PTR_ERR(data->xshut_reset),
655*128e5ebeSSiratul Islam "Cannot get reset control\n");
656*128e5ebeSSiratul Islam
657*128e5ebeSSiratul Islam /*
658*128e5ebeSSiratul Islam * 1.2 ms max boot duration.
659*128e5ebeSSiratul Islam * Datasheet Section 3.6 "Power up and boot sequence".
660*128e5ebeSSiratul Islam */
661*128e5ebeSSiratul Islam fsleep(1200);
662*128e5ebeSSiratul Islam
663*128e5ebeSSiratul Islam ret = vl53l1x_chip_init(data);
664*128e5ebeSSiratul Islam if (ret)
665*128e5ebeSSiratul Islam return ret;
666*128e5ebeSSiratul Islam
667*128e5ebeSSiratul Islam ret = vl53l1x_set_distance_mode(data, VL53L1X_LONG);
668*128e5ebeSSiratul Islam if (ret)
669*128e5ebeSSiratul Islam return ret;
670*128e5ebeSSiratul Islam
671*128e5ebeSSiratul Islam /* 50 ms timing budget (per ST Ultra Lite Driver) */
672*128e5ebeSSiratul Islam ret = vl53l1x_set_timing_budget(data, 50);
673*128e5ebeSSiratul Islam if (ret)
674*128e5ebeSSiratul Islam return ret;
675*128e5ebeSSiratul Islam
676*128e5ebeSSiratul Islam /* 50 ms inter-measurement period (per ST Ultra Lite Driver) */
677*128e5ebeSSiratul Islam ret = vl53l1x_set_inter_measurement_ms(data, 50);
678*128e5ebeSSiratul Islam if (ret)
679*128e5ebeSSiratul Islam return ret;
680*128e5ebeSSiratul Islam
681*128e5ebeSSiratul Islam /*
682*128e5ebeSSiratul Islam * The hardware only supports "autonomous" continuous ranging mode.
683*128e5ebeSSiratul Islam * Start ranging here and leave it running for the lifetime of
684*128e5ebeSSiratul Islam * the device. Both direct reads and the buffer path rely on this.
685*128e5ebeSSiratul Islam */
686*128e5ebeSSiratul Islam ret = vl53l1x_start_ranging(data);
687*128e5ebeSSiratul Islam if (ret)
688*128e5ebeSSiratul Islam return ret;
689*128e5ebeSSiratul Islam
690*128e5ebeSSiratul Islam ret = devm_add_action_or_reset(dev, vl53l1x_stop_ranging_action, data);
691*128e5ebeSSiratul Islam if (ret)
692*128e5ebeSSiratul Islam return ret;
693*128e5ebeSSiratul Islam
694*128e5ebeSSiratul Islam indio_dev->name = "vl53l1x";
695*128e5ebeSSiratul Islam indio_dev->info = &vl53l1x_info;
696*128e5ebeSSiratul Islam indio_dev->channels = vl53l1x_channels;
697*128e5ebeSSiratul Islam indio_dev->num_channels = ARRAY_SIZE(vl53l1x_channels);
698*128e5ebeSSiratul Islam indio_dev->modes = INDIO_DIRECT_MODE;
699*128e5ebeSSiratul Islam
700*128e5ebeSSiratul Islam if (client->irq) {
701*128e5ebeSSiratul Islam struct iio_trigger *trig;
702*128e5ebeSSiratul Islam
703*128e5ebeSSiratul Islam init_completion(&data->completion);
704*128e5ebeSSiratul Islam
705*128e5ebeSSiratul Islam trig = devm_iio_trigger_alloc(dev, "%s-dev%d", indio_dev->name,
706*128e5ebeSSiratul Islam iio_device_id(indio_dev));
707*128e5ebeSSiratul Islam if (!trig)
708*128e5ebeSSiratul Islam return -ENOMEM;
709*128e5ebeSSiratul Islam
710*128e5ebeSSiratul Islam trig->ops = &vl53l1x_trigger_ops;
711*128e5ebeSSiratul Islam iio_trigger_set_drvdata(trig, indio_dev);
712*128e5ebeSSiratul Islam ret = devm_iio_trigger_register(dev, trig);
713*128e5ebeSSiratul Islam if (ret)
714*128e5ebeSSiratul Islam return ret;
715*128e5ebeSSiratul Islam
716*128e5ebeSSiratul Islam indio_dev->trig = iio_trigger_get(trig);
717*128e5ebeSSiratul Islam
718*128e5ebeSSiratul Islam ret = vl53l1x_configure_irq(dev, client->irq, indio_dev);
719*128e5ebeSSiratul Islam if (ret)
720*128e5ebeSSiratul Islam return ret;
721*128e5ebeSSiratul Islam
722*128e5ebeSSiratul Islam ret = devm_iio_triggered_buffer_setup(dev, indio_dev, NULL,
723*128e5ebeSSiratul Islam &vl53l1x_trigger_handler,
724*128e5ebeSSiratul Islam NULL);
725*128e5ebeSSiratul Islam if (ret)
726*128e5ebeSSiratul Islam return ret;
727*128e5ebeSSiratul Islam }
728*128e5ebeSSiratul Islam
729*128e5ebeSSiratul Islam return devm_iio_device_register(dev, indio_dev);
730*128e5ebeSSiratul Islam }
731*128e5ebeSSiratul Islam
732*128e5ebeSSiratul Islam static const struct i2c_device_id vl53l1x_id[] = {
733*128e5ebeSSiratul Islam { "vl53l1x" },
734*128e5ebeSSiratul Islam { }
735*128e5ebeSSiratul Islam };
736*128e5ebeSSiratul Islam MODULE_DEVICE_TABLE(i2c, vl53l1x_id);
737*128e5ebeSSiratul Islam
738*128e5ebeSSiratul Islam static const struct of_device_id st_vl53l1x_dt_match[] = {
739*128e5ebeSSiratul Islam { .compatible = "st,vl53l1x" },
740*128e5ebeSSiratul Islam { }
741*128e5ebeSSiratul Islam };
742*128e5ebeSSiratul Islam MODULE_DEVICE_TABLE(of, st_vl53l1x_dt_match);
743*128e5ebeSSiratul Islam
744*128e5ebeSSiratul Islam static struct i2c_driver vl53l1x_driver = {
745*128e5ebeSSiratul Islam .driver = {
746*128e5ebeSSiratul Islam .name = "vl53l1x-i2c",
747*128e5ebeSSiratul Islam .of_match_table = st_vl53l1x_dt_match,
748*128e5ebeSSiratul Islam },
749*128e5ebeSSiratul Islam .probe = vl53l1x_probe,
750*128e5ebeSSiratul Islam .id_table = vl53l1x_id,
751*128e5ebeSSiratul Islam };
752*128e5ebeSSiratul Islam module_i2c_driver(vl53l1x_driver);
753*128e5ebeSSiratul Islam
754*128e5ebeSSiratul Islam MODULE_AUTHOR("Siratul Islam <email@sirat.me>");
755*128e5ebeSSiratul Islam MODULE_DESCRIPTION("ST VL53L1X ToF ranging sensor driver");
756*128e5ebeSSiratul Islam MODULE_LICENSE("Dual BSD/GPL");
757