led-class-flash.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * LED Flash class interface
  4. *
  5. * Copyright (C) 2015 Samsung Electronics Co., Ltd.
  6. * Author: Jacek Anaszewski <j.anaszewski@samsung.com>
  7. */
  8. #include <linux/device.h>
  9. #include <linux/init.h>
  10. #include <linux/led-class-flash.h>
  11. #include <linux/leds.h>
  12. #include <linux/module.h>
  13. #include <linux/slab.h>
  14. #define has_flash_op(fled_cdev, op) \
  15. (fled_cdev && fled_cdev->ops->op)
  16. #define call_flash_op(fled_cdev, op, args...) \
  17. ((has_flash_op(fled_cdev, op)) ? \
  18. (fled_cdev->ops->op(fled_cdev, args)) : \
  19. -EINVAL)
  20. static const char * const led_flash_fault_names[] = {
  21. "led-over-voltage",
  22. "flash-timeout-exceeded",
  23. "controller-over-temperature",
  24. "controller-short-circuit",
  25. "led-power-supply-over-current",
  26. "indicator-led-fault",
  27. "led-under-voltage",
  28. "controller-under-voltage",
  29. "led-over-temperature",
  30. };
  31. static ssize_t flash_brightness_store(struct device *dev,
  32. struct device_attribute *attr, const char *buf, size_t size)
  33. {
  34. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  35. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  36. unsigned long state;
  37. ssize_t ret;
  38. mutex_lock(&led_cdev->led_access);
  39. if (led_sysfs_is_disabled(led_cdev)) {
  40. ret = -EBUSY;
  41. goto unlock;
  42. }
  43. ret = kstrtoul(buf, 10, &state);
  44. if (ret)
  45. goto unlock;
  46. ret = led_set_flash_brightness(fled_cdev, state);
  47. if (ret < 0)
  48. goto unlock;
  49. ret = size;
  50. unlock:
  51. mutex_unlock(&led_cdev->led_access);
  52. return ret;
  53. }
  54. static ssize_t flash_brightness_show(struct device *dev,
  55. struct device_attribute *attr, char *buf)
  56. {
  57. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  58. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  59. /* no lock needed for this */
  60. led_update_flash_brightness(fled_cdev);
  61. return sprintf(buf, "%u\n", fled_cdev->brightness.val);
  62. }
  63. static DEVICE_ATTR_RW(flash_brightness);
  64. static ssize_t max_flash_brightness_show(struct device *dev,
  65. struct device_attribute *attr, char *buf)
  66. {
  67. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  68. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  69. return sprintf(buf, "%u\n", fled_cdev->brightness.max);
  70. }
  71. static DEVICE_ATTR_RO(max_flash_brightness);
  72. static ssize_t flash_strobe_store(struct device *dev,
  73. struct device_attribute *attr, const char *buf, size_t size)
  74. {
  75. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  76. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  77. unsigned long state;
  78. ssize_t ret = -EBUSY;
  79. mutex_lock(&led_cdev->led_access);
  80. if (led_sysfs_is_disabled(led_cdev))
  81. goto unlock;
  82. ret = kstrtoul(buf, 10, &state);
  83. if (ret)
  84. goto unlock;
  85. if (state > 1) {
  86. ret = -EINVAL;
  87. goto unlock;
  88. }
  89. ret = led_set_flash_strobe(fled_cdev, state);
  90. if (ret < 0)
  91. goto unlock;
  92. ret = size;
  93. unlock:
  94. mutex_unlock(&led_cdev->led_access);
  95. return ret;
  96. }
  97. static ssize_t flash_strobe_show(struct device *dev,
  98. struct device_attribute *attr, char *buf)
  99. {
  100. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  101. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  102. bool state;
  103. int ret;
  104. /* no lock needed for this */
  105. ret = led_get_flash_strobe(fled_cdev, &state);
  106. if (ret < 0)
  107. return ret;
  108. return sprintf(buf, "%u\n", state);
  109. }
  110. static DEVICE_ATTR_RW(flash_strobe);
  111. static ssize_t flash_timeout_store(struct device *dev,
  112. struct device_attribute *attr, const char *buf, size_t size)
  113. {
  114. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  115. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  116. unsigned long flash_timeout;
  117. ssize_t ret;
  118. mutex_lock(&led_cdev->led_access);
  119. if (led_sysfs_is_disabled(led_cdev)) {
  120. ret = -EBUSY;
  121. goto unlock;
  122. }
  123. ret = kstrtoul(buf, 10, &flash_timeout);
  124. if (ret)
  125. goto unlock;
  126. ret = led_set_flash_timeout(fled_cdev, flash_timeout);
  127. if (ret < 0)
  128. goto unlock;
  129. ret = size;
  130. unlock:
  131. mutex_unlock(&led_cdev->led_access);
  132. return ret;
  133. }
  134. static ssize_t flash_timeout_show(struct device *dev,
  135. struct device_attribute *attr, char *buf)
  136. {
  137. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  138. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  139. return sprintf(buf, "%u\n", fled_cdev->timeout.val);
  140. }
  141. static DEVICE_ATTR_RW(flash_timeout);
  142. static ssize_t max_flash_timeout_show(struct device *dev,
  143. struct device_attribute *attr, char *buf)
  144. {
  145. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  146. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  147. return sprintf(buf, "%u\n", fled_cdev->timeout.max);
  148. }
  149. static DEVICE_ATTR_RO(max_flash_timeout);
  150. static ssize_t flash_fault_show(struct device *dev,
  151. struct device_attribute *attr, char *buf)
  152. {
  153. struct led_classdev *led_cdev = dev_get_drvdata(dev);
  154. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  155. u32 fault, mask = 0x1;
  156. char *pbuf = buf;
  157. int i, ret, buf_len;
  158. ret = led_get_flash_fault(fled_cdev, &fault);
  159. if (ret < 0)
  160. return -EINVAL;
  161. *buf = '\0';
  162. for (i = 0; i < LED_NUM_FLASH_FAULTS; ++i) {
  163. if (fault & mask) {
  164. buf_len = sprintf(pbuf, "%s ",
  165. led_flash_fault_names[i]);
  166. pbuf += buf_len;
  167. }
  168. mask <<= 1;
  169. }
  170. return strlen(strcat(buf, "\n"));
  171. }
  172. static DEVICE_ATTR_RO(flash_fault);
  173. static struct attribute *led_flash_strobe_attrs[] = {
  174. &dev_attr_flash_strobe.attr,
  175. NULL,
  176. };
  177. static struct attribute *led_flash_timeout_attrs[] = {
  178. &dev_attr_flash_timeout.attr,
  179. &dev_attr_max_flash_timeout.attr,
  180. NULL,
  181. };
  182. static struct attribute *led_flash_brightness_attrs[] = {
  183. &dev_attr_flash_brightness.attr,
  184. &dev_attr_max_flash_brightness.attr,
  185. NULL,
  186. };
  187. static struct attribute *led_flash_fault_attrs[] = {
  188. &dev_attr_flash_fault.attr,
  189. NULL,
  190. };
  191. static const struct attribute_group led_flash_strobe_group = {
  192. .attrs = led_flash_strobe_attrs,
  193. };
  194. static const struct attribute_group led_flash_timeout_group = {
  195. .attrs = led_flash_timeout_attrs,
  196. };
  197. static const struct attribute_group led_flash_brightness_group = {
  198. .attrs = led_flash_brightness_attrs,
  199. };
  200. static const struct attribute_group led_flash_fault_group = {
  201. .attrs = led_flash_fault_attrs,
  202. };
  203. static void led_flash_resume(struct led_classdev *led_cdev)
  204. {
  205. struct led_classdev_flash *fled_cdev = lcdev_to_flcdev(led_cdev);
  206. call_flash_op(fled_cdev, flash_brightness_set,
  207. fled_cdev->brightness.val);
  208. call_flash_op(fled_cdev, timeout_set, fled_cdev->timeout.val);
  209. }
  210. static void led_flash_init_sysfs_groups(struct led_classdev_flash *fled_cdev)
  211. {
  212. struct led_classdev *led_cdev = &fled_cdev->led_cdev;
  213. const struct led_flash_ops *ops = fled_cdev->ops;
  214. const struct attribute_group **flash_groups = fled_cdev->sysfs_groups;
  215. int num_sysfs_groups = 0;
  216. flash_groups[num_sysfs_groups++] = &led_flash_strobe_group;
  217. if (ops->flash_brightness_set)
  218. flash_groups[num_sysfs_groups++] = &led_flash_brightness_group;
  219. if (ops->timeout_set)
  220. flash_groups[num_sysfs_groups++] = &led_flash_timeout_group;
  221. if (ops->fault_get)
  222. flash_groups[num_sysfs_groups++] = &led_flash_fault_group;
  223. led_cdev->groups = flash_groups;
  224. }
  225. int led_classdev_flash_register_ext(struct device *parent,
  226. struct led_classdev_flash *fled_cdev,
  227. struct led_init_data *init_data)
  228. {
  229. struct led_classdev *led_cdev;
  230. const struct led_flash_ops *ops;
  231. int ret;
  232. if (!fled_cdev)
  233. return -EINVAL;
  234. led_cdev = &fled_cdev->led_cdev;
  235. if (led_cdev->flags & LED_DEV_CAP_FLASH) {
  236. if (!led_cdev->brightness_set_blocking)
  237. return -EINVAL;
  238. ops = fled_cdev->ops;
  239. if (!ops || !ops->strobe_set)
  240. return -EINVAL;
  241. led_cdev->flash_resume = led_flash_resume;
  242. /* Select the sysfs attributes to be created for the device */
  243. led_flash_init_sysfs_groups(fled_cdev);
  244. }
  245. /* Register led class device */
  246. ret = led_classdev_register_ext(parent, led_cdev, init_data);
  247. if (ret < 0)
  248. return ret;
  249. return 0;
  250. }
  251. EXPORT_SYMBOL_GPL(led_classdev_flash_register_ext);
  252. void led_classdev_flash_unregister(struct led_classdev_flash *fled_cdev)
  253. {
  254. if (!fled_cdev)
  255. return;
  256. led_classdev_unregister(&fled_cdev->led_cdev);
  257. }
  258. EXPORT_SYMBOL_GPL(led_classdev_flash_unregister);
  259. static void devm_led_classdev_flash_release(struct device *dev, void *res)
  260. {
  261. led_classdev_flash_unregister(*(struct led_classdev_flash **)res);
  262. }
  263. int devm_led_classdev_flash_register_ext(struct device *parent,
  264. struct led_classdev_flash *fled_cdev,
  265. struct led_init_data *init_data)
  266. {
  267. struct led_classdev_flash **dr;
  268. int ret;
  269. dr = devres_alloc(devm_led_classdev_flash_release, sizeof(*dr),
  270. GFP_KERNEL);
  271. if (!dr)
  272. return -ENOMEM;
  273. ret = led_classdev_flash_register_ext(parent, fled_cdev, init_data);
  274. if (ret) {
  275. devres_free(dr);
  276. return ret;
  277. }
  278. *dr = fled_cdev;
  279. devres_add(parent, dr);
  280. return 0;
  281. }
  282. EXPORT_SYMBOL_GPL(devm_led_classdev_flash_register_ext);
  283. static int devm_led_classdev_flash_match(struct device *dev,
  284. void *res, void *data)
  285. {
  286. struct led_classdev_flash **p = res;
  287. if (WARN_ON(!p || !*p))
  288. return 0;
  289. return *p == data;
  290. }
  291. void devm_led_classdev_flash_unregister(struct device *dev,
  292. struct led_classdev_flash *fled_cdev)
  293. {
  294. WARN_ON(devres_release(dev,
  295. devm_led_classdev_flash_release,
  296. devm_led_classdev_flash_match, fled_cdev));
  297. }
  298. EXPORT_SYMBOL_GPL(devm_led_classdev_flash_unregister);
  299. static void led_clamp_align(struct led_flash_setting *s)
  300. {
  301. u32 v, offset;
  302. v = s->val + s->step / 2;
  303. v = clamp(v, s->min, s->max);
  304. offset = v - s->min;
  305. offset = s->step * (offset / s->step);
  306. s->val = s->min + offset;
  307. }
  308. int led_set_flash_timeout(struct led_classdev_flash *fled_cdev, u32 timeout)
  309. {
  310. struct led_classdev *led_cdev = &fled_cdev->led_cdev;
  311. struct led_flash_setting *s = &fled_cdev->timeout;
  312. s->val = timeout;
  313. led_clamp_align(s);
  314. if (!(led_cdev->flags & LED_SUSPENDED))
  315. return call_flash_op(fled_cdev, timeout_set, s->val);
  316. return 0;
  317. }
  318. EXPORT_SYMBOL_GPL(led_set_flash_timeout);
  319. int led_get_flash_fault(struct led_classdev_flash *fled_cdev, u32 *fault)
  320. {
  321. return call_flash_op(fled_cdev, fault_get, fault);
  322. }
  323. EXPORT_SYMBOL_GPL(led_get_flash_fault);
  324. int led_set_flash_brightness(struct led_classdev_flash *fled_cdev,
  325. u32 brightness)
  326. {
  327. struct led_classdev *led_cdev = &fled_cdev->led_cdev;
  328. struct led_flash_setting *s = &fled_cdev->brightness;
  329. s->val = brightness;
  330. led_clamp_align(s);
  331. if (!(led_cdev->flags & LED_SUSPENDED))
  332. return call_flash_op(fled_cdev, flash_brightness_set, s->val);
  333. return 0;
  334. }
  335. EXPORT_SYMBOL_GPL(led_set_flash_brightness);
  336. int led_update_flash_brightness(struct led_classdev_flash *fled_cdev)
  337. {
  338. struct led_flash_setting *s = &fled_cdev->brightness;
  339. u32 brightness;
  340. if (has_flash_op(fled_cdev, flash_brightness_get)) {
  341. int ret = call_flash_op(fled_cdev, flash_brightness_get,
  342. &brightness);
  343. if (ret < 0)
  344. return ret;
  345. s->val = brightness;
  346. }
  347. return 0;
  348. }
  349. EXPORT_SYMBOL_GPL(led_update_flash_brightness);
  350. int led_set_flash_duration(struct led_classdev_flash *fled_cdev, u32 duration)
  351. {
  352. struct led_classdev *led_cdev = &fled_cdev->led_cdev;
  353. struct led_flash_setting *s = &fled_cdev->duration;
  354. s->val = duration;
  355. led_clamp_align(s);
  356. if (!(led_cdev->flags & LED_SUSPENDED))
  357. return call_flash_op(fled_cdev, duration_set, s->val);
  358. return 0;
  359. }
  360. EXPORT_SYMBOL_GPL(led_set_flash_duration);
  361. MODULE_AUTHOR("Jacek Anaszewski <j.anaszewski@samsung.com>");
  362. MODULE_DESCRIPTION("LED Flash class interface");
  363. MODULE_LICENSE("GPL v2");