pwm_backlight.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (c) 2016 Google, Inc
  4. * Written by Simon Glass <sjg@chromium.org>
  5. */
  6. #define LOG_CATEGORY UCLASS_PANEL_BACKLIGHT
  7. #include <common.h>
  8. #include <dm.h>
  9. #include <backlight.h>
  10. #include <log.h>
  11. #include <malloc.h>
  12. #include <pwm.h>
  13. #include <asm/gpio.h>
  14. #include <linux/delay.h>
  15. #include <linux/math64.h>
  16. #include <power/regulator.h>
  17. /**
  18. * Private information for the PWM backlight
  19. *
  20. * If @num_levels is 0 then the levels are simple values with the backlight
  21. * value going between the minimum (default 0) and the maximum (default 255).
  22. * Otherwise the levels are an index into @levels (0..n-1).
  23. *
  24. * @reg: Regulator to enable to turn the backlight on (NULL if none)
  25. * @enable, GPIO to set to enable the backlight (can be missing)
  26. * @pwm: PWM to use to change the backlight brightness
  27. * @channel: PWM channel to use
  28. * @period_ns: Period of the backlight in nanoseconds
  29. * @levels: Levels for the backlight, or NULL if not using indexed levels
  30. * @num_levels: Number of levels
  31. * @cur_level: Current level for the backlight (index or value)
  32. * @default_level: Default level for the backlight (index or value)
  33. * @min_level: Minimum level of the backlight (full off)
  34. * @max_level: Maximum level of the backlight (full on)
  35. * @enabled: true if backlight is enabled
  36. */
  37. struct pwm_backlight_priv {
  38. struct udevice *reg;
  39. struct gpio_desc enable;
  40. struct udevice *pwm;
  41. uint channel;
  42. uint period_ns;
  43. /*
  44. * the polarity of one PWM
  45. * 0: normal polarity
  46. * 1: inverted polarity
  47. */
  48. bool polarity;
  49. u32 *levels;
  50. int num_levels;
  51. uint default_level;
  52. int cur_level;
  53. uint min_level;
  54. uint max_level;
  55. bool enabled;
  56. };
  57. static int set_pwm(struct pwm_backlight_priv *priv)
  58. {
  59. u64 width;
  60. uint duty_cycle;
  61. int ret;
  62. if (priv->period_ns) {
  63. width = priv->period_ns * (priv->cur_level - priv->min_level);
  64. duty_cycle = div_u64(width,
  65. (priv->max_level - priv->min_level));
  66. ret = pwm_set_config(priv->pwm, priv->channel, priv->period_ns,
  67. duty_cycle);
  68. } else {
  69. /* PWM driver will internally scale these like the above. */
  70. ret = pwm_set_config(priv->pwm, priv->channel,
  71. priv->max_level - priv->min_level,
  72. priv->cur_level - priv->min_level);
  73. }
  74. if (ret)
  75. return log_ret(ret);
  76. ret = pwm_set_invert(priv->pwm, priv->channel, priv->polarity);
  77. if (ret == -ENOSYS && !priv->polarity)
  78. ret = 0;
  79. return log_ret(ret);
  80. }
  81. static int enable_sequence(struct udevice *dev, int seq)
  82. {
  83. struct pwm_backlight_priv *priv = dev_get_priv(dev);
  84. int ret;
  85. switch (seq) {
  86. case 0:
  87. if (priv->reg) {
  88. __maybe_unused struct dm_regulator_uclass_plat
  89. *plat;
  90. plat = dev_get_uclass_plat(priv->reg);
  91. log_debug("Enable '%s', regulator '%s'/'%s'\n",
  92. dev->name, priv->reg->name, plat->name);
  93. ret = regulator_set_enable(priv->reg, true);
  94. if (ret) {
  95. log_debug("Cannot enable regulator for PWM '%s'\n",
  96. dev->name);
  97. return log_ret(ret);
  98. }
  99. mdelay(120);
  100. }
  101. break;
  102. case 1:
  103. mdelay(10);
  104. dm_gpio_set_value(&priv->enable, 1);
  105. break;
  106. }
  107. return 0;
  108. }
  109. static int pwm_backlight_enable(struct udevice *dev)
  110. {
  111. struct pwm_backlight_priv *priv = dev_get_priv(dev);
  112. int ret;
  113. ret = enable_sequence(dev, 0);
  114. if (ret)
  115. return log_ret(ret);
  116. ret = set_pwm(priv);
  117. if (ret)
  118. return log_ret(ret);
  119. ret = pwm_set_enable(priv->pwm, priv->channel, true);
  120. if (ret)
  121. return log_ret(ret);
  122. ret = enable_sequence(dev, 1);
  123. if (ret)
  124. return log_ret(ret);
  125. priv->enabled = true;
  126. return 0;
  127. }
  128. static int pwm_backlight_set_brightness(struct udevice *dev, int percent)
  129. {
  130. struct pwm_backlight_priv *priv = dev_get_priv(dev);
  131. bool disable = false;
  132. int level;
  133. int ret;
  134. if (!priv->enabled) {
  135. ret = enable_sequence(dev, 0);
  136. if (ret)
  137. return log_ret(ret);
  138. }
  139. if (percent == BACKLIGHT_OFF) {
  140. disable = true;
  141. percent = 0;
  142. }
  143. if (percent == BACKLIGHT_DEFAULT) {
  144. level = priv->default_level;
  145. } else {
  146. if (priv->levels) {
  147. level = priv->levels[percent * (priv->num_levels - 1)
  148. / 100];
  149. } else {
  150. level = priv->min_level +
  151. (priv->max_level - priv->min_level) *
  152. percent / 100;
  153. }
  154. }
  155. priv->cur_level = level;
  156. ret = set_pwm(priv);
  157. if (ret)
  158. return log_ret(ret);
  159. if (!priv->enabled) {
  160. ret = enable_sequence(dev, 1);
  161. if (ret)
  162. return log_ret(ret);
  163. priv->enabled = true;
  164. }
  165. if (disable) {
  166. dm_gpio_set_value(&priv->enable, 0);
  167. if (priv->reg) {
  168. ret = regulator_set_enable(priv->reg, false);
  169. if (ret)
  170. return log_ret(ret);
  171. }
  172. priv->enabled = false;
  173. }
  174. return 0;
  175. }
  176. static int pwm_backlight_of_to_plat(struct udevice *dev)
  177. {
  178. struct pwm_backlight_priv *priv = dev_get_priv(dev);
  179. struct ofnode_phandle_args args;
  180. int index, ret, count, len;
  181. const u32 *cell;
  182. log_debug("start\n");
  183. ret = uclass_get_device_by_phandle(UCLASS_REGULATOR, dev,
  184. "power-supply", &priv->reg);
  185. if (ret)
  186. log_debug("Cannot get power supply: ret=%d\n", ret);
  187. ret = gpio_request_by_name(dev, "enable-gpios", 0, &priv->enable,
  188. GPIOD_IS_OUT);
  189. if (ret) {
  190. log_debug("Warning: cannot get enable GPIO: ret=%d\n", ret);
  191. if (ret != -ENOENT)
  192. return log_ret(ret);
  193. }
  194. ret = dev_read_phandle_with_args(dev, "pwms", "#pwm-cells", 0, 0,
  195. &args);
  196. if (ret) {
  197. log_debug("Cannot get PWM phandle: ret=%d\n", ret);
  198. return log_ret(ret);
  199. }
  200. ret = uclass_get_device_by_ofnode(UCLASS_PWM, args.node, &priv->pwm);
  201. if (ret) {
  202. log_debug("Cannot get PWM: ret=%d\n", ret);
  203. return log_ret(ret);
  204. }
  205. if (args.args_count < 1)
  206. return log_msg_ret("Not enough arguments to pwm\n", -EINVAL);
  207. priv->channel = args.args[0];
  208. if (args.args_count > 1)
  209. priv->period_ns = args.args[1];
  210. if (args.args_count > 2)
  211. priv->polarity = args.args[2];
  212. index = dev_read_u32_default(dev, "default-brightness-level", 255);
  213. cell = dev_read_prop(dev, "brightness-levels", &len);
  214. count = len / sizeof(u32);
  215. if (cell && count > index) {
  216. priv->levels = malloc(len);
  217. if (!priv->levels)
  218. return log_ret(-ENOMEM);
  219. ret = dev_read_u32_array(dev, "brightness-levels", priv->levels,
  220. count);
  221. if (ret)
  222. return log_msg_ret("levels", ret);
  223. priv->num_levels = count;
  224. priv->default_level = priv->levels[index];
  225. priv->max_level = priv->levels[count - 1];
  226. } else {
  227. priv->default_level = index;
  228. priv->max_level = 255;
  229. }
  230. priv->cur_level = priv->default_level;
  231. log_debug("done\n");
  232. return 0;
  233. }
  234. static int pwm_backlight_probe(struct udevice *dev)
  235. {
  236. return 0;
  237. }
  238. static const struct backlight_ops pwm_backlight_ops = {
  239. .enable = pwm_backlight_enable,
  240. .set_brightness = pwm_backlight_set_brightness,
  241. };
  242. static const struct udevice_id pwm_backlight_ids[] = {
  243. { .compatible = "pwm-backlight" },
  244. { }
  245. };
  246. U_BOOT_DRIVER(pwm_backlight) = {
  247. .name = "pwm_backlight",
  248. .id = UCLASS_PANEL_BACKLIGHT,
  249. .of_match = pwm_backlight_ids,
  250. .ops = &pwm_backlight_ops,
  251. .of_to_plat = pwm_backlight_of_to_plat,
  252. .probe = pwm_backlight_probe,
  253. .priv_auto = sizeof(struct pwm_backlight_priv),
  254. };