leds-ktd2692.c 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * LED driver : leds-ktd2692.c
  4. *
  5. * Copyright (C) 2015 Samsung Electronics
  6. * Ingi Kim <ingi2.kim@samsung.com>
  7. */
  8. #include <linux/cleanup.h>
  9. #include <linux/err.h>
  10. #include <linux/gpio/consumer.h>
  11. #include <linux/leds-expresswire.h>
  12. #include <linux/led-class-flash.h>
  13. #include <linux/module.h>
  14. #include <linux/mutex.h>
  15. #include <linux/of.h>
  16. #include <linux/platform_device.h>
  17. #include <linux/regulator/consumer.h>
  18. /* Value related the movie mode */
  19. #define KTD2692_MOVIE_MODE_CURRENT_LEVELS 16
  20. #define KTD2692_MM_TO_FL_RATIO(x) ((x) / 3)
  21. #define KTD2692_MM_MIN_CURR_THRESHOLD_SCALE 8
  22. /* Value related the flash mode */
  23. #define KTD2692_FLASH_MODE_TIMEOUT_LEVELS 8
  24. #define KTD2692_FLASH_MODE_TIMEOUT_DISABLE 0
  25. #define KTD2692_FLASH_MODE_CURR_PERCENT(x) (((x) * 16) / 100)
  26. /* Macro for getting offset of flash timeout */
  27. #define GET_TIMEOUT_OFFSET(timeout, step) ((timeout) / (step))
  28. /* Base register address */
  29. #define KTD2692_REG_LVP_BASE 0x00
  30. #define KTD2692_REG_FLASH_TIMEOUT_BASE 0x20
  31. #define KTD2692_REG_MM_MIN_CURR_THRESHOLD_BASE 0x40
  32. #define KTD2692_REG_MOVIE_CURRENT_BASE 0x60
  33. #define KTD2692_REG_FLASH_CURRENT_BASE 0x80
  34. #define KTD2692_REG_MODE_BASE 0xA0
  35. /* KTD2692 default length of name */
  36. #define KTD2692_NAME_LENGTH 20
  37. /* Movie / Flash Mode Control */
  38. enum ktd2692_led_mode {
  39. KTD2692_MODE_DISABLE = 0, /* default */
  40. KTD2692_MODE_MOVIE,
  41. KTD2692_MODE_FLASH,
  42. };
  43. struct ktd2692_led_config_data {
  44. /* maximum LED current in movie mode */
  45. u32 movie_max_microamp;
  46. /* maximum LED current in flash mode */
  47. u32 flash_max_microamp;
  48. /* maximum flash timeout */
  49. u32 flash_max_timeout;
  50. /* max LED brightness level */
  51. enum led_brightness max_brightness;
  52. };
  53. const struct expresswire_timing ktd2692_timing = {
  54. .poweroff_us = 700,
  55. .data_start_us = 10,
  56. .end_of_data_low_us = 10,
  57. .end_of_data_high_us = 350,
  58. .short_bitset_us = 4,
  59. .long_bitset_us = 12
  60. };
  61. struct ktd2692_context {
  62. /* Common ExpressWire properties (ctrl GPIO and timing) */
  63. struct expresswire_common_props props;
  64. /* Related LED Flash class device */
  65. struct led_classdev_flash fled_cdev;
  66. /* secures access to the device */
  67. struct mutex lock;
  68. struct regulator *regulator;
  69. struct gpio_desc *aux_gpio;
  70. enum ktd2692_led_mode mode;
  71. enum led_brightness torch_brightness;
  72. };
  73. static struct ktd2692_context *fled_cdev_to_led(
  74. struct led_classdev_flash *fled_cdev)
  75. {
  76. return container_of(fled_cdev, struct ktd2692_context, fled_cdev);
  77. }
  78. static int ktd2692_led_brightness_set(struct led_classdev *led_cdev,
  79. enum led_brightness brightness)
  80. {
  81. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  82. struct ktd2692_context *led = fled_cdev_to_led(fled_cdev);
  83. mutex_lock(&led->lock);
  84. if (brightness == LED_OFF) {
  85. led->mode = KTD2692_MODE_DISABLE;
  86. gpiod_direction_output(led->aux_gpio, 0);
  87. } else {
  88. expresswire_write_u8(&led->props, brightness |
  89. KTD2692_REG_MOVIE_CURRENT_BASE);
  90. led->mode = KTD2692_MODE_MOVIE;
  91. }
  92. expresswire_write_u8(&led->props, led->mode | KTD2692_REG_MODE_BASE);
  93. mutex_unlock(&led->lock);
  94. return 0;
  95. }
  96. static int ktd2692_led_flash_strobe_set(struct led_classdev_flash *fled_cdev,
  97. bool state)
  98. {
  99. struct ktd2692_context *led = fled_cdev_to_led(fled_cdev);
  100. struct led_flash_setting *timeout = &fled_cdev->timeout;
  101. u32 flash_tm_reg;
  102. mutex_lock(&led->lock);
  103. if (state) {
  104. flash_tm_reg = GET_TIMEOUT_OFFSET(timeout->val, timeout->step);
  105. expresswire_write_u8(&led->props, flash_tm_reg
  106. | KTD2692_REG_FLASH_TIMEOUT_BASE);
  107. led->mode = KTD2692_MODE_FLASH;
  108. gpiod_direction_output(led->aux_gpio, 1);
  109. } else {
  110. led->mode = KTD2692_MODE_DISABLE;
  111. gpiod_direction_output(led->aux_gpio, 0);
  112. }
  113. expresswire_write_u8(&led->props, led->mode | KTD2692_REG_MODE_BASE);
  114. fled_cdev->led_cdev.brightness = LED_OFF;
  115. led->mode = KTD2692_MODE_DISABLE;
  116. mutex_unlock(&led->lock);
  117. return 0;
  118. }
  119. static int ktd2692_led_flash_timeout_set(struct led_classdev_flash *fled_cdev,
  120. u32 timeout)
  121. {
  122. return 0;
  123. }
  124. static void ktd2692_init_movie_current_max(struct ktd2692_led_config_data *cfg)
  125. {
  126. u32 offset, step;
  127. u32 movie_current_microamp;
  128. offset = KTD2692_MOVIE_MODE_CURRENT_LEVELS;
  129. step = KTD2692_MM_TO_FL_RATIO(cfg->flash_max_microamp)
  130. / KTD2692_MOVIE_MODE_CURRENT_LEVELS;
  131. do {
  132. movie_current_microamp = step * offset;
  133. offset--;
  134. } while ((movie_current_microamp > cfg->movie_max_microamp) &&
  135. (offset > 0));
  136. cfg->max_brightness = offset;
  137. }
  138. static void ktd2692_init_flash_timeout(struct led_classdev_flash *fled_cdev,
  139. struct ktd2692_led_config_data *cfg)
  140. {
  141. struct led_flash_setting *setting;
  142. setting = &fled_cdev->timeout;
  143. setting->min = KTD2692_FLASH_MODE_TIMEOUT_DISABLE;
  144. setting->max = cfg->flash_max_timeout;
  145. setting->step = cfg->flash_max_timeout
  146. / (KTD2692_FLASH_MODE_TIMEOUT_LEVELS - 1);
  147. setting->val = cfg->flash_max_timeout;
  148. }
  149. static void ktd2692_setup(struct ktd2692_context *led)
  150. {
  151. led->mode = KTD2692_MODE_DISABLE;
  152. expresswire_power_off(&led->props);
  153. gpiod_direction_output(led->aux_gpio, 0);
  154. expresswire_write_u8(&led->props, (KTD2692_MM_MIN_CURR_THRESHOLD_SCALE - 1)
  155. | KTD2692_REG_MM_MIN_CURR_THRESHOLD_BASE);
  156. expresswire_write_u8(&led->props, KTD2692_FLASH_MODE_CURR_PERCENT(45)
  157. | KTD2692_REG_FLASH_CURRENT_BASE);
  158. }
  159. static void regulator_disable_action(void *_data)
  160. {
  161. struct device *dev = _data;
  162. struct ktd2692_context *led = dev_get_drvdata(dev);
  163. int ret;
  164. ret = regulator_disable(led->regulator);
  165. if (ret)
  166. dev_err(dev, "Failed to disable supply: %d\n", ret);
  167. }
  168. static int ktd2692_parse_dt(struct ktd2692_context *led, struct device *dev,
  169. struct ktd2692_led_config_data *cfg)
  170. {
  171. struct device_node *np = dev_of_node(dev);
  172. int ret;
  173. if (!np)
  174. return -ENXIO;
  175. led->props.ctrl_gpio = devm_gpiod_get(dev, "ctrl", GPIOD_ASIS);
  176. ret = PTR_ERR_OR_ZERO(led->props.ctrl_gpio);
  177. if (ret)
  178. return dev_err_probe(dev, ret, "cannot get ctrl-gpios\n");
  179. led->aux_gpio = devm_gpiod_get_optional(dev, "aux", GPIOD_ASIS);
  180. if (IS_ERR(led->aux_gpio))
  181. return dev_err_probe(dev, PTR_ERR(led->aux_gpio), "cannot get aux-gpios\n");
  182. led->regulator = devm_regulator_get(dev, "vin");
  183. if (IS_ERR(led->regulator))
  184. led->regulator = NULL;
  185. if (led->regulator) {
  186. ret = regulator_enable(led->regulator);
  187. if (ret) {
  188. dev_err(dev, "Failed to enable supply: %d\n", ret);
  189. } else {
  190. ret = devm_add_action_or_reset(dev,
  191. regulator_disable_action, dev);
  192. if (ret)
  193. return ret;
  194. }
  195. }
  196. struct device_node *child_node __free(device_node) =
  197. of_get_next_available_child(np, NULL);
  198. if (!child_node) {
  199. dev_err(dev, "No DT child node found for connected LED.\n");
  200. return -EINVAL;
  201. }
  202. led->fled_cdev.led_cdev.name =
  203. of_get_property(child_node, "label", NULL) ? : child_node->name;
  204. ret = of_property_read_u32(child_node, "led-max-microamp",
  205. &cfg->movie_max_microamp);
  206. if (ret) {
  207. dev_err(dev, "failed to parse led-max-microamp\n");
  208. return ret;
  209. }
  210. ret = of_property_read_u32(child_node, "flash-max-microamp",
  211. &cfg->flash_max_microamp);
  212. if (ret) {
  213. dev_err(dev, "failed to parse flash-max-microamp\n");
  214. return ret;
  215. }
  216. ret = of_property_read_u32(child_node, "flash-max-timeout-us",
  217. &cfg->flash_max_timeout);
  218. if (ret) {
  219. dev_err(dev, "failed to parse flash-max-timeout-us\n");
  220. return ret;
  221. }
  222. return 0;
  223. }
  224. static const struct led_flash_ops flash_ops = {
  225. .strobe_set = ktd2692_led_flash_strobe_set,
  226. .timeout_set = ktd2692_led_flash_timeout_set,
  227. };
  228. static int ktd2692_probe(struct platform_device *pdev)
  229. {
  230. struct ktd2692_context *led;
  231. struct led_classdev *led_cdev;
  232. struct led_classdev_flash *fled_cdev;
  233. struct ktd2692_led_config_data led_cfg;
  234. int ret;
  235. led = devm_kzalloc(&pdev->dev, sizeof(*led), GFP_KERNEL);
  236. if (!led)
  237. return -ENOMEM;
  238. fled_cdev = &led->fled_cdev;
  239. led_cdev = &fled_cdev->led_cdev;
  240. led->props.timing = ktd2692_timing;
  241. ret = ktd2692_parse_dt(led, &pdev->dev, &led_cfg);
  242. if (ret)
  243. return ret;
  244. ktd2692_init_flash_timeout(fled_cdev, &led_cfg);
  245. ktd2692_init_movie_current_max(&led_cfg);
  246. fled_cdev->ops = &flash_ops;
  247. led_cdev->max_brightness = led_cfg.max_brightness;
  248. led_cdev->brightness_set_blocking = ktd2692_led_brightness_set;
  249. led_cdev->flags |= LED_CORE_SUSPENDRESUME | LED_DEV_CAP_FLASH;
  250. mutex_init(&led->lock);
  251. platform_set_drvdata(pdev, led);
  252. ret = led_classdev_flash_register(&pdev->dev, fled_cdev);
  253. if (ret) {
  254. dev_err(&pdev->dev, "can't register LED %s\n", led_cdev->name);
  255. mutex_destroy(&led->lock);
  256. return ret;
  257. }
  258. ktd2692_setup(led);
  259. return 0;
  260. }
  261. static void ktd2692_remove(struct platform_device *pdev)
  262. {
  263. struct ktd2692_context *led = platform_get_drvdata(pdev);
  264. led_classdev_flash_unregister(&led->fled_cdev);
  265. mutex_destroy(&led->lock);
  266. }
  267. static const struct of_device_id ktd2692_match[] = {
  268. { .compatible = "kinetic,ktd2692", },
  269. { /* sentinel */ },
  270. };
  271. MODULE_DEVICE_TABLE(of, ktd2692_match);
  272. static struct platform_driver ktd2692_driver = {
  273. .driver = {
  274. .name = "ktd2692",
  275. .of_match_table = ktd2692_match,
  276. },
  277. .probe = ktd2692_probe,
  278. .remove_new = ktd2692_remove,
  279. };
  280. module_platform_driver(ktd2692_driver);
  281. MODULE_IMPORT_NS(EXPRESSWIRE);
  282. MODULE_AUTHOR("Ingi Kim <ingi2.kim@samsung.com>");
  283. MODULE_DESCRIPTION("Kinetic KTD2692 LED driver");
  284. MODULE_LICENSE("GPL v2");