leds-nic78bx.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. /*
  2. * Copyright (C) 2016 National Instruments Corp.
  3. *
  4. * This program is free software; you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation; either version 2 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. */
  14. #include <linux/acpi.h>
  15. #include <linux/leds.h>
  16. #include <linux/module.h>
  17. #include <linux/platform_device.h>
  18. #include <linux/spinlock.h>
  19. #define NIC78BX_USER1_LED_MASK 0x3
  20. #define NIC78BX_USER1_GREEN_LED BIT(0)
  21. #define NIC78BX_USER1_YELLOW_LED BIT(1)
  22. #define NIC78BX_USER2_LED_MASK 0xC
  23. #define NIC78BX_USER2_GREEN_LED BIT(2)
  24. #define NIC78BX_USER2_YELLOW_LED BIT(3)
  25. #define NIC78BX_LOCK_REG_OFFSET 1
  26. #define NIC78BX_LOCK_VALUE 0xA5
  27. #define NIC78BX_UNLOCK_VALUE 0x5A
  28. #define NIC78BX_USER_LED_IO_SIZE 2
  29. struct nic78bx_led_data {
  30. u16 io_base;
  31. spinlock_t lock;
  32. struct platform_device *pdev;
  33. };
  34. struct nic78bx_led {
  35. u8 bit;
  36. u8 mask;
  37. struct nic78bx_led_data *data;
  38. struct led_classdev cdev;
  39. };
  40. static inline struct nic78bx_led *to_nic78bx_led(struct led_classdev *cdev)
  41. {
  42. return container_of(cdev, struct nic78bx_led, cdev);
  43. }
  44. static void nic78bx_brightness_set(struct led_classdev *cdev,
  45. enum led_brightness brightness)
  46. {
  47. struct nic78bx_led *nled = to_nic78bx_led(cdev);
  48. unsigned long flags;
  49. u8 value;
  50. spin_lock_irqsave(&nled->data->lock, flags);
  51. value = inb(nled->data->io_base);
  52. if (brightness) {
  53. value &= ~nled->mask;
  54. value |= nled->bit;
  55. } else {
  56. value &= ~nled->bit;
  57. }
  58. outb(value, nled->data->io_base);
  59. spin_unlock_irqrestore(&nled->data->lock, flags);
  60. }
  61. static enum led_brightness nic78bx_brightness_get(struct led_classdev *cdev)
  62. {
  63. struct nic78bx_led *nled = to_nic78bx_led(cdev);
  64. unsigned long flags;
  65. u8 value;
  66. spin_lock_irqsave(&nled->data->lock, flags);
  67. value = inb(nled->data->io_base);
  68. spin_unlock_irqrestore(&nled->data->lock, flags);
  69. return (value & nled->bit) ? 1 : LED_OFF;
  70. }
  71. static struct nic78bx_led nic78bx_leds[] = {
  72. {
  73. .bit = NIC78BX_USER1_GREEN_LED,
  74. .mask = NIC78BX_USER1_LED_MASK,
  75. .cdev = {
  76. .name = "nilrt:green:user1",
  77. .max_brightness = 1,
  78. .brightness_set = nic78bx_brightness_set,
  79. .brightness_get = nic78bx_brightness_get,
  80. }
  81. },
  82. {
  83. .bit = NIC78BX_USER1_YELLOW_LED,
  84. .mask = NIC78BX_USER1_LED_MASK,
  85. .cdev = {
  86. .name = "nilrt:yellow:user1",
  87. .max_brightness = 1,
  88. .brightness_set = nic78bx_brightness_set,
  89. .brightness_get = nic78bx_brightness_get,
  90. }
  91. },
  92. {
  93. .bit = NIC78BX_USER2_GREEN_LED,
  94. .mask = NIC78BX_USER2_LED_MASK,
  95. .cdev = {
  96. .name = "nilrt:green:user2",
  97. .max_brightness = 1,
  98. .brightness_set = nic78bx_brightness_set,
  99. .brightness_get = nic78bx_brightness_get,
  100. }
  101. },
  102. {
  103. .bit = NIC78BX_USER2_YELLOW_LED,
  104. .mask = NIC78BX_USER2_LED_MASK,
  105. .cdev = {
  106. .name = "nilrt:yellow:user2",
  107. .max_brightness = 1,
  108. .brightness_set = nic78bx_brightness_set,
  109. .brightness_get = nic78bx_brightness_get,
  110. }
  111. }
  112. };
  113. static int nic78bx_probe(struct platform_device *pdev)
  114. {
  115. struct device *dev = &pdev->dev;
  116. struct nic78bx_led_data *led_data;
  117. struct resource *io_rc;
  118. int ret, i;
  119. led_data = devm_kzalloc(dev, sizeof(*led_data), GFP_KERNEL);
  120. if (!led_data)
  121. return -ENOMEM;
  122. led_data->pdev = pdev;
  123. platform_set_drvdata(pdev, led_data);
  124. io_rc = platform_get_resource(pdev, IORESOURCE_IO, 0);
  125. if (!io_rc) {
  126. dev_err(dev, "missing IO resources\n");
  127. return -EINVAL;
  128. }
  129. if (resource_size(io_rc) < NIC78BX_USER_LED_IO_SIZE) {
  130. dev_err(dev, "IO region too small\n");
  131. return -EINVAL;
  132. }
  133. if (!devm_request_region(dev, io_rc->start, resource_size(io_rc),
  134. KBUILD_MODNAME)) {
  135. dev_err(dev, "failed to get IO region\n");
  136. return -EBUSY;
  137. }
  138. led_data->io_base = io_rc->start;
  139. spin_lock_init(&led_data->lock);
  140. for (i = 0; i < ARRAY_SIZE(nic78bx_leds); i++) {
  141. nic78bx_leds[i].data = led_data;
  142. ret = devm_led_classdev_register(dev, &nic78bx_leds[i].cdev);
  143. if (ret)
  144. return ret;
  145. }
  146. /* Unlock LED register */
  147. outb(NIC78BX_UNLOCK_VALUE,
  148. led_data->io_base + NIC78BX_LOCK_REG_OFFSET);
  149. return ret;
  150. }
  151. static int nic78bx_remove(struct platform_device *pdev)
  152. {
  153. struct nic78bx_led_data *led_data = platform_get_drvdata(pdev);
  154. /* Lock LED register */
  155. outb(NIC78BX_LOCK_VALUE,
  156. led_data->io_base + NIC78BX_LOCK_REG_OFFSET);
  157. return 0;
  158. }
  159. static const struct acpi_device_id led_device_ids[] = {
  160. {"NIC78B3", 0},
  161. {"", 0},
  162. };
  163. MODULE_DEVICE_TABLE(acpi, led_device_ids);
  164. static struct platform_driver led_driver = {
  165. .probe = nic78bx_probe,
  166. .remove = nic78bx_remove,
  167. .driver = {
  168. .name = KBUILD_MODNAME,
  169. .acpi_match_table = ACPI_PTR(led_device_ids),
  170. },
  171. };
  172. module_platform_driver(led_driver);
  173. MODULE_DESCRIPTION("National Instruments PXI User LEDs driver");
  174. MODULE_AUTHOR("Hui Chun Ong <hui.chun.ong@ni.com>");
  175. MODULE_LICENSE("GPL");