leds-ktd2692.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444
  1. /*
  2. * LED driver : leds-ktd2692.c
  3. *
  4. * Copyright (C) 2015 Samsung Electronics
  5. * Ingi Kim <ingi2.kim@samsung.com>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License version 2 as
  9. * published by the Free Software Foundation.
  10. */
  11. #include <linux/delay.h>
  12. #include <linux/err.h>
  13. #include <linux/gpio/consumer.h>
  14. #include <linux/led-class-flash.h>
  15. #include <linux/module.h>
  16. #include <linux/mutex.h>
  17. #include <linux/of.h>
  18. #include <linux/platform_device.h>
  19. #include <linux/regulator/consumer.h>
  20. #include <linux/workqueue.h>
  21. /* Value related the movie mode */
  22. #define KTD2692_MOVIE_MODE_CURRENT_LEVELS 16
  23. #define KTD2692_MM_TO_FL_RATIO(x) ((x) / 3)
  24. #define KTD2962_MM_MIN_CURR_THRESHOLD_SCALE 8
  25. /* Value related the flash mode */
  26. #define KTD2692_FLASH_MODE_TIMEOUT_LEVELS 8
  27. #define KTD2692_FLASH_MODE_TIMEOUT_DISABLE 0
  28. #define KTD2692_FLASH_MODE_CURR_PERCENT(x) (((x) * 16) / 100)
  29. /* Macro for getting offset of flash timeout */
  30. #define GET_TIMEOUT_OFFSET(timeout, step) ((timeout) / (step))
  31. /* Base register address */
  32. #define KTD2692_REG_LVP_BASE 0x00
  33. #define KTD2692_REG_FLASH_TIMEOUT_BASE 0x20
  34. #define KTD2692_REG_MM_MIN_CURR_THRESHOLD_BASE 0x40
  35. #define KTD2692_REG_MOVIE_CURRENT_BASE 0x60
  36. #define KTD2692_REG_FLASH_CURRENT_BASE 0x80
  37. #define KTD2692_REG_MODE_BASE 0xA0
  38. /* Set bit coding time for expresswire interface */
  39. #define KTD2692_TIME_RESET_US 700
  40. #define KTD2692_TIME_DATA_START_TIME_US 10
  41. #define KTD2692_TIME_HIGH_END_OF_DATA_US 350
  42. #define KTD2692_TIME_LOW_END_OF_DATA_US 10
  43. #define KTD2692_TIME_SHORT_BITSET_US 4
  44. #define KTD2692_TIME_LONG_BITSET_US 12
  45. /* KTD2692 default length of name */
  46. #define KTD2692_NAME_LENGTH 20
  47. enum ktd2692_bitset {
  48. KTD2692_LOW = 0,
  49. KTD2692_HIGH,
  50. };
  51. /* Movie / Flash Mode Control */
  52. enum ktd2692_led_mode {
  53. KTD2692_MODE_DISABLE = 0, /* default */
  54. KTD2692_MODE_MOVIE,
  55. KTD2692_MODE_FLASH,
  56. };
  57. struct ktd2692_led_config_data {
  58. /* maximum LED current in movie mode */
  59. u32 movie_max_microamp;
  60. /* maximum LED current in flash mode */
  61. u32 flash_max_microamp;
  62. /* maximum flash timeout */
  63. u32 flash_max_timeout;
  64. /* max LED brightness level */
  65. enum led_brightness max_brightness;
  66. };
  67. struct ktd2692_context {
  68. /* Related LED Flash class device */
  69. struct led_classdev_flash fled_cdev;
  70. /* secures access to the device */
  71. struct mutex lock;
  72. struct regulator *regulator;
  73. struct work_struct work_brightness_set;
  74. struct gpio_desc *aux_gpio;
  75. struct gpio_desc *ctrl_gpio;
  76. enum ktd2692_led_mode mode;
  77. enum led_brightness torch_brightness;
  78. };
  79. static struct ktd2692_context *fled_cdev_to_led(
  80. struct led_classdev_flash *fled_cdev)
  81. {
  82. return container_of(fled_cdev, struct ktd2692_context, fled_cdev);
  83. }
  84. static void ktd2692_expresswire_start(struct ktd2692_context *led)
  85. {
  86. gpiod_direction_output(led->ctrl_gpio, KTD2692_HIGH);
  87. udelay(KTD2692_TIME_DATA_START_TIME_US);
  88. }
  89. static void ktd2692_expresswire_reset(struct ktd2692_context *led)
  90. {
  91. gpiod_direction_output(led->ctrl_gpio, KTD2692_LOW);
  92. udelay(KTD2692_TIME_RESET_US);
  93. }
  94. static void ktd2692_expresswire_end(struct ktd2692_context *led)
  95. {
  96. gpiod_direction_output(led->ctrl_gpio, KTD2692_LOW);
  97. udelay(KTD2692_TIME_LOW_END_OF_DATA_US);
  98. gpiod_direction_output(led->ctrl_gpio, KTD2692_HIGH);
  99. udelay(KTD2692_TIME_HIGH_END_OF_DATA_US);
  100. }
  101. static void ktd2692_expresswire_set_bit(struct ktd2692_context *led, bool bit)
  102. {
  103. /*
  104. * The Low Bit(0) and High Bit(1) is based on a time detection
  105. * algorithm between time low and time high
  106. * Time_(L_LB) : Low time of the Low Bit(0)
  107. * Time_(H_LB) : High time of the LOW Bit(0)
  108. * Time_(L_HB) : Low time of the High Bit(1)
  109. * Time_(H_HB) : High time of the High Bit(1)
  110. *
  111. * It can be simplified to:
  112. * Low Bit(0) : 2 * Time_(H_LB) < Time_(L_LB)
  113. * High Bit(1) : 2 * Time_(L_HB) < Time_(H_HB)
  114. * HIGH ___ ____ _.. _________ ___
  115. * |_________| |_.. |____| |__|
  116. * LOW <L_LB> <H_LB> <L_HB> <H_HB>
  117. * [ Low Bit (0) ] [ High Bit(1) ]
  118. */
  119. if (bit) {
  120. gpiod_direction_output(led->ctrl_gpio, KTD2692_LOW);
  121. udelay(KTD2692_TIME_SHORT_BITSET_US);
  122. gpiod_direction_output(led->ctrl_gpio, KTD2692_HIGH);
  123. udelay(KTD2692_TIME_LONG_BITSET_US);
  124. } else {
  125. gpiod_direction_output(led->ctrl_gpio, KTD2692_LOW);
  126. udelay(KTD2692_TIME_LONG_BITSET_US);
  127. gpiod_direction_output(led->ctrl_gpio, KTD2692_HIGH);
  128. udelay(KTD2692_TIME_SHORT_BITSET_US);
  129. }
  130. }
  131. static void ktd2692_expresswire_write(struct ktd2692_context *led, u8 value)
  132. {
  133. int i;
  134. ktd2692_expresswire_start(led);
  135. for (i = 7; i >= 0; i--)
  136. ktd2692_expresswire_set_bit(led, value & BIT(i));
  137. ktd2692_expresswire_end(led);
  138. }
  139. static void ktd2692_brightness_set(struct ktd2692_context *led,
  140. enum led_brightness brightness)
  141. {
  142. mutex_lock(&led->lock);
  143. if (brightness == LED_OFF) {
  144. led->mode = KTD2692_MODE_DISABLE;
  145. gpiod_direction_output(led->aux_gpio, KTD2692_LOW);
  146. } else {
  147. ktd2692_expresswire_write(led, brightness |
  148. KTD2692_REG_MOVIE_CURRENT_BASE);
  149. led->mode = KTD2692_MODE_MOVIE;
  150. }
  151. ktd2692_expresswire_write(led, led->mode | KTD2692_REG_MODE_BASE);
  152. mutex_unlock(&led->lock);
  153. }
  154. static void ktd2692_brightness_set_work(struct work_struct *work)
  155. {
  156. struct ktd2692_context *led =
  157. container_of(work, struct ktd2692_context, work_brightness_set);
  158. ktd2692_brightness_set(led, led->torch_brightness);
  159. }
  160. static void ktd2692_led_brightness_set(struct led_classdev *led_cdev,
  161. enum led_brightness brightness)
  162. {
  163. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  164. struct ktd2692_context *led = fled_cdev_to_led(fled_cdev);
  165. led->torch_brightness = brightness;
  166. schedule_work(&led->work_brightness_set);
  167. }
  168. static int ktd2692_led_brightness_set_sync(struct led_classdev *led_cdev,
  169. enum led_brightness brightness)
  170. {
  171. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  172. struct ktd2692_context *led = fled_cdev_to_led(fled_cdev);
  173. ktd2692_brightness_set(led, brightness);
  174. return 0;
  175. }
  176. static int ktd2692_led_flash_strobe_set(struct led_classdev_flash *fled_cdev,
  177. bool state)
  178. {
  179. struct ktd2692_context *led = fled_cdev_to_led(fled_cdev);
  180. struct led_flash_setting *timeout = &fled_cdev->timeout;
  181. u32 flash_tm_reg;
  182. mutex_lock(&led->lock);
  183. if (state) {
  184. flash_tm_reg = GET_TIMEOUT_OFFSET(timeout->val, timeout->step);
  185. ktd2692_expresswire_write(led, flash_tm_reg
  186. | KTD2692_REG_FLASH_TIMEOUT_BASE);
  187. led->mode = KTD2692_MODE_FLASH;
  188. gpiod_direction_output(led->aux_gpio, KTD2692_HIGH);
  189. } else {
  190. led->mode = KTD2692_MODE_DISABLE;
  191. gpiod_direction_output(led->aux_gpio, KTD2692_LOW);
  192. }
  193. ktd2692_expresswire_write(led, led->mode | KTD2692_REG_MODE_BASE);
  194. fled_cdev->led_cdev.brightness = LED_OFF;
  195. led->mode = KTD2692_MODE_DISABLE;
  196. mutex_unlock(&led->lock);
  197. return 0;
  198. }
  199. static int ktd2692_led_flash_timeout_set(struct led_classdev_flash *fled_cdev,
  200. u32 timeout)
  201. {
  202. return 0;
  203. }
  204. static void ktd2692_init_movie_current_max(struct ktd2692_led_config_data *cfg)
  205. {
  206. u32 offset, step;
  207. u32 movie_current_microamp;
  208. offset = KTD2692_MOVIE_MODE_CURRENT_LEVELS;
  209. step = KTD2692_MM_TO_FL_RATIO(cfg->flash_max_microamp)
  210. / KTD2692_MOVIE_MODE_CURRENT_LEVELS;
  211. do {
  212. movie_current_microamp = step * offset;
  213. offset--;
  214. } while ((movie_current_microamp > cfg->movie_max_microamp) &&
  215. (offset > 0));
  216. cfg->max_brightness = offset;
  217. }
  218. static void ktd2692_init_flash_timeout(struct led_classdev_flash *fled_cdev,
  219. struct ktd2692_led_config_data *cfg)
  220. {
  221. struct led_flash_setting *setting;
  222. setting = &fled_cdev->timeout;
  223. setting->min = KTD2692_FLASH_MODE_TIMEOUT_DISABLE;
  224. setting->max = cfg->flash_max_timeout;
  225. setting->step = cfg->flash_max_timeout
  226. / (KTD2692_FLASH_MODE_TIMEOUT_LEVELS - 1);
  227. setting->val = cfg->flash_max_timeout;
  228. }
  229. static void ktd2692_setup(struct ktd2692_context *led)
  230. {
  231. led->mode = KTD2692_MODE_DISABLE;
  232. ktd2692_expresswire_reset(led);
  233. gpiod_direction_output(led->aux_gpio, KTD2692_LOW);
  234. ktd2692_expresswire_write(led, (KTD2962_MM_MIN_CURR_THRESHOLD_SCALE - 1)
  235. | KTD2692_REG_MM_MIN_CURR_THRESHOLD_BASE);
  236. ktd2692_expresswire_write(led, KTD2692_FLASH_MODE_CURR_PERCENT(45)
  237. | KTD2692_REG_FLASH_CURRENT_BASE);
  238. }
  239. static int ktd2692_parse_dt(struct ktd2692_context *led, struct device *dev,
  240. struct ktd2692_led_config_data *cfg)
  241. {
  242. struct device_node *np = dev->of_node;
  243. struct device_node *child_node;
  244. int ret;
  245. if (!dev->of_node)
  246. return -ENXIO;
  247. led->ctrl_gpio = devm_gpiod_get(dev, "ctrl", GPIOD_ASIS);
  248. ret = PTR_ERR_OR_ZERO(led->ctrl_gpio);
  249. if (ret) {
  250. dev_err(dev, "cannot get ctrl-gpios %d\n", ret);
  251. return ret;
  252. }
  253. led->aux_gpio = devm_gpiod_get(dev, "aux", GPIOD_ASIS);
  254. ret = PTR_ERR_OR_ZERO(led->aux_gpio);
  255. if (ret) {
  256. dev_err(dev, "cannot get aux-gpios %d\n", ret);
  257. return ret;
  258. }
  259. led->regulator = devm_regulator_get(dev, "vin");
  260. if (IS_ERR(led->regulator))
  261. led->regulator = NULL;
  262. if (led->regulator) {
  263. ret = regulator_enable(led->regulator);
  264. if (ret)
  265. dev_err(dev, "Failed to enable supply: %d\n", ret);
  266. }
  267. child_node = of_get_next_available_child(np, NULL);
  268. if (!child_node) {
  269. dev_err(dev, "No DT child node found for connected LED.\n");
  270. return -EINVAL;
  271. }
  272. led->fled_cdev.led_cdev.name =
  273. of_get_property(child_node, "label", NULL) ? : child_node->name;
  274. ret = of_property_read_u32(child_node, "led-max-microamp",
  275. &cfg->movie_max_microamp);
  276. if (ret) {
  277. dev_err(dev, "failed to parse led-max-microamp\n");
  278. return ret;
  279. }
  280. ret = of_property_read_u32(child_node, "flash-max-microamp",
  281. &cfg->flash_max_microamp);
  282. if (ret) {
  283. dev_err(dev, "failed to parse flash-max-microamp\n");
  284. return ret;
  285. }
  286. ret = of_property_read_u32(child_node, "flash-max-timeout-us",
  287. &cfg->flash_max_timeout);
  288. if (ret)
  289. dev_err(dev, "failed to parse flash-max-timeout-us\n");
  290. of_node_put(child_node);
  291. return ret;
  292. }
  293. static const struct led_flash_ops flash_ops = {
  294. .strobe_set = ktd2692_led_flash_strobe_set,
  295. .timeout_set = ktd2692_led_flash_timeout_set,
  296. };
  297. static int ktd2692_probe(struct platform_device *pdev)
  298. {
  299. struct ktd2692_context *led;
  300. struct led_classdev *led_cdev;
  301. struct led_classdev_flash *fled_cdev;
  302. struct ktd2692_led_config_data led_cfg;
  303. int ret;
  304. led = devm_kzalloc(&pdev->dev, sizeof(*led), GFP_KERNEL);
  305. if (!led)
  306. return -ENOMEM;
  307. fled_cdev = &led->fled_cdev;
  308. led_cdev = &fled_cdev->led_cdev;
  309. ret = ktd2692_parse_dt(led, &pdev->dev, &led_cfg);
  310. if (ret)
  311. return ret;
  312. ktd2692_init_flash_timeout(fled_cdev, &led_cfg);
  313. ktd2692_init_movie_current_max(&led_cfg);
  314. fled_cdev->ops = &flash_ops;
  315. led_cdev->max_brightness = led_cfg.max_brightness;
  316. led_cdev->brightness_set = ktd2692_led_brightness_set;
  317. led_cdev->brightness_set_sync = ktd2692_led_brightness_set_sync;
  318. led_cdev->flags |= LED_CORE_SUSPENDRESUME | LED_DEV_CAP_FLASH;
  319. mutex_init(&led->lock);
  320. INIT_WORK(&led->work_brightness_set, ktd2692_brightness_set_work);
  321. platform_set_drvdata(pdev, led);
  322. ret = led_classdev_flash_register(&pdev->dev, fled_cdev);
  323. if (ret) {
  324. dev_err(&pdev->dev, "can't register LED %s\n", led_cdev->name);
  325. mutex_destroy(&led->lock);
  326. return ret;
  327. }
  328. ktd2692_setup(led);
  329. return 0;
  330. }
  331. static int ktd2692_remove(struct platform_device *pdev)
  332. {
  333. struct ktd2692_context *led = platform_get_drvdata(pdev);
  334. int ret;
  335. led_classdev_flash_unregister(&led->fled_cdev);
  336. cancel_work_sync(&led->work_brightness_set);
  337. if (led->regulator) {
  338. ret = regulator_disable(led->regulator);
  339. if (ret)
  340. dev_err(&pdev->dev,
  341. "Failed to disable supply: %d\n", ret);
  342. }
  343. mutex_destroy(&led->lock);
  344. return 0;
  345. }
  346. static const struct of_device_id ktd2692_match[] = {
  347. { .compatible = "kinetic,ktd2692", },
  348. { /* sentinel */ },
  349. };
  350. MODULE_DEVICE_TABLE(of, ktd2692_match);
  351. static struct platform_driver ktd2692_driver = {
  352. .driver = {
  353. .name = "ktd2692",
  354. .of_match_table = ktd2692_match,
  355. },
  356. .probe = ktd2692_probe,
  357. .remove = ktd2692_remove,
  358. };
  359. module_platform_driver(ktd2692_driver);
  360. MODULE_AUTHOR("Ingi Kim <ingi2.kim@samsung.com>");
  361. MODULE_DESCRIPTION("Kinetic KTD2692 LED driver");
  362. MODULE_LICENSE("GPL v2");