ams-iaq-core.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * ams-iaq-core.c - Support for AMS iAQ-Core VOC sensors
  4. *
  5. * Copyright (C) 2015, 2018
  6. * Author: Matt Ranostay <matt.ranostay@konsulko.com>
  7. */
  8. #include <linux/module.h>
  9. #include <linux/mutex.h>
  10. #include <linux/init.h>
  11. #include <linux/i2c.h>
  12. #include <linux/iio/iio.h>
  13. #define AMS_IAQCORE_DATA_SIZE 9
  14. #define AMS_IAQCORE_VOC_CO2_IDX 0
  15. #define AMS_IAQCORE_VOC_RESISTANCE_IDX 1
  16. #define AMS_IAQCORE_VOC_TVOC_IDX 2
  17. struct ams_iaqcore_reading {
  18. __be16 co2_ppm;
  19. u8 status;
  20. __be32 resistance;
  21. __be16 voc_ppb;
  22. } __attribute__((__packed__));
  23. struct ams_iaqcore_data {
  24. struct i2c_client *client;
  25. struct mutex lock;
  26. unsigned long last_update;
  27. struct ams_iaqcore_reading buffer;
  28. };
  29. static const struct iio_chan_spec ams_iaqcore_channels[] = {
  30. {
  31. .type = IIO_CONCENTRATION,
  32. .channel2 = IIO_MOD_CO2,
  33. .modified = 1,
  34. .info_mask_separate = BIT(IIO_CHAN_INFO_PROCESSED),
  35. .address = AMS_IAQCORE_VOC_CO2_IDX,
  36. },
  37. {
  38. .type = IIO_RESISTANCE,
  39. .info_mask_separate = BIT(IIO_CHAN_INFO_PROCESSED),
  40. .address = AMS_IAQCORE_VOC_RESISTANCE_IDX,
  41. },
  42. {
  43. .type = IIO_CONCENTRATION,
  44. .channel2 = IIO_MOD_VOC,
  45. .modified = 1,
  46. .info_mask_separate = BIT(IIO_CHAN_INFO_PROCESSED),
  47. .address = AMS_IAQCORE_VOC_TVOC_IDX,
  48. },
  49. };
  50. static int ams_iaqcore_read_measurement(struct ams_iaqcore_data *data)
  51. {
  52. struct i2c_client *client = data->client;
  53. int ret;
  54. struct i2c_msg msg = {
  55. .addr = client->addr,
  56. .flags = client->flags | I2C_M_RD,
  57. .len = AMS_IAQCORE_DATA_SIZE,
  58. .buf = (char *) &data->buffer,
  59. };
  60. ret = i2c_transfer(client->adapter, &msg, 1);
  61. return (ret == AMS_IAQCORE_DATA_SIZE) ? 0 : ret;
  62. }
  63. static int ams_iaqcore_get_measurement(struct ams_iaqcore_data *data)
  64. {
  65. int ret;
  66. /* sensor can only be polled once a second max per datasheet */
  67. if (!time_after(jiffies, data->last_update + HZ))
  68. return 0;
  69. ret = ams_iaqcore_read_measurement(data);
  70. if (ret < 0)
  71. return ret;
  72. data->last_update = jiffies;
  73. return 0;
  74. }
  75. static int ams_iaqcore_read_raw(struct iio_dev *indio_dev,
  76. struct iio_chan_spec const *chan, int *val,
  77. int *val2, long mask)
  78. {
  79. struct ams_iaqcore_data *data = iio_priv(indio_dev);
  80. int ret;
  81. if (mask != IIO_CHAN_INFO_PROCESSED)
  82. return -EINVAL;
  83. mutex_lock(&data->lock);
  84. ret = ams_iaqcore_get_measurement(data);
  85. if (ret)
  86. goto err_out;
  87. switch (chan->address) {
  88. case AMS_IAQCORE_VOC_CO2_IDX:
  89. *val = 0;
  90. *val2 = be16_to_cpu(data->buffer.co2_ppm);
  91. ret = IIO_VAL_INT_PLUS_MICRO;
  92. break;
  93. case AMS_IAQCORE_VOC_RESISTANCE_IDX:
  94. *val = be32_to_cpu(data->buffer.resistance);
  95. ret = IIO_VAL_INT;
  96. break;
  97. case AMS_IAQCORE_VOC_TVOC_IDX:
  98. *val = 0;
  99. *val2 = be16_to_cpu(data->buffer.voc_ppb);
  100. ret = IIO_VAL_INT_PLUS_NANO;
  101. break;
  102. default:
  103. ret = -EINVAL;
  104. }
  105. err_out:
  106. mutex_unlock(&data->lock);
  107. return ret;
  108. }
  109. static const struct iio_info ams_iaqcore_info = {
  110. .read_raw = ams_iaqcore_read_raw,
  111. };
  112. static int ams_iaqcore_probe(struct i2c_client *client,
  113. const struct i2c_device_id *id)
  114. {
  115. struct iio_dev *indio_dev;
  116. struct ams_iaqcore_data *data;
  117. indio_dev = devm_iio_device_alloc(&client->dev, sizeof(*data));
  118. if (!indio_dev)
  119. return -ENOMEM;
  120. data = iio_priv(indio_dev);
  121. i2c_set_clientdata(client, indio_dev);
  122. data->client = client;
  123. /* so initial reading will complete */
  124. data->last_update = jiffies - HZ;
  125. mutex_init(&data->lock);
  126. indio_dev->dev.parent = &client->dev;
  127. indio_dev->info = &ams_iaqcore_info;
  128. indio_dev->name = dev_name(&client->dev);
  129. indio_dev->modes = INDIO_DIRECT_MODE;
  130. indio_dev->channels = ams_iaqcore_channels;
  131. indio_dev->num_channels = ARRAY_SIZE(ams_iaqcore_channels);
  132. return devm_iio_device_register(&client->dev, indio_dev);
  133. }
  134. static const struct i2c_device_id ams_iaqcore_id[] = {
  135. { "ams-iaq-core", 0 },
  136. { }
  137. };
  138. MODULE_DEVICE_TABLE(i2c, ams_iaqcore_id);
  139. static const struct of_device_id ams_iaqcore_dt_ids[] = {
  140. { .compatible = "ams,iaq-core" },
  141. { }
  142. };
  143. MODULE_DEVICE_TABLE(of, ams_iaqcore_dt_ids);
  144. static struct i2c_driver ams_iaqcore_driver = {
  145. .driver = {
  146. .name = "ams-iaq-core",
  147. .of_match_table = of_match_ptr(ams_iaqcore_dt_ids),
  148. },
  149. .probe = ams_iaqcore_probe,
  150. .id_table = ams_iaqcore_id,
  151. };
  152. module_i2c_driver(ams_iaqcore_driver);
  153. MODULE_AUTHOR("Matt Ranostay <matt.ranostay@konsulko.com>");
  154. MODULE_DESCRIPTION("AMS iAQ-Core VOC sensors");
  155. MODULE_LICENSE("GPL v2");