inv_icm42600_temp.c 1.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * Copyright (C) 2020 Invensense, Inc.
  4. */
  5. #include <linux/kernel.h>
  6. #include <linux/device.h>
  7. #include <linux/mutex.h>
  8. #include <linux/pm_runtime.h>
  9. #include <linux/regmap.h>
  10. #include <linux/iio/iio.h>
  11. #include "inv_icm42600.h"
  12. #include "inv_icm42600_temp.h"
  13. static int inv_icm42600_temp_read(struct inv_icm42600_state *st, int16_t *temp)
  14. {
  15. struct device *dev = regmap_get_device(st->map);
  16. __be16 *raw;
  17. int ret;
  18. pm_runtime_get_sync(dev);
  19. mutex_lock(&st->lock);
  20. ret = inv_icm42600_set_temp_conf(st, true, NULL);
  21. if (ret)
  22. goto exit;
  23. raw = (__be16 *)&st->buffer[0];
  24. ret = regmap_bulk_read(st->map, INV_ICM42600_REG_TEMP_DATA, raw, sizeof(*raw));
  25. if (ret)
  26. goto exit;
  27. *temp = (int16_t)be16_to_cpup(raw);
  28. if (*temp == INV_ICM42600_DATA_INVALID)
  29. ret = -EINVAL;
  30. exit:
  31. mutex_unlock(&st->lock);
  32. pm_runtime_mark_last_busy(dev);
  33. pm_runtime_put_autosuspend(dev);
  34. return ret;
  35. }
  36. int inv_icm42600_temp_read_raw(struct iio_dev *indio_dev,
  37. struct iio_chan_spec const *chan,
  38. int *val, int *val2, long mask)
  39. {
  40. struct inv_icm42600_state *st = iio_device_get_drvdata(indio_dev);
  41. int16_t temp;
  42. int ret;
  43. if (chan->type != IIO_TEMP)
  44. return -EINVAL;
  45. switch (mask) {
  46. case IIO_CHAN_INFO_RAW:
  47. ret = iio_device_claim_direct_mode(indio_dev);
  48. if (ret)
  49. return ret;
  50. ret = inv_icm42600_temp_read(st, &temp);
  51. iio_device_release_direct_mode(indio_dev);
  52. if (ret)
  53. return ret;
  54. *val = temp;
  55. return IIO_VAL_INT;
  56. /*
  57. * T°C = (temp / 132.48) + 25
  58. * Tm°C = 1000 * ((temp * 100 / 13248) + 25)
  59. * scale: 100000 / 13248 ~= 7.548309
  60. * offset: 25000
  61. */
  62. case IIO_CHAN_INFO_SCALE:
  63. *val = 7;
  64. *val2 = 548309;
  65. return IIO_VAL_INT_PLUS_MICRO;
  66. case IIO_CHAN_INFO_OFFSET:
  67. *val = 25000;
  68. return IIO_VAL_INT;
  69. default:
  70. return -EINVAL;
  71. }
  72. }