lm3639_bl.c 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437
  1. /*
  2. * Simple driver for Texas Instruments LM3639 Backlight + Flash LED driver chip
  3. * Copyright (C) 2012 Texas Instruments
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License version 2 as
  7. * published by the Free Software Foundation.
  8. *
  9. */
  10. #include <linux/module.h>
  11. #include <linux/slab.h>
  12. #include <linux/i2c.h>
  13. #include <linux/leds.h>
  14. #include <linux/backlight.h>
  15. #include <linux/err.h>
  16. #include <linux/delay.h>
  17. #include <linux/uaccess.h>
  18. #include <linux/interrupt.h>
  19. #include <linux/regmap.h>
  20. #include <linux/platform_data/lm3639_bl.h>
  21. #define REG_DEV_ID 0x00
  22. #define REG_CHECKSUM 0x01
  23. #define REG_BL_CONF_1 0x02
  24. #define REG_BL_CONF_2 0x03
  25. #define REG_BL_CONF_3 0x04
  26. #define REG_BL_CONF_4 0x05
  27. #define REG_FL_CONF_1 0x06
  28. #define REG_FL_CONF_2 0x07
  29. #define REG_FL_CONF_3 0x08
  30. #define REG_IO_CTRL 0x09
  31. #define REG_ENABLE 0x0A
  32. #define REG_FLAG 0x0B
  33. #define REG_MAX REG_FLAG
  34. struct lm3639_chip_data {
  35. struct device *dev;
  36. struct lm3639_platform_data *pdata;
  37. struct backlight_device *bled;
  38. struct led_classdev cdev_flash;
  39. struct led_classdev cdev_torch;
  40. struct regmap *regmap;
  41. unsigned int bled_mode;
  42. unsigned int bled_map;
  43. unsigned int last_flag;
  44. };
  45. /* initialize chip */
  46. static int lm3639_chip_init(struct lm3639_chip_data *pchip)
  47. {
  48. int ret;
  49. unsigned int reg_val;
  50. struct lm3639_platform_data *pdata = pchip->pdata;
  51. /* input pins config. */
  52. ret =
  53. regmap_update_bits(pchip->regmap, REG_BL_CONF_1, 0x08,
  54. pdata->pin_pwm);
  55. if (ret < 0)
  56. goto out;
  57. reg_val = (pdata->pin_pwm & 0x40) | pdata->pin_strobe | pdata->pin_tx;
  58. ret = regmap_update_bits(pchip->regmap, REG_IO_CTRL, 0x7C, reg_val);
  59. if (ret < 0)
  60. goto out;
  61. /* init brightness */
  62. ret = regmap_write(pchip->regmap, REG_BL_CONF_4, pdata->init_brt_led);
  63. if (ret < 0)
  64. goto out;
  65. ret = regmap_write(pchip->regmap, REG_BL_CONF_3, pdata->init_brt_led);
  66. if (ret < 0)
  67. goto out;
  68. /* output pins config. */
  69. if (!pdata->init_brt_led)
  70. reg_val = pdata->fled_pins | pdata->bled_pins;
  71. else
  72. reg_val = pdata->fled_pins | pdata->bled_pins | 0x01;
  73. ret = regmap_update_bits(pchip->regmap, REG_ENABLE, 0x79, reg_val);
  74. if (ret < 0)
  75. goto out;
  76. return ret;
  77. out:
  78. dev_err(pchip->dev, "i2c failed to access register\n");
  79. return ret;
  80. }
  81. /* update and get brightness */
  82. static int lm3639_bled_update_status(struct backlight_device *bl)
  83. {
  84. int ret;
  85. unsigned int reg_val;
  86. struct lm3639_chip_data *pchip = bl_get_data(bl);
  87. struct lm3639_platform_data *pdata = pchip->pdata;
  88. ret = regmap_read(pchip->regmap, REG_FLAG, &reg_val);
  89. if (ret < 0)
  90. goto out;
  91. if (reg_val != 0)
  92. dev_info(pchip->dev, "last flag is 0x%x\n", reg_val);
  93. /* pwm control */
  94. if (pdata->pin_pwm) {
  95. if (pdata->pwm_set_intensity)
  96. pdata->pwm_set_intensity(bl->props.brightness,
  97. pdata->max_brt_led);
  98. else
  99. dev_err(pchip->dev,
  100. "No pwm control func. in plat-data\n");
  101. return bl->props.brightness;
  102. }
  103. /* i2c control and set brigtness */
  104. ret = regmap_write(pchip->regmap, REG_BL_CONF_4, bl->props.brightness);
  105. if (ret < 0)
  106. goto out;
  107. ret = regmap_write(pchip->regmap, REG_BL_CONF_3, bl->props.brightness);
  108. if (ret < 0)
  109. goto out;
  110. if (!bl->props.brightness)
  111. ret = regmap_update_bits(pchip->regmap, REG_ENABLE, 0x01, 0x00);
  112. else
  113. ret = regmap_update_bits(pchip->regmap, REG_ENABLE, 0x01, 0x01);
  114. if (ret < 0)
  115. goto out;
  116. return bl->props.brightness;
  117. out:
  118. dev_err(pchip->dev, "i2c failed to access registers\n");
  119. return bl->props.brightness;
  120. }
  121. static int lm3639_bled_get_brightness(struct backlight_device *bl)
  122. {
  123. int ret;
  124. unsigned int reg_val;
  125. struct lm3639_chip_data *pchip = bl_get_data(bl);
  126. struct lm3639_platform_data *pdata = pchip->pdata;
  127. if (pdata->pin_pwm) {
  128. if (pdata->pwm_get_intensity)
  129. bl->props.brightness = pdata->pwm_get_intensity();
  130. else
  131. dev_err(pchip->dev,
  132. "No pwm control func. in plat-data\n");
  133. return bl->props.brightness;
  134. }
  135. ret = regmap_read(pchip->regmap, REG_BL_CONF_1, &reg_val);
  136. if (ret < 0)
  137. goto out;
  138. if (reg_val & 0x10)
  139. ret = regmap_read(pchip->regmap, REG_BL_CONF_4, &reg_val);
  140. else
  141. ret = regmap_read(pchip->regmap, REG_BL_CONF_3, &reg_val);
  142. if (ret < 0)
  143. goto out;
  144. bl->props.brightness = reg_val;
  145. return bl->props.brightness;
  146. out:
  147. dev_err(pchip->dev, "i2c failed to access register\n");
  148. return bl->props.brightness;
  149. }
  150. static const struct backlight_ops lm3639_bled_ops = {
  151. .options = BL_CORE_SUSPENDRESUME,
  152. .update_status = lm3639_bled_update_status,
  153. .get_brightness = lm3639_bled_get_brightness,
  154. };
  155. /* backlight mapping mode */
  156. static ssize_t lm3639_bled_mode_store(struct device *dev,
  157. struct device_attribute *devAttr,
  158. const char *buf, size_t size)
  159. {
  160. ssize_t ret;
  161. struct lm3639_chip_data *pchip = dev_get_drvdata(dev);
  162. unsigned int state;
  163. ret = kstrtouint(buf, 10, &state);
  164. if (ret)
  165. goto out_input;
  166. if (!state)
  167. ret =
  168. regmap_update_bits(pchip->regmap, REG_BL_CONF_1, 0x10,
  169. 0x00);
  170. else
  171. ret =
  172. regmap_update_bits(pchip->regmap, REG_BL_CONF_1, 0x10,
  173. 0x10);
  174. if (ret < 0)
  175. goto out;
  176. return size;
  177. out:
  178. dev_err(pchip->dev, "%s:i2c access fail to register\n", __func__);
  179. return ret;
  180. out_input:
  181. dev_err(pchip->dev, "%s:input conversion fail\n", __func__);
  182. return ret;
  183. }
  184. static DEVICE_ATTR(bled_mode, S_IWUSR, NULL, lm3639_bled_mode_store);
  185. /* torch */
  186. static void lm3639_torch_brightness_set(struct led_classdev *cdev,
  187. enum led_brightness brightness)
  188. {
  189. int ret;
  190. unsigned int reg_val;
  191. struct lm3639_chip_data *pchip;
  192. pchip = container_of(cdev, struct lm3639_chip_data, cdev_torch);
  193. ret = regmap_read(pchip->regmap, REG_FLAG, &reg_val);
  194. if (ret < 0)
  195. goto out;
  196. if (reg_val != 0)
  197. dev_info(pchip->dev, "last flag is 0x%x\n", reg_val);
  198. /* brightness 0 means off state */
  199. if (!brightness) {
  200. ret = regmap_update_bits(pchip->regmap, REG_ENABLE, 0x06, 0x00);
  201. if (ret < 0)
  202. goto out;
  203. return;
  204. }
  205. ret = regmap_update_bits(pchip->regmap,
  206. REG_FL_CONF_1, 0x70, (brightness - 1) << 4);
  207. if (ret < 0)
  208. goto out;
  209. ret = regmap_update_bits(pchip->regmap, REG_ENABLE, 0x06, 0x02);
  210. if (ret < 0)
  211. goto out;
  212. return;
  213. out:
  214. dev_err(pchip->dev, "i2c failed to access register\n");
  215. return;
  216. }
  217. /* flash */
  218. static void lm3639_flash_brightness_set(struct led_classdev *cdev,
  219. enum led_brightness brightness)
  220. {
  221. int ret;
  222. unsigned int reg_val;
  223. struct lm3639_chip_data *pchip;
  224. pchip = container_of(cdev, struct lm3639_chip_data, cdev_flash);
  225. ret = regmap_read(pchip->regmap, REG_FLAG, &reg_val);
  226. if (ret < 0)
  227. goto out;
  228. if (reg_val != 0)
  229. dev_info(pchip->dev, "last flag is 0x%x\n", reg_val);
  230. /* torch off before flash control */
  231. ret = regmap_update_bits(pchip->regmap, REG_ENABLE, 0x06, 0x00);
  232. if (ret < 0)
  233. goto out;
  234. /* brightness 0 means off state */
  235. if (!brightness)
  236. return;
  237. ret = regmap_update_bits(pchip->regmap,
  238. REG_FL_CONF_1, 0x0F, brightness - 1);
  239. if (ret < 0)
  240. goto out;
  241. ret = regmap_update_bits(pchip->regmap, REG_ENABLE, 0x06, 0x06);
  242. if (ret < 0)
  243. goto out;
  244. return;
  245. out:
  246. dev_err(pchip->dev, "i2c failed to access register\n");
  247. return;
  248. }
  249. static const struct regmap_config lm3639_regmap = {
  250. .reg_bits = 8,
  251. .val_bits = 8,
  252. .max_register = REG_MAX,
  253. };
  254. static int lm3639_probe(struct i2c_client *client,
  255. const struct i2c_device_id *id)
  256. {
  257. int ret;
  258. struct lm3639_chip_data *pchip;
  259. struct lm3639_platform_data *pdata = client->dev.platform_data;
  260. struct backlight_properties props;
  261. if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) {
  262. dev_err(&client->dev, "i2c functionality check fail.\n");
  263. return -EOPNOTSUPP;
  264. }
  265. if (pdata == NULL) {
  266. dev_err(&client->dev, "Needs Platform Data.\n");
  267. return -ENODATA;
  268. }
  269. pchip = devm_kzalloc(&client->dev,
  270. sizeof(struct lm3639_chip_data), GFP_KERNEL);
  271. if (!pchip)
  272. return -ENOMEM;
  273. pchip->pdata = pdata;
  274. pchip->dev = &client->dev;
  275. pchip->regmap = devm_regmap_init_i2c(client, &lm3639_regmap);
  276. if (IS_ERR(pchip->regmap)) {
  277. ret = PTR_ERR(pchip->regmap);
  278. dev_err(&client->dev, "fail : allocate register map: %d\n",
  279. ret);
  280. return ret;
  281. }
  282. i2c_set_clientdata(client, pchip);
  283. /* chip initialize */
  284. ret = lm3639_chip_init(pchip);
  285. if (ret < 0) {
  286. dev_err(&client->dev, "fail : chip init\n");
  287. goto err_out;
  288. }
  289. /* backlight */
  290. props.type = BACKLIGHT_RAW;
  291. props.brightness = pdata->init_brt_led;
  292. props.max_brightness = pdata->max_brt_led;
  293. pchip->bled =
  294. backlight_device_register("lm3639_bled", pchip->dev, pchip,
  295. &lm3639_bled_ops, &props);
  296. if (IS_ERR(pchip->bled)) {
  297. dev_err(&client->dev, "fail : backlight register\n");
  298. ret = -EIO;
  299. goto err_out;
  300. }
  301. ret = device_create_file(&(pchip->bled->dev), &dev_attr_bled_mode);
  302. if (ret < 0) {
  303. dev_err(&client->dev, "failed : add sysfs entries\n");
  304. ret = -EIO;
  305. goto err_bled_mode;
  306. }
  307. /* flash */
  308. pchip->cdev_flash.name = "lm3639_flash";
  309. pchip->cdev_flash.max_brightness = 16;
  310. pchip->cdev_flash.brightness_set = lm3639_flash_brightness_set;
  311. ret = led_classdev_register((struct device *)
  312. &client->dev, &pchip->cdev_flash);
  313. if (ret < 0) {
  314. dev_err(&client->dev, "fail : flash register\n");
  315. ret = -EIO;
  316. goto err_flash;
  317. }
  318. /* torch */
  319. pchip->cdev_torch.name = "lm3639_torch";
  320. pchip->cdev_torch.max_brightness = 8;
  321. pchip->cdev_torch.brightness_set = lm3639_torch_brightness_set;
  322. ret = led_classdev_register((struct device *)
  323. &client->dev, &pchip->cdev_torch);
  324. if (ret < 0) {
  325. dev_err(&client->dev, "fail : torch register\n");
  326. ret = -EIO;
  327. goto err_torch;
  328. }
  329. return 0;
  330. err_torch:
  331. led_classdev_unregister(&pchip->cdev_flash);
  332. err_flash:
  333. device_remove_file(&(pchip->bled->dev), &dev_attr_bled_mode);
  334. err_bled_mode:
  335. backlight_device_unregister(pchip->bled);
  336. err_out:
  337. return ret;
  338. }
  339. static int lm3639_remove(struct i2c_client *client)
  340. {
  341. struct lm3639_chip_data *pchip = i2c_get_clientdata(client);
  342. regmap_write(pchip->regmap, REG_ENABLE, 0x00);
  343. if (&pchip->cdev_torch)
  344. led_classdev_unregister(&pchip->cdev_torch);
  345. if (&pchip->cdev_flash)
  346. led_classdev_unregister(&pchip->cdev_flash);
  347. if (pchip->bled) {
  348. device_remove_file(&(pchip->bled->dev), &dev_attr_bled_mode);
  349. backlight_device_unregister(pchip->bled);
  350. }
  351. return 0;
  352. }
  353. static const struct i2c_device_id lm3639_id[] = {
  354. {LM3639_NAME, 0},
  355. {}
  356. };
  357. MODULE_DEVICE_TABLE(i2c, lm3639_id);
  358. static struct i2c_driver lm3639_i2c_driver = {
  359. .driver = {
  360. .name = LM3639_NAME,
  361. },
  362. .probe = lm3639_probe,
  363. .remove = lm3639_remove,
  364. .id_table = lm3639_id,
  365. };
  366. module_i2c_driver(lm3639_i2c_driver);
  367. MODULE_DESCRIPTION("Texas Instruments Backlight+Flash LED driver for LM3639");
  368. MODULE_AUTHOR("Daniel Jeong <daniel.jeong@ti.com>");
  369. MODULE_AUTHOR("G.Shark Jeong <gshark.jeong@gmail.com>");
  370. MODULE_LICENSE("GPL v2");