kbatt.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (C) 2025 KEBA Industrial Automation GmbH
  4. *
  5. * Driver for KEBA battery monitoring controller FPGA IP core
  6. */
  7. #include <linux/hwmon.h>
  8. #include <linux/io.h>
  9. #include <linux/delay.h>
  10. #include <linux/module.h>
  11. #include <linux/device.h>
  12. #include <linux/auxiliary_bus.h>
  13. #include <linux/misc/keba.h>
  14. #include <linux/mutex.h>
  15. #define KBATT "kbatt"
  16. #define KBATT_CONTROL_REG 0x4
  17. #define KBATT_CONTROL_BAT_TEST 0x01
  18. #define KBATT_STATUS_REG 0x8
  19. #define KBATT_STATUS_BAT_OK 0x01
  20. #define KBATT_MAX_UPD_INTERVAL (10 * HZ)
  21. #define KBATT_SETTLE_TIME_US (100 * USEC_PER_MSEC)
  22. struct kbatt {
  23. /* update lock */
  24. struct mutex lock;
  25. void __iomem *base;
  26. unsigned long next_update; /* in jiffies */
  27. bool alarm;
  28. };
  29. static bool kbatt_alarm(struct kbatt *kbatt)
  30. {
  31. mutex_lock(&kbatt->lock);
  32. if (!kbatt->next_update || time_after(jiffies, kbatt->next_update)) {
  33. /* switch load on */
  34. iowrite8(KBATT_CONTROL_BAT_TEST,
  35. kbatt->base + KBATT_CONTROL_REG);
  36. /* wait some time to let things settle */
  37. fsleep(KBATT_SETTLE_TIME_US);
  38. /* check battery state */
  39. if (ioread8(kbatt->base + KBATT_STATUS_REG) &
  40. KBATT_STATUS_BAT_OK)
  41. kbatt->alarm = false;
  42. else
  43. kbatt->alarm = true;
  44. /* switch load off */
  45. iowrite8(0, kbatt->base + KBATT_CONTROL_REG);
  46. kbatt->next_update = jiffies + KBATT_MAX_UPD_INTERVAL;
  47. }
  48. mutex_unlock(&kbatt->lock);
  49. return kbatt->alarm;
  50. }
  51. static int kbatt_read(struct device *dev, enum hwmon_sensor_types type,
  52. u32 attr, int channel, long *val)
  53. {
  54. struct kbatt *kbatt = dev_get_drvdata(dev);
  55. *val = kbatt_alarm(kbatt) ? 1 : 0;
  56. return 0;
  57. }
  58. static umode_t kbatt_is_visible(const void *data, enum hwmon_sensor_types type,
  59. u32 attr, int channel)
  60. {
  61. if (channel == 0 && attr == hwmon_in_min_alarm)
  62. return 0444;
  63. return 0;
  64. }
  65. static const struct hwmon_channel_info *kbatt_info[] = {
  66. HWMON_CHANNEL_INFO(in,
  67. /* 0: input minimum alarm channel */
  68. HWMON_I_MIN_ALARM),
  69. NULL
  70. };
  71. static const struct hwmon_ops kbatt_hwmon_ops = {
  72. .is_visible = kbatt_is_visible,
  73. .read = kbatt_read,
  74. };
  75. static const struct hwmon_chip_info kbatt_chip_info = {
  76. .ops = &kbatt_hwmon_ops,
  77. .info = kbatt_info,
  78. };
  79. static int kbatt_probe(struct auxiliary_device *auxdev,
  80. const struct auxiliary_device_id *id)
  81. {
  82. struct keba_batt_auxdev *kbatt_auxdev =
  83. container_of(auxdev, struct keba_batt_auxdev, auxdev);
  84. struct device *dev = &auxdev->dev;
  85. struct device *hwmon_dev;
  86. struct kbatt *kbatt;
  87. int retval;
  88. kbatt = devm_kzalloc(dev, sizeof(*kbatt), GFP_KERNEL);
  89. if (!kbatt)
  90. return -ENOMEM;
  91. retval = devm_mutex_init(dev, &kbatt->lock);
  92. if (retval)
  93. return retval;
  94. kbatt->base = devm_ioremap_resource(dev, &kbatt_auxdev->io);
  95. if (IS_ERR(kbatt->base))
  96. return PTR_ERR(kbatt->base);
  97. hwmon_dev = devm_hwmon_device_register_with_info(dev, KBATT, kbatt,
  98. &kbatt_chip_info,
  99. NULL);
  100. return PTR_ERR_OR_ZERO(hwmon_dev);
  101. }
  102. static const struct auxiliary_device_id kbatt_devtype_aux[] = {
  103. { .name = "keba.batt" },
  104. {}
  105. };
  106. MODULE_DEVICE_TABLE(auxiliary, kbatt_devtype_aux);
  107. static struct auxiliary_driver kbatt_driver_aux = {
  108. .name = KBATT,
  109. .id_table = kbatt_devtype_aux,
  110. .probe = kbatt_probe,
  111. };
  112. module_auxiliary_driver(kbatt_driver_aux);
  113. MODULE_AUTHOR("Petar Bojanic <boja@keba.com>");
  114. MODULE_AUTHOR("Gerhard Engleder <eg@keba.com>");
  115. MODULE_DESCRIPTION("KEBA battery monitoring controller driver");
  116. MODULE_LICENSE("GPL");