vl53l0x-i2c.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Support for ST VL53L0X FlightSense ToF Ranging Sensor on a i2c bus.
  4. *
  5. * Copyright (C) 2016 STMicroelectronics Imaging Division.
  6. * Copyright (C) 2018 Song Qiang <songqiang1304521@gmail.com>
  7. *
  8. * Datasheet available at
  9. * <https://www.st.com/resource/en/datasheet/vl53l0x.pdf>
  10. *
  11. * Default 7-bit i2c slave address 0x29.
  12. *
  13. * TODO: FIFO buffer, continuous mode, interrupts, range selection,
  14. * sensor ID check.
  15. */
  16. #include <linux/delay.h>
  17. #include <linux/i2c.h>
  18. #include <linux/module.h>
  19. #include <linux/iio/iio.h>
  20. #define VL_REG_SYSRANGE_START 0x00
  21. #define VL_REG_SYSRANGE_MODE_MASK GENMASK(3, 0)
  22. #define VL_REG_SYSRANGE_MODE_SINGLESHOT 0x00
  23. #define VL_REG_SYSRANGE_MODE_START_STOP BIT(0)
  24. #define VL_REG_SYSRANGE_MODE_BACKTOBACK BIT(1)
  25. #define VL_REG_SYSRANGE_MODE_TIMED BIT(2)
  26. #define VL_REG_SYSRANGE_MODE_HISTOGRAM BIT(3)
  27. #define VL_REG_RESULT_INT_STATUS 0x13
  28. #define VL_REG_RESULT_RANGE_STATUS 0x14
  29. #define VL_REG_RESULT_RANGE_STATUS_COMPLETE BIT(0)
  30. struct vl53l0x_data {
  31. struct i2c_client *client;
  32. };
  33. static int vl53l0x_read_proximity(struct vl53l0x_data *data,
  34. const struct iio_chan_spec *chan,
  35. int *val)
  36. {
  37. struct i2c_client *client = data->client;
  38. u16 tries = 20;
  39. u8 buffer[12];
  40. int ret;
  41. ret = i2c_smbus_write_byte_data(client, VL_REG_SYSRANGE_START, 1);
  42. if (ret < 0)
  43. return ret;
  44. do {
  45. ret = i2c_smbus_read_byte_data(client,
  46. VL_REG_RESULT_RANGE_STATUS);
  47. if (ret < 0)
  48. return ret;
  49. if (ret & VL_REG_RESULT_RANGE_STATUS_COMPLETE)
  50. break;
  51. usleep_range(1000, 5000);
  52. } while (--tries);
  53. if (!tries)
  54. return -ETIMEDOUT;
  55. ret = i2c_smbus_read_i2c_block_data(client, VL_REG_RESULT_RANGE_STATUS,
  56. 12, buffer);
  57. if (ret < 0)
  58. return ret;
  59. else if (ret != 12)
  60. return -EREMOTEIO;
  61. /* Values should be between 30~1200 in millimeters. */
  62. *val = (buffer[10] << 8) + buffer[11];
  63. return 0;
  64. }
  65. static const struct iio_chan_spec vl53l0x_channels[] = {
  66. {
  67. .type = IIO_DISTANCE,
  68. .info_mask_separate = BIT(IIO_CHAN_INFO_RAW) |
  69. BIT(IIO_CHAN_INFO_SCALE),
  70. },
  71. };
  72. static int vl53l0x_read_raw(struct iio_dev *indio_dev,
  73. const struct iio_chan_spec *chan,
  74. int *val, int *val2, long mask)
  75. {
  76. struct vl53l0x_data *data = iio_priv(indio_dev);
  77. int ret;
  78. if (chan->type != IIO_DISTANCE)
  79. return -EINVAL;
  80. switch (mask) {
  81. case IIO_CHAN_INFO_RAW:
  82. ret = vl53l0x_read_proximity(data, chan, val);
  83. if (ret < 0)
  84. return ret;
  85. return IIO_VAL_INT;
  86. case IIO_CHAN_INFO_SCALE:
  87. *val = 0;
  88. *val2 = 1000;
  89. return IIO_VAL_INT_PLUS_MICRO;
  90. default:
  91. return -EINVAL;
  92. }
  93. }
  94. static const struct iio_info vl53l0x_info = {
  95. .read_raw = vl53l0x_read_raw,
  96. };
  97. static int vl53l0x_probe(struct i2c_client *client)
  98. {
  99. struct vl53l0x_data *data;
  100. struct iio_dev *indio_dev;
  101. indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data));
  102. if (!indio_dev)
  103. return -ENOMEM;
  104. data = iio_priv(indio_dev);
  105. data->client = client;
  106. i2c_set_clientdata(client, indio_dev);
  107. if (!i2c_check_functionality(client->adapter,
  108. I2C_FUNC_SMBUS_READ_I2C_BLOCK |
  109. I2C_FUNC_SMBUS_BYTE_DATA))
  110. return -EOPNOTSUPP;
  111. indio_dev->dev.parent = &client->dev;
  112. indio_dev->name = "vl53l0x";
  113. indio_dev->info = &vl53l0x_info;
  114. indio_dev->channels = vl53l0x_channels;
  115. indio_dev->num_channels = ARRAY_SIZE(vl53l0x_channels);
  116. indio_dev->modes = INDIO_DIRECT_MODE;
  117. return devm_iio_device_register(&client->dev, indio_dev);
  118. }
  119. static const struct of_device_id st_vl53l0x_dt_match[] = {
  120. { .compatible = "st,vl53l0x", },
  121. { }
  122. };
  123. MODULE_DEVICE_TABLE(of, st_vl53l0x_dt_match);
  124. static struct i2c_driver vl53l0x_driver = {
  125. .driver = {
  126. .name = "vl53l0x-i2c",
  127. .of_match_table = st_vl53l0x_dt_match,
  128. },
  129. .probe_new = vl53l0x_probe,
  130. };
  131. module_i2c_driver(vl53l0x_driver);
  132. MODULE_AUTHOR("Song Qiang <songqiang1304521@gmail.com>");
  133. MODULE_DESCRIPTION("ST vl53l0x ToF ranging sensor driver");
  134. MODULE_LICENSE("GPL v2");