qcom-wled.c 44 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /* Copyright (c) 2015, Sony Mobile Communications, AB.
  3. */
  4. #include <linux/delay.h>
  5. #include <linux/interrupt.h>
  6. #include <linux/ktime.h>
  7. #include <linux/kernel.h>
  8. #include <linux/backlight.h>
  9. #include <linux/module.h>
  10. #include <linux/of.h>
  11. #include <linux/of_address.h>
  12. #include <linux/platform_device.h>
  13. #include <linux/regmap.h>
  14. /* From DT binding */
  15. #define WLED_MAX_STRINGS 4
  16. #define MOD_A 0
  17. #define MOD_B 1
  18. #define WLED_DEFAULT_BRIGHTNESS 2048
  19. #define WLED_SOFT_START_DLY_US 10000
  20. #define WLED3_SINK_REG_BRIGHT_MAX 0xFFF
  21. #define WLED5_SINK_REG_BRIGHT_MAX_12B 0xFFF
  22. #define WLED5_SINK_REG_BRIGHT_MAX_15B 0x7FFF
  23. /* WLED3/WLED4 control registers */
  24. #define WLED3_CTRL_REG_FAULT_STATUS 0x08
  25. #define WLED3_CTRL_REG_ILIM_FAULT_BIT BIT(0)
  26. #define WLED3_CTRL_REG_OVP_FAULT_BIT BIT(1)
  27. #define WLED4_CTRL_REG_SC_FAULT_BIT BIT(2)
  28. #define WLED5_CTRL_REG_OVP_PRE_ALARM_BIT BIT(4)
  29. #define WLED3_CTRL_REG_INT_RT_STS 0x10
  30. #define WLED3_CTRL_REG_OVP_FAULT_STATUS BIT(1)
  31. #define WLED3_CTRL_REG_MOD_EN 0x46
  32. #define WLED3_CTRL_REG_MOD_EN_MASK BIT(7)
  33. #define WLED3_CTRL_REG_MOD_EN_SHIFT 7
  34. #define WLED3_CTRL_REG_FEEDBACK_CONTROL 0x48
  35. #define WLED3_CTRL_REG_FREQ 0x4c
  36. #define WLED3_CTRL_REG_FREQ_MASK GENMASK(3, 0)
  37. #define WLED3_CTRL_REG_OVP 0x4d
  38. #define WLED3_CTRL_REG_OVP_MASK GENMASK(1, 0)
  39. #define WLED5_CTRL_REG_OVP_MASK GENMASK(3, 0)
  40. #define WLED3_CTRL_REG_ILIMIT 0x4e
  41. #define WLED3_CTRL_REG_ILIMIT_MASK GENMASK(2, 0)
  42. /* WLED3/WLED4 sink registers */
  43. #define WLED3_SINK_REG_SYNC 0x47
  44. #define WLED3_SINK_REG_SYNC_CLEAR 0x00
  45. #define WLED3_SINK_REG_CURR_SINK 0x4f
  46. #define WLED3_SINK_REG_CURR_SINK_MASK GENMASK(7, 5)
  47. #define WLED3_SINK_REG_CURR_SINK_SHFT 5
  48. /* WLED3 specific per-'string' registers below */
  49. #define WLED3_SINK_REG_BRIGHT(n) (0x40 + n)
  50. #define WLED3_SINK_REG_STR_MOD_EN(n) (0x60 + (n * 0x10))
  51. #define WLED3_SINK_REG_STR_MOD_MASK BIT(7)
  52. #define WLED3_SINK_REG_STR_FULL_SCALE_CURR(n) (0x62 + (n * 0x10))
  53. #define WLED3_SINK_REG_STR_FULL_SCALE_CURR_MASK GENMASK(4, 0)
  54. #define WLED3_SINK_REG_STR_MOD_SRC(n) (0x63 + (n * 0x10))
  55. #define WLED3_SINK_REG_STR_MOD_SRC_MASK BIT(0)
  56. #define WLED3_SINK_REG_STR_MOD_SRC_INT 0x00
  57. #define WLED3_SINK_REG_STR_MOD_SRC_EXT 0x01
  58. #define WLED3_SINK_REG_STR_CABC(n) (0x66 + (n * 0x10))
  59. #define WLED3_SINK_REG_STR_CABC_MASK BIT(7)
  60. /* WLED4 specific control registers */
  61. #define WLED4_CTRL_REG_SHORT_PROTECT 0x5e
  62. #define WLED4_CTRL_REG_SHORT_EN_MASK BIT(7)
  63. #define WLED4_CTRL_REG_SEC_ACCESS 0xd0
  64. #define WLED4_CTRL_REG_SEC_UNLOCK 0xa5
  65. #define WLED4_CTRL_REG_TEST1 0xe2
  66. #define WLED4_CTRL_REG_TEST1_EXT_FET_DTEST2 0x09
  67. /* WLED4 specific sink registers */
  68. #define WLED4_SINK_REG_CURR_SINK 0x46
  69. #define WLED4_SINK_REG_CURR_SINK_MASK GENMASK(7, 4)
  70. #define WLED4_SINK_REG_CURR_SINK_SHFT 4
  71. /* WLED4 specific per-'string' registers below */
  72. #define WLED4_SINK_REG_STR_MOD_EN(n) (0x50 + (n * 0x10))
  73. #define WLED4_SINK_REG_STR_MOD_MASK BIT(7)
  74. #define WLED4_SINK_REG_STR_FULL_SCALE_CURR(n) (0x52 + (n * 0x10))
  75. #define WLED4_SINK_REG_STR_FULL_SCALE_CURR_MASK GENMASK(3, 0)
  76. #define WLED4_SINK_REG_STR_MOD_SRC(n) (0x53 + (n * 0x10))
  77. #define WLED4_SINK_REG_STR_MOD_SRC_MASK BIT(0)
  78. #define WLED4_SINK_REG_STR_MOD_SRC_INT 0x00
  79. #define WLED4_SINK_REG_STR_MOD_SRC_EXT 0x01
  80. #define WLED4_SINK_REG_STR_CABC(n) (0x56 + (n * 0x10))
  81. #define WLED4_SINK_REG_STR_CABC_MASK BIT(7)
  82. #define WLED4_SINK_REG_BRIGHT(n) (0x57 + (n * 0x10))
  83. /* WLED5 specific control registers */
  84. #define WLED5_CTRL_REG_OVP_INT_CTL 0x5f
  85. #define WLED5_CTRL_REG_OVP_INT_TIMER_MASK GENMASK(2, 0)
  86. /* WLED5 specific sink registers */
  87. #define WLED5_SINK_REG_MOD_A_EN 0x50
  88. #define WLED5_SINK_REG_MOD_B_EN 0x60
  89. #define WLED5_SINK_REG_MOD_EN_MASK BIT(7)
  90. #define WLED5_SINK_REG_MOD_A_SRC_SEL 0x51
  91. #define WLED5_SINK_REG_MOD_B_SRC_SEL 0x61
  92. #define WLED5_SINK_REG_MOD_SRC_SEL_HIGH 0
  93. #define WLED5_SINK_REG_MOD_SRC_SEL_EXT 0x03
  94. #define WLED5_SINK_REG_MOD_SRC_SEL_MASK GENMASK(1, 0)
  95. #define WLED5_SINK_REG_MOD_A_BRIGHTNESS_WIDTH_SEL 0x52
  96. #define WLED5_SINK_REG_MOD_B_BRIGHTNESS_WIDTH_SEL 0x62
  97. #define WLED5_SINK_REG_BRIGHTNESS_WIDTH_12B 0
  98. #define WLED5_SINK_REG_BRIGHTNESS_WIDTH_15B 1
  99. #define WLED5_SINK_REG_MOD_A_BRIGHTNESS_LSB 0x53
  100. #define WLED5_SINK_REG_MOD_A_BRIGHTNESS_MSB 0x54
  101. #define WLED5_SINK_REG_MOD_B_BRIGHTNESS_LSB 0x63
  102. #define WLED5_SINK_REG_MOD_B_BRIGHTNESS_MSB 0x64
  103. #define WLED5_SINK_REG_MOD_SYNC_BIT 0x65
  104. #define WLED5_SINK_REG_SYNC_MOD_A_BIT BIT(0)
  105. #define WLED5_SINK_REG_SYNC_MOD_B_BIT BIT(1)
  106. #define WLED5_SINK_REG_SYNC_MASK GENMASK(1, 0)
  107. /* WLED5 specific per-'string' registers below */
  108. #define WLED5_SINK_REG_STR_FULL_SCALE_CURR(n) (0x72 + (n * 0x10))
  109. #define WLED5_SINK_REG_STR_SRC_SEL(n) (0x73 + (n * 0x10))
  110. #define WLED5_SINK_REG_SRC_SEL_MOD_A 0
  111. #define WLED5_SINK_REG_SRC_SEL_MOD_B 1
  112. #define WLED5_SINK_REG_SRC_SEL_MASK GENMASK(1, 0)
  113. struct wled_var_cfg {
  114. const u32 *values;
  115. u32 (*fn)(u32);
  116. int size;
  117. };
  118. struct wled_u32_opts {
  119. const char *name;
  120. u32 *val_ptr;
  121. const struct wled_var_cfg *cfg;
  122. };
  123. struct wled_bool_opts {
  124. const char *name;
  125. bool *val_ptr;
  126. };
  127. struct wled_config {
  128. u32 boost_i_limit;
  129. u32 ovp;
  130. u32 switch_freq;
  131. u32 num_strings;
  132. u32 string_i_limit;
  133. u32 enabled_strings[WLED_MAX_STRINGS];
  134. u32 mod_sel;
  135. u32 cabc_sel;
  136. bool cs_out_en;
  137. bool ext_gen;
  138. bool cabc;
  139. bool external_pfet;
  140. bool auto_detection_enabled;
  141. };
  142. struct wled {
  143. const char *name;
  144. struct device *dev;
  145. struct regmap *regmap;
  146. struct mutex lock; /* Lock to avoid race from thread irq handler */
  147. ktime_t last_short_event;
  148. ktime_t start_ovp_fault_time;
  149. u16 ctrl_addr;
  150. u16 sink_addr;
  151. u16 max_string_count;
  152. u16 auto_detection_ovp_count;
  153. u32 brightness;
  154. u32 max_brightness;
  155. u32 short_count;
  156. u32 auto_detect_count;
  157. u32 version;
  158. bool disabled_by_short;
  159. bool has_short_detect;
  160. bool cabc_disabled;
  161. int short_irq;
  162. int ovp_irq;
  163. struct wled_config cfg;
  164. struct delayed_work ovp_work;
  165. /* Configures the brightness. Applicable for wled3, wled4 and wled5 */
  166. int (*wled_set_brightness)(struct wled *wled, u16 brightness);
  167. /* Configures the cabc register. Applicable for wled4 and wled5 */
  168. int (*wled_cabc_config)(struct wled *wled, bool enable);
  169. /*
  170. * Toggles the sync bit for the brightness update to take place.
  171. * Applicable for WLED3, WLED4 and WLED5.
  172. */
  173. int (*wled_sync_toggle)(struct wled *wled);
  174. /*
  175. * Time to wait before checking the OVP status after wled module enable.
  176. * Applicable for WLED4 and WLED5.
  177. */
  178. int (*wled_ovp_delay)(struct wled *wled);
  179. /*
  180. * Determines if the auto string detection is required.
  181. * Applicable for WLED4 and WLED5
  182. */
  183. bool (*wled_auto_detection_required)(struct wled *wled);
  184. };
  185. static int wled3_set_brightness(struct wled *wled, u16 brightness)
  186. {
  187. int rc, i;
  188. __le16 v;
  189. v = cpu_to_le16(brightness & WLED3_SINK_REG_BRIGHT_MAX);
  190. for (i = 0; i < wled->cfg.num_strings; ++i) {
  191. rc = regmap_bulk_write(wled->regmap, wled->ctrl_addr +
  192. WLED3_SINK_REG_BRIGHT(wled->cfg.enabled_strings[i]),
  193. &v, sizeof(v));
  194. if (rc < 0)
  195. return rc;
  196. }
  197. return 0;
  198. }
  199. static int wled4_set_brightness(struct wled *wled, u16 brightness)
  200. {
  201. int rc, i;
  202. u16 low_limit = wled->max_brightness * 4 / 1000;
  203. __le16 v;
  204. /* WLED4's lower limit of operation is 0.4% */
  205. if (brightness > 0 && brightness < low_limit)
  206. brightness = low_limit;
  207. v = cpu_to_le16(brightness & WLED3_SINK_REG_BRIGHT_MAX);
  208. for (i = 0; i < wled->cfg.num_strings; ++i) {
  209. rc = regmap_bulk_write(wled->regmap, wled->sink_addr +
  210. WLED4_SINK_REG_BRIGHT(wled->cfg.enabled_strings[i]),
  211. &v, sizeof(v));
  212. if (rc < 0)
  213. return rc;
  214. }
  215. return 0;
  216. }
  217. static int wled5_set_brightness(struct wled *wled, u16 brightness)
  218. {
  219. int rc, offset;
  220. u16 low_limit = wled->max_brightness * 1 / 1000;
  221. __le16 v;
  222. /* WLED5's lower limit is 0.1% */
  223. if (brightness < low_limit)
  224. brightness = low_limit;
  225. v = cpu_to_le16(brightness & WLED5_SINK_REG_BRIGHT_MAX_15B);
  226. offset = (wled->cfg.mod_sel == MOD_A) ?
  227. WLED5_SINK_REG_MOD_A_BRIGHTNESS_LSB :
  228. WLED5_SINK_REG_MOD_B_BRIGHTNESS_LSB;
  229. rc = regmap_bulk_write(wled->regmap, wled->sink_addr + offset,
  230. &v, sizeof(v));
  231. return rc;
  232. }
  233. static void wled_ovp_work(struct work_struct *work)
  234. {
  235. struct wled *wled = container_of(work,
  236. struct wled, ovp_work.work);
  237. enable_irq(wled->ovp_irq);
  238. }
  239. static int wled_module_enable(struct wled *wled, int val)
  240. {
  241. int rc;
  242. if (wled->disabled_by_short)
  243. return -ENXIO;
  244. rc = regmap_update_bits(wled->regmap, wled->ctrl_addr +
  245. WLED3_CTRL_REG_MOD_EN,
  246. WLED3_CTRL_REG_MOD_EN_MASK,
  247. val << WLED3_CTRL_REG_MOD_EN_SHIFT);
  248. if (rc < 0)
  249. return rc;
  250. if (wled->ovp_irq > 0) {
  251. if (val) {
  252. /*
  253. * The hardware generates a storm of spurious OVP
  254. * interrupts during soft start operations. So defer
  255. * enabling the IRQ for 10ms to ensure that the
  256. * soft start is complete.
  257. */
  258. schedule_delayed_work(&wled->ovp_work, HZ / 100);
  259. } else {
  260. if (!cancel_delayed_work_sync(&wled->ovp_work))
  261. disable_irq(wled->ovp_irq);
  262. }
  263. }
  264. return 0;
  265. }
  266. static int wled3_sync_toggle(struct wled *wled)
  267. {
  268. int rc;
  269. unsigned int mask = GENMASK(wled->max_string_count - 1, 0);
  270. rc = regmap_update_bits(wled->regmap,
  271. wled->sink_addr + WLED3_SINK_REG_SYNC,
  272. mask, WLED3_SINK_REG_SYNC_CLEAR);
  273. if (rc < 0)
  274. return rc;
  275. rc = regmap_update_bits(wled->regmap,
  276. wled->sink_addr + WLED3_SINK_REG_SYNC,
  277. mask, mask);
  278. return rc;
  279. }
  280. static int wled5_mod_sync_toggle(struct wled *wled)
  281. {
  282. int rc;
  283. u8 val;
  284. rc = regmap_update_bits(wled->regmap,
  285. wled->sink_addr + WLED5_SINK_REG_MOD_SYNC_BIT,
  286. WLED5_SINK_REG_SYNC_MASK, 0);
  287. if (rc < 0)
  288. return rc;
  289. val = (wled->cfg.mod_sel == MOD_A) ? WLED5_SINK_REG_SYNC_MOD_A_BIT :
  290. WLED5_SINK_REG_SYNC_MOD_B_BIT;
  291. return regmap_update_bits(wled->regmap,
  292. wled->sink_addr + WLED5_SINK_REG_MOD_SYNC_BIT,
  293. WLED5_SINK_REG_SYNC_MASK, val);
  294. }
  295. static int wled_ovp_fault_status(struct wled *wled, bool *fault_set)
  296. {
  297. int rc;
  298. u32 int_rt_sts, fault_sts;
  299. *fault_set = false;
  300. rc = regmap_read(wled->regmap,
  301. wled->ctrl_addr + WLED3_CTRL_REG_INT_RT_STS,
  302. &int_rt_sts);
  303. if (rc < 0) {
  304. dev_err(wled->dev, "Failed to read INT_RT_STS rc=%d\n", rc);
  305. return rc;
  306. }
  307. rc = regmap_read(wled->regmap,
  308. wled->ctrl_addr + WLED3_CTRL_REG_FAULT_STATUS,
  309. &fault_sts);
  310. if (rc < 0) {
  311. dev_err(wled->dev, "Failed to read FAULT_STATUS rc=%d\n", rc);
  312. return rc;
  313. }
  314. if (int_rt_sts & WLED3_CTRL_REG_OVP_FAULT_STATUS)
  315. *fault_set = true;
  316. if (wled->version == 4 && (fault_sts & WLED3_CTRL_REG_OVP_FAULT_BIT))
  317. *fault_set = true;
  318. if (wled->version == 5 && (fault_sts & (WLED3_CTRL_REG_OVP_FAULT_BIT |
  319. WLED5_CTRL_REG_OVP_PRE_ALARM_BIT)))
  320. *fault_set = true;
  321. if (*fault_set)
  322. dev_dbg(wled->dev, "WLED OVP fault detected, int_rt_sts=0x%x fault_sts=0x%x\n",
  323. int_rt_sts, fault_sts);
  324. return rc;
  325. }
  326. static int wled4_ovp_delay(struct wled *wled)
  327. {
  328. return WLED_SOFT_START_DLY_US;
  329. }
  330. static int wled5_ovp_delay(struct wled *wled)
  331. {
  332. int rc, delay_us;
  333. u32 val;
  334. u8 ovp_timer_ms[8] = {1, 2, 4, 8, 12, 16, 20, 24};
  335. /* For WLED5, get the delay based on OVP timer */
  336. rc = regmap_read(wled->regmap, wled->ctrl_addr +
  337. WLED5_CTRL_REG_OVP_INT_CTL, &val);
  338. if (rc < 0)
  339. delay_us =
  340. ovp_timer_ms[val & WLED5_CTRL_REG_OVP_INT_TIMER_MASK] * 1000;
  341. else
  342. delay_us = 2 * WLED_SOFT_START_DLY_US;
  343. dev_dbg(wled->dev, "delay_time_us: %d\n", delay_us);
  344. return delay_us;
  345. }
  346. static int wled_update_status(struct backlight_device *bl)
  347. {
  348. struct wled *wled = bl_get_data(bl);
  349. u16 brightness = backlight_get_brightness(bl);
  350. int rc = 0;
  351. mutex_lock(&wled->lock);
  352. if (brightness) {
  353. rc = wled->wled_set_brightness(wled, brightness);
  354. if (rc < 0) {
  355. dev_err(wled->dev, "wled failed to set brightness rc:%d\n",
  356. rc);
  357. goto unlock_mutex;
  358. }
  359. if (wled->version < 5) {
  360. rc = wled->wled_sync_toggle(wled);
  361. if (rc < 0) {
  362. dev_err(wled->dev, "wled sync failed rc:%d\n", rc);
  363. goto unlock_mutex;
  364. }
  365. } else {
  366. /*
  367. * For WLED5 toggling the MOD_SYNC_BIT updates the
  368. * brightness
  369. */
  370. rc = wled5_mod_sync_toggle(wled);
  371. if (rc < 0) {
  372. dev_err(wled->dev, "wled mod sync failed rc:%d\n",
  373. rc);
  374. goto unlock_mutex;
  375. }
  376. }
  377. }
  378. if (!!brightness != !!wled->brightness) {
  379. rc = wled_module_enable(wled, !!brightness);
  380. if (rc < 0) {
  381. dev_err(wled->dev, "wled enable failed rc:%d\n", rc);
  382. goto unlock_mutex;
  383. }
  384. }
  385. wled->brightness = brightness;
  386. unlock_mutex:
  387. mutex_unlock(&wled->lock);
  388. return rc;
  389. }
  390. static int wled4_cabc_config(struct wled *wled, bool enable)
  391. {
  392. int i, j, rc;
  393. u8 val;
  394. for (i = 0; i < wled->cfg.num_strings; i++) {
  395. j = wled->cfg.enabled_strings[i];
  396. val = enable ? WLED4_SINK_REG_STR_CABC_MASK : 0;
  397. rc = regmap_update_bits(wled->regmap, wled->sink_addr +
  398. WLED4_SINK_REG_STR_CABC(j),
  399. WLED4_SINK_REG_STR_CABC_MASK, val);
  400. if (rc < 0)
  401. return rc;
  402. }
  403. return 0;
  404. }
  405. static int wled5_cabc_config(struct wled *wled, bool enable)
  406. {
  407. int rc, offset;
  408. u8 reg;
  409. if (wled->cabc_disabled)
  410. return 0;
  411. reg = enable ? wled->cfg.cabc_sel : 0;
  412. offset = (wled->cfg.mod_sel == MOD_A) ? WLED5_SINK_REG_MOD_A_SRC_SEL :
  413. WLED5_SINK_REG_MOD_B_SRC_SEL;
  414. rc = regmap_update_bits(wled->regmap, wled->sink_addr + offset,
  415. WLED5_SINK_REG_MOD_SRC_SEL_MASK, reg);
  416. if (rc < 0) {
  417. pr_err("Error in configuring CABC rc=%d\n", rc);
  418. return rc;
  419. }
  420. if (!wled->cfg.cabc_sel)
  421. wled->cabc_disabled = true;
  422. return 0;
  423. }
  424. #define WLED_SHORT_DLY_MS 20
  425. #define WLED_SHORT_CNT_MAX 5
  426. #define WLED_SHORT_RESET_CNT_DLY_US USEC_PER_SEC
  427. static irqreturn_t wled_short_irq_handler(int irq, void *_wled)
  428. {
  429. struct wled *wled = _wled;
  430. int rc;
  431. s64 elapsed_time;
  432. wled->short_count++;
  433. mutex_lock(&wled->lock);
  434. rc = wled_module_enable(wled, false);
  435. if (rc < 0) {
  436. dev_err(wled->dev, "wled disable failed rc:%d\n", rc);
  437. goto unlock_mutex;
  438. }
  439. elapsed_time = ktime_us_delta(ktime_get(),
  440. wled->last_short_event);
  441. if (elapsed_time > WLED_SHORT_RESET_CNT_DLY_US)
  442. wled->short_count = 1;
  443. if (wled->short_count > WLED_SHORT_CNT_MAX) {
  444. dev_err(wled->dev, "Short triggered %d times, disabling WLED forever!\n",
  445. wled->short_count);
  446. wled->disabled_by_short = true;
  447. goto unlock_mutex;
  448. }
  449. wled->last_short_event = ktime_get();
  450. msleep(WLED_SHORT_DLY_MS);
  451. rc = wled_module_enable(wled, true);
  452. if (rc < 0)
  453. dev_err(wled->dev, "wled enable failed rc:%d\n", rc);
  454. unlock_mutex:
  455. mutex_unlock(&wled->lock);
  456. return IRQ_HANDLED;
  457. }
  458. #define AUTO_DETECT_BRIGHTNESS 200
  459. static void wled_auto_string_detection(struct wled *wled)
  460. {
  461. int rc = 0, i, j, delay_time_us;
  462. u32 sink_config = 0;
  463. u8 sink_test = 0, sink_valid = 0, val;
  464. bool fault_set;
  465. /* Read configured sink configuration */
  466. rc = regmap_read(wled->regmap, wled->sink_addr +
  467. WLED4_SINK_REG_CURR_SINK, &sink_config);
  468. if (rc < 0) {
  469. dev_err(wled->dev, "Failed to read SINK configuration rc=%d\n",
  470. rc);
  471. goto failed_detect;
  472. }
  473. /* Disable the module before starting detection */
  474. rc = regmap_update_bits(wled->regmap,
  475. wled->ctrl_addr + WLED3_CTRL_REG_MOD_EN,
  476. WLED3_CTRL_REG_MOD_EN_MASK, 0);
  477. if (rc < 0) {
  478. dev_err(wled->dev, "Failed to disable WLED module rc=%d\n", rc);
  479. goto failed_detect;
  480. }
  481. /* Set low brightness across all sinks */
  482. rc = wled4_set_brightness(wled, AUTO_DETECT_BRIGHTNESS);
  483. if (rc < 0) {
  484. dev_err(wled->dev, "Failed to set brightness for auto detection rc=%d\n",
  485. rc);
  486. goto failed_detect;
  487. }
  488. if (wled->cfg.cabc) {
  489. rc = wled->wled_cabc_config(wled, false);
  490. if (rc < 0)
  491. goto failed_detect;
  492. }
  493. /* Disable all sinks */
  494. rc = regmap_write(wled->regmap,
  495. wled->sink_addr + WLED4_SINK_REG_CURR_SINK, 0);
  496. if (rc < 0) {
  497. dev_err(wled->dev, "Failed to disable all sinks rc=%d\n", rc);
  498. goto failed_detect;
  499. }
  500. /* Iterate through the strings one by one */
  501. for (i = 0; i < wled->cfg.num_strings; i++) {
  502. j = wled->cfg.enabled_strings[i];
  503. sink_test = BIT((WLED4_SINK_REG_CURR_SINK_SHFT + j));
  504. /* Enable feedback control */
  505. rc = regmap_write(wled->regmap, wled->ctrl_addr +
  506. WLED3_CTRL_REG_FEEDBACK_CONTROL, j + 1);
  507. if (rc < 0) {
  508. dev_err(wled->dev, "Failed to enable feedback for SINK %d rc = %d\n",
  509. j + 1, rc);
  510. goto failed_detect;
  511. }
  512. /* Enable the sink */
  513. rc = regmap_write(wled->regmap, wled->sink_addr +
  514. WLED4_SINK_REG_CURR_SINK, sink_test);
  515. if (rc < 0) {
  516. dev_err(wled->dev, "Failed to configure SINK %d rc=%d\n",
  517. j + 1, rc);
  518. goto failed_detect;
  519. }
  520. /* Enable the module */
  521. rc = regmap_update_bits(wled->regmap, wled->ctrl_addr +
  522. WLED3_CTRL_REG_MOD_EN,
  523. WLED3_CTRL_REG_MOD_EN_MASK,
  524. WLED3_CTRL_REG_MOD_EN_MASK);
  525. if (rc < 0) {
  526. dev_err(wled->dev, "Failed to enable WLED module rc=%d\n",
  527. rc);
  528. goto failed_detect;
  529. }
  530. delay_time_us = wled->wled_ovp_delay(wled);
  531. usleep_range(delay_time_us, delay_time_us + 1000);
  532. rc = wled_ovp_fault_status(wled, &fault_set);
  533. if (rc < 0) {
  534. dev_err(wled->dev, "Error in getting OVP fault_sts, rc=%d\n",
  535. rc);
  536. goto failed_detect;
  537. }
  538. if (fault_set)
  539. dev_dbg(wled->dev, "WLED OVP fault detected with SINK %d\n",
  540. j + 1);
  541. else
  542. sink_valid |= sink_test;
  543. /* Disable the module */
  544. rc = regmap_update_bits(wled->regmap,
  545. wled->ctrl_addr + WLED3_CTRL_REG_MOD_EN,
  546. WLED3_CTRL_REG_MOD_EN_MASK, 0);
  547. if (rc < 0) {
  548. dev_err(wled->dev, "Failed to disable WLED module rc=%d\n",
  549. rc);
  550. goto failed_detect;
  551. }
  552. }
  553. if (!sink_valid) {
  554. dev_err(wled->dev, "No valid WLED sinks found\n");
  555. wled->disabled_by_short = true;
  556. goto failed_detect;
  557. }
  558. if (sink_valid != sink_config) {
  559. dev_warn(wled->dev, "%x is not a valid sink configuration - using %x instead\n",
  560. sink_config, sink_valid);
  561. sink_config = sink_valid;
  562. }
  563. /* Write the new sink configuration */
  564. rc = regmap_write(wled->regmap,
  565. wled->sink_addr + WLED4_SINK_REG_CURR_SINK,
  566. sink_config);
  567. if (rc < 0) {
  568. dev_err(wled->dev, "Failed to reconfigure the default sink rc=%d\n",
  569. rc);
  570. goto failed_detect;
  571. }
  572. /* Enable valid sinks */
  573. if (wled->version == 4) {
  574. for (i = 0; i < wled->cfg.num_strings; i++) {
  575. j = wled->cfg.enabled_strings[i];
  576. if (sink_config &
  577. BIT(WLED4_SINK_REG_CURR_SINK_SHFT + j))
  578. val = WLED4_SINK_REG_STR_MOD_MASK;
  579. else
  580. /* Disable modulator_en for unused sink */
  581. val = 0;
  582. rc = regmap_write(wled->regmap, wled->sink_addr +
  583. WLED4_SINK_REG_STR_MOD_EN(j), val);
  584. if (rc < 0) {
  585. dev_err(wled->dev, "Failed to configure MODULATOR_EN rc=%d\n",
  586. rc);
  587. goto failed_detect;
  588. }
  589. }
  590. }
  591. /* Enable CABC */
  592. rc = wled->wled_cabc_config(wled, true);
  593. if (rc < 0)
  594. goto failed_detect;
  595. /* Restore the feedback setting */
  596. rc = regmap_write(wled->regmap,
  597. wled->ctrl_addr + WLED3_CTRL_REG_FEEDBACK_CONTROL, 0);
  598. if (rc < 0) {
  599. dev_err(wled->dev, "Failed to restore feedback setting rc=%d\n",
  600. rc);
  601. goto failed_detect;
  602. }
  603. /* Restore brightness */
  604. rc = wled4_set_brightness(wled, wled->brightness);
  605. if (rc < 0) {
  606. dev_err(wled->dev, "Failed to set brightness after auto detection rc=%d\n",
  607. rc);
  608. goto failed_detect;
  609. }
  610. rc = regmap_update_bits(wled->regmap,
  611. wled->ctrl_addr + WLED3_CTRL_REG_MOD_EN,
  612. WLED3_CTRL_REG_MOD_EN_MASK,
  613. WLED3_CTRL_REG_MOD_EN_MASK);
  614. if (rc < 0) {
  615. dev_err(wled->dev, "Failed to enable WLED module rc=%d\n", rc);
  616. goto failed_detect;
  617. }
  618. failed_detect:
  619. return;
  620. }
  621. #define WLED_AUTO_DETECT_OVP_COUNT 5
  622. #define WLED_AUTO_DETECT_CNT_DLY_US USEC_PER_SEC
  623. static bool wled4_auto_detection_required(struct wled *wled)
  624. {
  625. s64 elapsed_time_us;
  626. if (!wled->cfg.auto_detection_enabled)
  627. return false;
  628. /*
  629. * Check if the OVP fault was an occasional one
  630. * or if it's firing continuously, the latter qualifies
  631. * for an auto-detection check.
  632. */
  633. if (!wled->auto_detection_ovp_count) {
  634. wled->start_ovp_fault_time = ktime_get();
  635. wled->auto_detection_ovp_count++;
  636. } else {
  637. elapsed_time_us = ktime_us_delta(ktime_get(),
  638. wled->start_ovp_fault_time);
  639. if (elapsed_time_us > WLED_AUTO_DETECT_CNT_DLY_US)
  640. wled->auto_detection_ovp_count = 0;
  641. else
  642. wled->auto_detection_ovp_count++;
  643. if (wled->auto_detection_ovp_count >=
  644. WLED_AUTO_DETECT_OVP_COUNT) {
  645. wled->auto_detection_ovp_count = 0;
  646. return true;
  647. }
  648. }
  649. return false;
  650. }
  651. static bool wled5_auto_detection_required(struct wled *wled)
  652. {
  653. if (!wled->cfg.auto_detection_enabled)
  654. return false;
  655. /*
  656. * Unlike WLED4, WLED5 has OVP fault density interrupt configuration
  657. * i.e. to count the number of OVP alarms for a certain duration before
  658. * triggering OVP fault interrupt. By default, number of OVP fault
  659. * events counted before an interrupt is fired is 32 and the time
  660. * interval is 12 ms. If we see one OVP fault interrupt, then that
  661. * should qualify for a real OVP fault condition to run auto detection
  662. * algorithm.
  663. */
  664. return true;
  665. }
  666. static int wled_auto_detection_at_init(struct wled *wled)
  667. {
  668. int rc;
  669. bool fault_set;
  670. if (!wled->cfg.auto_detection_enabled)
  671. return 0;
  672. rc = wled_ovp_fault_status(wled, &fault_set);
  673. if (rc < 0) {
  674. dev_err(wled->dev, "Error in getting OVP fault_sts, rc=%d\n",
  675. rc);
  676. return rc;
  677. }
  678. if (fault_set) {
  679. mutex_lock(&wled->lock);
  680. wled_auto_string_detection(wled);
  681. mutex_unlock(&wled->lock);
  682. }
  683. return rc;
  684. }
  685. static irqreturn_t wled_ovp_irq_handler(int irq, void *_wled)
  686. {
  687. struct wled *wled = _wled;
  688. int rc;
  689. u32 int_sts, fault_sts;
  690. rc = regmap_read(wled->regmap,
  691. wled->ctrl_addr + WLED3_CTRL_REG_INT_RT_STS, &int_sts);
  692. if (rc < 0) {
  693. dev_err(wled->dev, "Error in reading WLED3_INT_RT_STS rc=%d\n",
  694. rc);
  695. return IRQ_HANDLED;
  696. }
  697. rc = regmap_read(wled->regmap, wled->ctrl_addr +
  698. WLED3_CTRL_REG_FAULT_STATUS, &fault_sts);
  699. if (rc < 0) {
  700. dev_err(wled->dev, "Error in reading WLED_FAULT_STATUS rc=%d\n",
  701. rc);
  702. return IRQ_HANDLED;
  703. }
  704. if (fault_sts & (WLED3_CTRL_REG_OVP_FAULT_BIT |
  705. WLED3_CTRL_REG_ILIM_FAULT_BIT))
  706. dev_dbg(wled->dev, "WLED OVP fault detected, int_sts=%x fault_sts= %x\n",
  707. int_sts, fault_sts);
  708. if (fault_sts & WLED3_CTRL_REG_OVP_FAULT_BIT) {
  709. if (wled->wled_auto_detection_required(wled)) {
  710. mutex_lock(&wled->lock);
  711. wled_auto_string_detection(wled);
  712. mutex_unlock(&wled->lock);
  713. }
  714. }
  715. return IRQ_HANDLED;
  716. }
  717. static int wled3_setup(struct wled *wled)
  718. {
  719. u16 addr;
  720. u8 sink_en = 0;
  721. int rc, i, j;
  722. rc = regmap_update_bits(wled->regmap,
  723. wled->ctrl_addr + WLED3_CTRL_REG_OVP,
  724. WLED3_CTRL_REG_OVP_MASK, wled->cfg.ovp);
  725. if (rc)
  726. return rc;
  727. rc = regmap_update_bits(wled->regmap,
  728. wled->ctrl_addr + WLED3_CTRL_REG_ILIMIT,
  729. WLED3_CTRL_REG_ILIMIT_MASK,
  730. wled->cfg.boost_i_limit);
  731. if (rc)
  732. return rc;
  733. rc = regmap_update_bits(wled->regmap,
  734. wled->ctrl_addr + WLED3_CTRL_REG_FREQ,
  735. WLED3_CTRL_REG_FREQ_MASK,
  736. wled->cfg.switch_freq);
  737. if (rc)
  738. return rc;
  739. for (i = 0; i < wled->cfg.num_strings; ++i) {
  740. j = wled->cfg.enabled_strings[i];
  741. addr = wled->ctrl_addr + WLED3_SINK_REG_STR_MOD_EN(j);
  742. rc = regmap_update_bits(wled->regmap, addr,
  743. WLED3_SINK_REG_STR_MOD_MASK,
  744. WLED3_SINK_REG_STR_MOD_MASK);
  745. if (rc)
  746. return rc;
  747. if (wled->cfg.ext_gen) {
  748. addr = wled->ctrl_addr + WLED3_SINK_REG_STR_MOD_SRC(j);
  749. rc = regmap_update_bits(wled->regmap, addr,
  750. WLED3_SINK_REG_STR_MOD_SRC_MASK,
  751. WLED3_SINK_REG_STR_MOD_SRC_EXT);
  752. if (rc)
  753. return rc;
  754. }
  755. addr = wled->ctrl_addr + WLED3_SINK_REG_STR_FULL_SCALE_CURR(j);
  756. rc = regmap_update_bits(wled->regmap, addr,
  757. WLED3_SINK_REG_STR_FULL_SCALE_CURR_MASK,
  758. wled->cfg.string_i_limit);
  759. if (rc)
  760. return rc;
  761. addr = wled->ctrl_addr + WLED3_SINK_REG_STR_CABC(j);
  762. rc = regmap_update_bits(wled->regmap, addr,
  763. WLED3_SINK_REG_STR_CABC_MASK,
  764. wled->cfg.cabc ?
  765. WLED3_SINK_REG_STR_CABC_MASK : 0);
  766. if (rc)
  767. return rc;
  768. sink_en |= BIT(j + WLED3_SINK_REG_CURR_SINK_SHFT);
  769. }
  770. rc = regmap_update_bits(wled->regmap,
  771. wled->ctrl_addr + WLED3_SINK_REG_CURR_SINK,
  772. WLED3_SINK_REG_CURR_SINK_MASK, sink_en);
  773. if (rc)
  774. return rc;
  775. return 0;
  776. }
  777. static const struct wled_config wled3_config_defaults = {
  778. .boost_i_limit = 3,
  779. .string_i_limit = 20,
  780. .ovp = 2,
  781. .num_strings = 3,
  782. .switch_freq = 5,
  783. .cs_out_en = false,
  784. .ext_gen = false,
  785. .cabc = false,
  786. .enabled_strings = {0, 1, 2},
  787. };
  788. static int wled4_setup(struct wled *wled)
  789. {
  790. int rc, temp, i, j;
  791. u16 addr;
  792. u8 sink_en = 0;
  793. u32 sink_cfg;
  794. rc = regmap_update_bits(wled->regmap,
  795. wled->ctrl_addr + WLED3_CTRL_REG_OVP,
  796. WLED3_CTRL_REG_OVP_MASK, wled->cfg.ovp);
  797. if (rc < 0)
  798. return rc;
  799. rc = regmap_update_bits(wled->regmap,
  800. wled->ctrl_addr + WLED3_CTRL_REG_ILIMIT,
  801. WLED3_CTRL_REG_ILIMIT_MASK,
  802. wled->cfg.boost_i_limit);
  803. if (rc < 0)
  804. return rc;
  805. rc = regmap_update_bits(wled->regmap,
  806. wled->ctrl_addr + WLED3_CTRL_REG_FREQ,
  807. WLED3_CTRL_REG_FREQ_MASK,
  808. wled->cfg.switch_freq);
  809. if (rc < 0)
  810. return rc;
  811. if (wled->cfg.external_pfet) {
  812. /* Unlock the secure register access */
  813. rc = regmap_write(wled->regmap, wled->ctrl_addr +
  814. WLED4_CTRL_REG_SEC_ACCESS,
  815. WLED4_CTRL_REG_SEC_UNLOCK);
  816. if (rc < 0)
  817. return rc;
  818. rc = regmap_write(wled->regmap,
  819. wled->ctrl_addr + WLED4_CTRL_REG_TEST1,
  820. WLED4_CTRL_REG_TEST1_EXT_FET_DTEST2);
  821. if (rc < 0)
  822. return rc;
  823. }
  824. rc = regmap_read(wled->regmap, wled->sink_addr +
  825. WLED4_SINK_REG_CURR_SINK, &sink_cfg);
  826. if (rc < 0)
  827. return rc;
  828. for (i = 0; i < wled->cfg.num_strings; i++) {
  829. j = wled->cfg.enabled_strings[i];
  830. temp = j + WLED4_SINK_REG_CURR_SINK_SHFT;
  831. sink_en |= 1 << temp;
  832. }
  833. if (sink_cfg == sink_en) {
  834. rc = wled_auto_detection_at_init(wled);
  835. return rc;
  836. }
  837. rc = regmap_update_bits(wled->regmap,
  838. wled->sink_addr + WLED4_SINK_REG_CURR_SINK,
  839. WLED4_SINK_REG_CURR_SINK_MASK, 0);
  840. if (rc < 0)
  841. return rc;
  842. rc = regmap_update_bits(wled->regmap, wled->ctrl_addr +
  843. WLED3_CTRL_REG_MOD_EN,
  844. WLED3_CTRL_REG_MOD_EN_MASK, 0);
  845. if (rc < 0)
  846. return rc;
  847. /* Per sink/string configuration */
  848. for (i = 0; i < wled->cfg.num_strings; i++) {
  849. j = wled->cfg.enabled_strings[i];
  850. addr = wled->sink_addr +
  851. WLED4_SINK_REG_STR_MOD_EN(j);
  852. rc = regmap_update_bits(wled->regmap, addr,
  853. WLED4_SINK_REG_STR_MOD_MASK,
  854. WLED4_SINK_REG_STR_MOD_MASK);
  855. if (rc < 0)
  856. return rc;
  857. addr = wled->sink_addr +
  858. WLED4_SINK_REG_STR_FULL_SCALE_CURR(j);
  859. rc = regmap_update_bits(wled->regmap, addr,
  860. WLED4_SINK_REG_STR_FULL_SCALE_CURR_MASK,
  861. wled->cfg.string_i_limit);
  862. if (rc < 0)
  863. return rc;
  864. }
  865. rc = wled4_cabc_config(wled, wled->cfg.cabc);
  866. if (rc < 0)
  867. return rc;
  868. rc = regmap_update_bits(wled->regmap, wled->ctrl_addr +
  869. WLED3_CTRL_REG_MOD_EN,
  870. WLED3_CTRL_REG_MOD_EN_MASK,
  871. WLED3_CTRL_REG_MOD_EN_MASK);
  872. if (rc < 0)
  873. return rc;
  874. rc = regmap_update_bits(wled->regmap,
  875. wled->sink_addr + WLED4_SINK_REG_CURR_SINK,
  876. WLED4_SINK_REG_CURR_SINK_MASK, sink_en);
  877. if (rc < 0)
  878. return rc;
  879. rc = wled->wled_sync_toggle(wled);
  880. if (rc < 0) {
  881. dev_err(wled->dev, "Failed to toggle sync reg rc:%d\n", rc);
  882. return rc;
  883. }
  884. rc = wled_auto_detection_at_init(wled);
  885. return rc;
  886. }
  887. static const struct wled_config wled4_config_defaults = {
  888. .boost_i_limit = 4,
  889. .string_i_limit = 10,
  890. .ovp = 1,
  891. .num_strings = 4,
  892. .switch_freq = 11,
  893. .cabc = false,
  894. .external_pfet = false,
  895. .auto_detection_enabled = false,
  896. .enabled_strings = {0, 1, 2, 3},
  897. };
  898. static int wled5_setup(struct wled *wled)
  899. {
  900. int rc, temp, i, j, offset;
  901. u8 sink_en = 0;
  902. u16 addr;
  903. u32 val;
  904. rc = regmap_update_bits(wled->regmap,
  905. wled->ctrl_addr + WLED3_CTRL_REG_OVP,
  906. WLED5_CTRL_REG_OVP_MASK, wled->cfg.ovp);
  907. if (rc < 0)
  908. return rc;
  909. rc = regmap_update_bits(wled->regmap,
  910. wled->ctrl_addr + WLED3_CTRL_REG_ILIMIT,
  911. WLED3_CTRL_REG_ILIMIT_MASK,
  912. wled->cfg.boost_i_limit);
  913. if (rc < 0)
  914. return rc;
  915. rc = regmap_update_bits(wled->regmap,
  916. wled->ctrl_addr + WLED3_CTRL_REG_FREQ,
  917. WLED3_CTRL_REG_FREQ_MASK,
  918. wled->cfg.switch_freq);
  919. if (rc < 0)
  920. return rc;
  921. /* Per sink/string configuration */
  922. for (i = 0; i < wled->cfg.num_strings; ++i) {
  923. j = wled->cfg.enabled_strings[i];
  924. addr = wled->sink_addr +
  925. WLED4_SINK_REG_STR_FULL_SCALE_CURR(j);
  926. rc = regmap_update_bits(wled->regmap, addr,
  927. WLED4_SINK_REG_STR_FULL_SCALE_CURR_MASK,
  928. wled->cfg.string_i_limit);
  929. if (rc < 0)
  930. return rc;
  931. addr = wled->sink_addr + WLED5_SINK_REG_STR_SRC_SEL(j);
  932. rc = regmap_update_bits(wled->regmap, addr,
  933. WLED5_SINK_REG_SRC_SEL_MASK,
  934. wled->cfg.mod_sel == MOD_A ?
  935. WLED5_SINK_REG_SRC_SEL_MOD_A :
  936. WLED5_SINK_REG_SRC_SEL_MOD_B);
  937. temp = j + WLED4_SINK_REG_CURR_SINK_SHFT;
  938. sink_en |= 1 << temp;
  939. }
  940. rc = wled5_cabc_config(wled, wled->cfg.cabc_sel ? true : false);
  941. if (rc < 0)
  942. return rc;
  943. /* Enable one of the modulators A or B based on mod_sel */
  944. addr = wled->sink_addr + WLED5_SINK_REG_MOD_A_EN;
  945. val = (wled->cfg.mod_sel == MOD_A) ? WLED5_SINK_REG_MOD_EN_MASK : 0;
  946. rc = regmap_update_bits(wled->regmap, addr,
  947. WLED5_SINK_REG_MOD_EN_MASK, val);
  948. if (rc < 0)
  949. return rc;
  950. addr = wled->sink_addr + WLED5_SINK_REG_MOD_B_EN;
  951. val = (wled->cfg.mod_sel == MOD_B) ? WLED5_SINK_REG_MOD_EN_MASK : 0;
  952. rc = regmap_update_bits(wled->regmap, addr,
  953. WLED5_SINK_REG_MOD_EN_MASK, val);
  954. if (rc < 0)
  955. return rc;
  956. offset = (wled->cfg.mod_sel == MOD_A) ?
  957. WLED5_SINK_REG_MOD_A_BRIGHTNESS_WIDTH_SEL :
  958. WLED5_SINK_REG_MOD_B_BRIGHTNESS_WIDTH_SEL;
  959. addr = wled->sink_addr + offset;
  960. val = (wled->max_brightness == WLED5_SINK_REG_BRIGHT_MAX_15B) ?
  961. WLED5_SINK_REG_BRIGHTNESS_WIDTH_15B :
  962. WLED5_SINK_REG_BRIGHTNESS_WIDTH_12B;
  963. rc = regmap_write(wled->regmap, addr, val);
  964. if (rc < 0)
  965. return rc;
  966. rc = regmap_update_bits(wled->regmap,
  967. wled->sink_addr + WLED4_SINK_REG_CURR_SINK,
  968. WLED4_SINK_REG_CURR_SINK_MASK, sink_en);
  969. if (rc < 0)
  970. return rc;
  971. /* This updates only FSC configuration in WLED5 */
  972. rc = wled->wled_sync_toggle(wled);
  973. if (rc < 0) {
  974. pr_err("Failed to toggle sync reg rc:%d\n", rc);
  975. return rc;
  976. }
  977. rc = wled_auto_detection_at_init(wled);
  978. if (rc < 0)
  979. return rc;
  980. return 0;
  981. }
  982. static const struct wled_config wled5_config_defaults = {
  983. .boost_i_limit = 5,
  984. .string_i_limit = 10,
  985. .ovp = 4,
  986. .num_strings = 4,
  987. .switch_freq = 11,
  988. .mod_sel = 0,
  989. .cabc_sel = 0,
  990. .cabc = false,
  991. .external_pfet = false,
  992. .auto_detection_enabled = false,
  993. .enabled_strings = {0, 1, 2, 3},
  994. };
  995. static const u32 wled3_boost_i_limit_values[] = {
  996. 105, 385, 525, 805, 980, 1260, 1400, 1680,
  997. };
  998. static const struct wled_var_cfg wled3_boost_i_limit_cfg = {
  999. .values = wled3_boost_i_limit_values,
  1000. .size = ARRAY_SIZE(wled3_boost_i_limit_values),
  1001. };
  1002. static const u32 wled4_boost_i_limit_values[] = {
  1003. 105, 280, 450, 620, 970, 1150, 1300, 1500,
  1004. };
  1005. static const struct wled_var_cfg wled4_boost_i_limit_cfg = {
  1006. .values = wled4_boost_i_limit_values,
  1007. .size = ARRAY_SIZE(wled4_boost_i_limit_values),
  1008. };
  1009. static inline u32 wled5_boost_i_limit_values_fn(u32 idx)
  1010. {
  1011. return 525 + (idx * 175);
  1012. }
  1013. static const struct wled_var_cfg wled5_boost_i_limit_cfg = {
  1014. .fn = wled5_boost_i_limit_values_fn,
  1015. .size = 8,
  1016. };
  1017. static const u32 wled3_ovp_values[] = {
  1018. 35, 32, 29, 27,
  1019. };
  1020. static const struct wled_var_cfg wled3_ovp_cfg = {
  1021. .values = wled3_ovp_values,
  1022. .size = ARRAY_SIZE(wled3_ovp_values),
  1023. };
  1024. static const u32 wled4_ovp_values[] = {
  1025. 31100, 29600, 19600, 18100,
  1026. };
  1027. static const struct wled_var_cfg wled4_ovp_cfg = {
  1028. .values = wled4_ovp_values,
  1029. .size = ARRAY_SIZE(wled4_ovp_values),
  1030. };
  1031. static const u32 pmi8994_wled_ovp_values[] = {
  1032. 31000, 29500, 19400, 17800,
  1033. };
  1034. static const struct wled_var_cfg pmi8994_wled_ovp_cfg = {
  1035. .values = pmi8994_wled_ovp_values,
  1036. .size = ARRAY_SIZE(pmi8994_wled_ovp_values),
  1037. };
  1038. static inline u32 wled5_ovp_values_fn(u32 idx)
  1039. {
  1040. /*
  1041. * 0000 - 38.5 V
  1042. * 0001 - 37 V ..
  1043. * 1111 - 16 V
  1044. */
  1045. return 38500 - (idx * 1500);
  1046. }
  1047. static const struct wled_var_cfg wled5_ovp_cfg = {
  1048. .fn = wled5_ovp_values_fn,
  1049. .size = 16,
  1050. };
  1051. static u32 wled3_switch_freq_values_fn(u32 idx)
  1052. {
  1053. return 19200 / (2 * (1 + idx));
  1054. }
  1055. static const struct wled_var_cfg wled3_switch_freq_cfg = {
  1056. .fn = wled3_switch_freq_values_fn,
  1057. .size = 16,
  1058. };
  1059. static const struct wled_var_cfg wled3_string_i_limit_cfg = {
  1060. .size = 26,
  1061. };
  1062. static const u32 wled4_string_i_limit_values[] = {
  1063. 0, 2500, 5000, 7500, 10000, 12500, 15000, 17500, 20000,
  1064. 22500, 25000, 27500, 30000,
  1065. };
  1066. static const struct wled_var_cfg wled4_string_i_limit_cfg = {
  1067. .values = wled4_string_i_limit_values,
  1068. .size = ARRAY_SIZE(wled4_string_i_limit_values),
  1069. };
  1070. static const struct wled_var_cfg wled5_mod_sel_cfg = {
  1071. .size = 2,
  1072. };
  1073. static const struct wled_var_cfg wled5_cabc_sel_cfg = {
  1074. .size = 4,
  1075. };
  1076. static u32 wled_values(const struct wled_var_cfg *cfg, u32 idx)
  1077. {
  1078. if (idx >= cfg->size)
  1079. return UINT_MAX;
  1080. if (cfg->fn)
  1081. return cfg->fn(idx);
  1082. if (cfg->values)
  1083. return cfg->values[idx];
  1084. return idx;
  1085. }
  1086. static int wled_configure(struct wled *wled)
  1087. {
  1088. struct wled_config *cfg = &wled->cfg;
  1089. struct device *dev = wled->dev;
  1090. const __be32 *prop_addr;
  1091. u32 size, val, c;
  1092. int rc, i, j, string_len;
  1093. const struct wled_u32_opts *u32_opts = NULL;
  1094. const struct wled_u32_opts wled3_opts[] = {
  1095. {
  1096. .name = "qcom,current-boost-limit",
  1097. .val_ptr = &cfg->boost_i_limit,
  1098. .cfg = &wled3_boost_i_limit_cfg,
  1099. },
  1100. {
  1101. .name = "qcom,current-limit",
  1102. .val_ptr = &cfg->string_i_limit,
  1103. .cfg = &wled3_string_i_limit_cfg,
  1104. },
  1105. {
  1106. .name = "qcom,ovp",
  1107. .val_ptr = &cfg->ovp,
  1108. .cfg = &wled3_ovp_cfg,
  1109. },
  1110. {
  1111. .name = "qcom,switching-freq",
  1112. .val_ptr = &cfg->switch_freq,
  1113. .cfg = &wled3_switch_freq_cfg,
  1114. },
  1115. };
  1116. const struct wled_u32_opts wled4_opts[] = {
  1117. {
  1118. .name = "qcom,current-boost-limit",
  1119. .val_ptr = &cfg->boost_i_limit,
  1120. .cfg = &wled4_boost_i_limit_cfg,
  1121. },
  1122. {
  1123. .name = "qcom,current-limit-microamp",
  1124. .val_ptr = &cfg->string_i_limit,
  1125. .cfg = &wled4_string_i_limit_cfg,
  1126. },
  1127. {
  1128. .name = "qcom,ovp-millivolt",
  1129. .val_ptr = &cfg->ovp,
  1130. .cfg = &wled4_ovp_cfg,
  1131. },
  1132. {
  1133. .name = "qcom,switching-freq",
  1134. .val_ptr = &cfg->switch_freq,
  1135. .cfg = &wled3_switch_freq_cfg,
  1136. },
  1137. };
  1138. const struct wled_u32_opts pmi8994_wled_opts[] = {
  1139. {
  1140. .name = "qcom,current-boost-limit",
  1141. .val_ptr = &cfg->boost_i_limit,
  1142. .cfg = &wled4_boost_i_limit_cfg,
  1143. },
  1144. {
  1145. .name = "qcom,current-limit-microamp",
  1146. .val_ptr = &cfg->string_i_limit,
  1147. .cfg = &wled4_string_i_limit_cfg,
  1148. },
  1149. {
  1150. .name = "qcom,ovp-millivolt",
  1151. .val_ptr = &cfg->ovp,
  1152. .cfg = &pmi8994_wled_ovp_cfg,
  1153. },
  1154. {
  1155. .name = "qcom,switching-freq",
  1156. .val_ptr = &cfg->switch_freq,
  1157. .cfg = &wled3_switch_freq_cfg,
  1158. },
  1159. };
  1160. const struct wled_u32_opts wled5_opts[] = {
  1161. {
  1162. .name = "qcom,current-boost-limit",
  1163. .val_ptr = &cfg->boost_i_limit,
  1164. .cfg = &wled5_boost_i_limit_cfg,
  1165. },
  1166. {
  1167. .name = "qcom,current-limit-microamp",
  1168. .val_ptr = &cfg->string_i_limit,
  1169. .cfg = &wled4_string_i_limit_cfg,
  1170. },
  1171. {
  1172. .name = "qcom,ovp-millivolt",
  1173. .val_ptr = &cfg->ovp,
  1174. .cfg = &wled5_ovp_cfg,
  1175. },
  1176. {
  1177. .name = "qcom,switching-freq",
  1178. .val_ptr = &cfg->switch_freq,
  1179. .cfg = &wled3_switch_freq_cfg,
  1180. },
  1181. {
  1182. .name = "qcom,modulator-sel",
  1183. .val_ptr = &cfg->mod_sel,
  1184. .cfg = &wled5_mod_sel_cfg,
  1185. },
  1186. {
  1187. .name = "qcom,cabc-sel",
  1188. .val_ptr = &cfg->cabc_sel,
  1189. .cfg = &wled5_cabc_sel_cfg,
  1190. },
  1191. };
  1192. const struct wled_bool_opts bool_opts[] = {
  1193. { "qcom,cs-out", &cfg->cs_out_en, },
  1194. { "qcom,ext-gen", &cfg->ext_gen, },
  1195. { "qcom,cabc", &cfg->cabc, },
  1196. { "qcom,external-pfet", &cfg->external_pfet, },
  1197. { "qcom,auto-string-detection", &cfg->auto_detection_enabled, },
  1198. };
  1199. prop_addr = of_get_address(dev->of_node, 0, NULL, NULL);
  1200. if (!prop_addr) {
  1201. dev_err(wled->dev, "invalid IO resources\n");
  1202. return -EINVAL;
  1203. }
  1204. wled->ctrl_addr = be32_to_cpu(*prop_addr);
  1205. rc = of_property_read_string(dev->of_node, "label", &wled->name);
  1206. if (rc) {
  1207. wled->name = devm_kasprintf(dev, GFP_KERNEL, "%pOFn", dev->of_node);
  1208. if (!wled->name)
  1209. return -ENOMEM;
  1210. }
  1211. switch (wled->version) {
  1212. case 3:
  1213. u32_opts = wled3_opts;
  1214. size = ARRAY_SIZE(wled3_opts);
  1215. *cfg = wled3_config_defaults;
  1216. wled->wled_set_brightness = wled3_set_brightness;
  1217. wled->wled_sync_toggle = wled3_sync_toggle;
  1218. wled->max_string_count = 3;
  1219. wled->sink_addr = wled->ctrl_addr;
  1220. break;
  1221. case 4:
  1222. if (of_device_is_compatible(dev->of_node, "qcom,pmi8950-wled") ||
  1223. of_device_is_compatible(dev->of_node, "qcom,pmi8994-wled")) {
  1224. u32_opts = pmi8994_wled_opts;
  1225. size = ARRAY_SIZE(pmi8994_wled_opts);
  1226. } else {
  1227. u32_opts = wled4_opts;
  1228. size = ARRAY_SIZE(wled4_opts);
  1229. }
  1230. *cfg = wled4_config_defaults;
  1231. wled->wled_set_brightness = wled4_set_brightness;
  1232. wled->wled_sync_toggle = wled3_sync_toggle;
  1233. wled->wled_cabc_config = wled4_cabc_config;
  1234. wled->wled_ovp_delay = wled4_ovp_delay;
  1235. wled->wled_auto_detection_required =
  1236. wled4_auto_detection_required;
  1237. wled->max_string_count = 4;
  1238. prop_addr = of_get_address(dev->of_node, 1, NULL, NULL);
  1239. if (!prop_addr) {
  1240. dev_err(wled->dev, "invalid IO resources\n");
  1241. return -EINVAL;
  1242. }
  1243. wled->sink_addr = be32_to_cpu(*prop_addr);
  1244. break;
  1245. case 5:
  1246. u32_opts = wled5_opts;
  1247. size = ARRAY_SIZE(wled5_opts);
  1248. *cfg = wled5_config_defaults;
  1249. wled->wled_set_brightness = wled5_set_brightness;
  1250. wled->wled_sync_toggle = wled3_sync_toggle;
  1251. wled->wled_cabc_config = wled5_cabc_config;
  1252. wled->wled_ovp_delay = wled5_ovp_delay;
  1253. wled->wled_auto_detection_required =
  1254. wled5_auto_detection_required;
  1255. wled->max_string_count = 4;
  1256. prop_addr = of_get_address(dev->of_node, 1, NULL, NULL);
  1257. if (!prop_addr) {
  1258. dev_err(wled->dev, "invalid IO resources\n");
  1259. return -EINVAL;
  1260. }
  1261. wled->sink_addr = be32_to_cpu(*prop_addr);
  1262. break;
  1263. default:
  1264. dev_err(wled->dev, "Invalid WLED version\n");
  1265. return -EINVAL;
  1266. }
  1267. for (i = 0; i < size; ++i) {
  1268. rc = of_property_read_u32(dev->of_node, u32_opts[i].name, &val);
  1269. if (rc == -EINVAL) {
  1270. continue;
  1271. } else if (rc) {
  1272. dev_err(dev, "error reading '%s'\n", u32_opts[i].name);
  1273. return rc;
  1274. }
  1275. c = UINT_MAX;
  1276. for (j = 0; c != val; j++) {
  1277. c = wled_values(u32_opts[i].cfg, j);
  1278. if (c == UINT_MAX) {
  1279. dev_err(dev, "invalid value for '%s'\n",
  1280. u32_opts[i].name);
  1281. return -EINVAL;
  1282. }
  1283. if (c == val)
  1284. break;
  1285. }
  1286. dev_dbg(dev, "'%s' = %u\n", u32_opts[i].name, c);
  1287. *u32_opts[i].val_ptr = j;
  1288. }
  1289. for (i = 0; i < ARRAY_SIZE(bool_opts); ++i) {
  1290. if (of_property_read_bool(dev->of_node, bool_opts[i].name))
  1291. *bool_opts[i].val_ptr = true;
  1292. }
  1293. string_len = of_property_count_elems_of_size(dev->of_node,
  1294. "qcom,enabled-strings",
  1295. sizeof(u32));
  1296. if (string_len > 0) {
  1297. if (string_len > wled->max_string_count) {
  1298. dev_err(dev, "Cannot have more than %d strings\n",
  1299. wled->max_string_count);
  1300. return -EINVAL;
  1301. }
  1302. rc = of_property_read_u32_array(dev->of_node,
  1303. "qcom,enabled-strings",
  1304. wled->cfg.enabled_strings,
  1305. string_len);
  1306. if (rc) {
  1307. dev_err(dev, "Failed to read %d elements from qcom,enabled-strings: %d\n",
  1308. string_len, rc);
  1309. return rc;
  1310. }
  1311. for (i = 0; i < string_len; ++i) {
  1312. if (wled->cfg.enabled_strings[i] >= wled->max_string_count) {
  1313. dev_err(dev,
  1314. "qcom,enabled-strings index %d at %d is out of bounds\n",
  1315. wled->cfg.enabled_strings[i], i);
  1316. return -EINVAL;
  1317. }
  1318. }
  1319. cfg->num_strings = string_len;
  1320. }
  1321. rc = of_property_read_u32(dev->of_node, "qcom,num-strings", &val);
  1322. if (!rc) {
  1323. if (val < 1 || val > wled->max_string_count) {
  1324. dev_err(dev, "qcom,num-strings must be between 1 and %d\n",
  1325. wled->max_string_count);
  1326. return -EINVAL;
  1327. }
  1328. if (string_len > 0) {
  1329. dev_warn(dev, "Only one of qcom,num-strings or qcom,enabled-strings"
  1330. " should be set\n");
  1331. if (val > string_len) {
  1332. dev_err(dev, "qcom,num-strings exceeds qcom,enabled-strings\n");
  1333. return -EINVAL;
  1334. }
  1335. }
  1336. cfg->num_strings = val;
  1337. }
  1338. return 0;
  1339. }
  1340. static int wled_configure_short_irq(struct wled *wled,
  1341. struct platform_device *pdev)
  1342. {
  1343. int rc;
  1344. if (!wled->has_short_detect)
  1345. return 0;
  1346. rc = regmap_update_bits(wled->regmap, wled->ctrl_addr +
  1347. WLED4_CTRL_REG_SHORT_PROTECT,
  1348. WLED4_CTRL_REG_SHORT_EN_MASK,
  1349. WLED4_CTRL_REG_SHORT_EN_MASK);
  1350. if (rc < 0)
  1351. return rc;
  1352. wled->short_irq = platform_get_irq_byname(pdev, "short");
  1353. if (wled->short_irq < 0) {
  1354. dev_dbg(&pdev->dev, "short irq is not used\n");
  1355. return 0;
  1356. }
  1357. rc = devm_request_threaded_irq(wled->dev, wled->short_irq,
  1358. NULL, wled_short_irq_handler,
  1359. IRQF_ONESHOT,
  1360. "wled_short_irq", wled);
  1361. if (rc < 0)
  1362. dev_err(wled->dev, "Unable to request short_irq (err:%d)\n",
  1363. rc);
  1364. return rc;
  1365. }
  1366. static int wled_configure_ovp_irq(struct wled *wled,
  1367. struct platform_device *pdev)
  1368. {
  1369. int rc;
  1370. u32 val;
  1371. wled->ovp_irq = platform_get_irq_byname(pdev, "ovp");
  1372. if (wled->ovp_irq < 0) {
  1373. dev_dbg(&pdev->dev, "OVP IRQ not found - disabling automatic string detection\n");
  1374. return 0;
  1375. }
  1376. rc = devm_request_threaded_irq(wled->dev, wled->ovp_irq, NULL,
  1377. wled_ovp_irq_handler, IRQF_ONESHOT,
  1378. "wled_ovp_irq", wled);
  1379. if (rc < 0) {
  1380. dev_err(wled->dev, "Unable to request ovp_irq (err:%d)\n",
  1381. rc);
  1382. wled->ovp_irq = 0;
  1383. return 0;
  1384. }
  1385. rc = regmap_read(wled->regmap, wled->ctrl_addr +
  1386. WLED3_CTRL_REG_MOD_EN, &val);
  1387. if (rc < 0)
  1388. return rc;
  1389. /* Keep OVP irq disabled until module is enabled */
  1390. if (!(val & WLED3_CTRL_REG_MOD_EN_MASK))
  1391. disable_irq(wled->ovp_irq);
  1392. return 0;
  1393. }
  1394. static const struct backlight_ops wled_ops = {
  1395. .update_status = wled_update_status,
  1396. };
  1397. static int wled_probe(struct platform_device *pdev)
  1398. {
  1399. struct backlight_properties props;
  1400. struct backlight_device *bl;
  1401. struct wled *wled;
  1402. struct regmap *regmap;
  1403. u32 val;
  1404. int rc;
  1405. regmap = dev_get_regmap(pdev->dev.parent, NULL);
  1406. if (!regmap) {
  1407. dev_err(&pdev->dev, "Unable to get regmap\n");
  1408. return -EINVAL;
  1409. }
  1410. wled = devm_kzalloc(&pdev->dev, sizeof(*wled), GFP_KERNEL);
  1411. if (!wled)
  1412. return -ENOMEM;
  1413. wled->regmap = regmap;
  1414. wled->dev = &pdev->dev;
  1415. wled->version = (uintptr_t)of_device_get_match_data(&pdev->dev);
  1416. if (!wled->version) {
  1417. dev_err(&pdev->dev, "Unknown device version\n");
  1418. return -ENODEV;
  1419. }
  1420. mutex_init(&wled->lock);
  1421. rc = wled_configure(wled);
  1422. if (rc)
  1423. return rc;
  1424. val = WLED3_SINK_REG_BRIGHT_MAX;
  1425. of_property_read_u32(pdev->dev.of_node, "max-brightness", &val);
  1426. wled->max_brightness = val;
  1427. switch (wled->version) {
  1428. case 3:
  1429. wled->cfg.auto_detection_enabled = false;
  1430. rc = wled3_setup(wled);
  1431. if (rc) {
  1432. dev_err(&pdev->dev, "wled3_setup failed\n");
  1433. return rc;
  1434. }
  1435. break;
  1436. case 4:
  1437. wled->has_short_detect = true;
  1438. rc = wled4_setup(wled);
  1439. if (rc) {
  1440. dev_err(&pdev->dev, "wled4_setup failed\n");
  1441. return rc;
  1442. }
  1443. break;
  1444. case 5:
  1445. wled->has_short_detect = true;
  1446. if (wled->cfg.cabc_sel)
  1447. wled->max_brightness = WLED5_SINK_REG_BRIGHT_MAX_12B;
  1448. rc = wled5_setup(wled);
  1449. if (rc) {
  1450. dev_err(&pdev->dev, "wled5_setup failed\n");
  1451. return rc;
  1452. }
  1453. break;
  1454. default:
  1455. dev_err(wled->dev, "Invalid WLED version\n");
  1456. break;
  1457. }
  1458. INIT_DELAYED_WORK(&wled->ovp_work, wled_ovp_work);
  1459. rc = wled_configure_short_irq(wled, pdev);
  1460. if (rc < 0)
  1461. return rc;
  1462. rc = wled_configure_ovp_irq(wled, pdev);
  1463. if (rc < 0)
  1464. return rc;
  1465. val = WLED_DEFAULT_BRIGHTNESS;
  1466. of_property_read_u32(pdev->dev.of_node, "default-brightness", &val);
  1467. memset(&props, 0, sizeof(struct backlight_properties));
  1468. props.type = BACKLIGHT_RAW;
  1469. props.brightness = val;
  1470. props.max_brightness = wled->max_brightness;
  1471. bl = devm_backlight_device_register(&pdev->dev, wled->name,
  1472. &pdev->dev, wled,
  1473. &wled_ops, &props);
  1474. return PTR_ERR_OR_ZERO(bl);
  1475. };
  1476. static void wled_remove(struct platform_device *pdev)
  1477. {
  1478. struct wled *wled = platform_get_drvdata(pdev);
  1479. mutex_destroy(&wled->lock);
  1480. cancel_delayed_work_sync(&wled->ovp_work);
  1481. disable_irq(wled->short_irq);
  1482. disable_irq(wled->ovp_irq);
  1483. }
  1484. static const struct of_device_id wled_match_table[] = {
  1485. { .compatible = "qcom,pm8941-wled", .data = (void *)3 },
  1486. { .compatible = "qcom,pmi8950-wled", .data = (void *)4 },
  1487. { .compatible = "qcom,pmi8994-wled", .data = (void *)4 },
  1488. { .compatible = "qcom,pmi8998-wled", .data = (void *)4 },
  1489. { .compatible = "qcom,pm660l-wled", .data = (void *)4 },
  1490. { .compatible = "qcom,pm6150l-wled", .data = (void *)5 },
  1491. { .compatible = "qcom,pm8150l-wled", .data = (void *)5 },
  1492. {}
  1493. };
  1494. MODULE_DEVICE_TABLE(of, wled_match_table);
  1495. static struct platform_driver wled_driver = {
  1496. .probe = wled_probe,
  1497. .remove = wled_remove,
  1498. .driver = {
  1499. .name = "qcom,wled",
  1500. .of_match_table = wled_match_table,
  1501. },
  1502. };
  1503. module_platform_driver(wled_driver);
  1504. MODULE_DESCRIPTION("Qualcomm WLED driver");
  1505. MODULE_LICENSE("GPL v2");