wm8350-irq.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  1. /*
  2. * wm8350-irq.c -- IRQ support for Wolfson WM8350
  3. *
  4. * Copyright 2007, 2008, 2009 Wolfson Microelectronics PLC.
  5. *
  6. * Author: Liam Girdwood, Mark Brown
  7. *
  8. * This program is free software; you can redistribute it and/or modify it
  9. * under the terms of the GNU General Public License as published by the
  10. * Free Software Foundation; either version 2 of the License, or (at your
  11. * option) any later version.
  12. *
  13. */
  14. #include <linux/kernel.h>
  15. #include <linux/module.h>
  16. #include <linux/init.h>
  17. #include <linux/bug.h>
  18. #include <linux/device.h>
  19. #include <linux/interrupt.h>
  20. #include <linux/workqueue.h>
  21. #include <linux/mfd/wm8350/core.h>
  22. #include <linux/mfd/wm8350/audio.h>
  23. #include <linux/mfd/wm8350/comparator.h>
  24. #include <linux/mfd/wm8350/gpio.h>
  25. #include <linux/mfd/wm8350/pmic.h>
  26. #include <linux/mfd/wm8350/rtc.h>
  27. #include <linux/mfd/wm8350/supply.h>
  28. #include <linux/mfd/wm8350/wdt.h>
  29. #define WM8350_NUM_IRQ_REGS 7
  30. #define WM8350_INT_OFFSET_1 0
  31. #define WM8350_INT_OFFSET_2 1
  32. #define WM8350_POWER_UP_INT_OFFSET 2
  33. #define WM8350_UNDER_VOLTAGE_INT_OFFSET 3
  34. #define WM8350_OVER_CURRENT_INT_OFFSET 4
  35. #define WM8350_GPIO_INT_OFFSET 5
  36. #define WM8350_COMPARATOR_INT_OFFSET 6
  37. struct wm8350_irq_data {
  38. int primary;
  39. int reg;
  40. int mask;
  41. int primary_only;
  42. };
  43. static struct wm8350_irq_data wm8350_irqs[] = {
  44. [WM8350_IRQ_OC_LS] = {
  45. .primary = WM8350_OC_INT,
  46. .reg = WM8350_OVER_CURRENT_INT_OFFSET,
  47. .mask = WM8350_OC_LS_EINT,
  48. .primary_only = 1,
  49. },
  50. [WM8350_IRQ_UV_DC1] = {
  51. .primary = WM8350_UV_INT,
  52. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  53. .mask = WM8350_UV_DC1_EINT,
  54. },
  55. [WM8350_IRQ_UV_DC2] = {
  56. .primary = WM8350_UV_INT,
  57. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  58. .mask = WM8350_UV_DC2_EINT,
  59. },
  60. [WM8350_IRQ_UV_DC3] = {
  61. .primary = WM8350_UV_INT,
  62. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  63. .mask = WM8350_UV_DC3_EINT,
  64. },
  65. [WM8350_IRQ_UV_DC4] = {
  66. .primary = WM8350_UV_INT,
  67. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  68. .mask = WM8350_UV_DC4_EINT,
  69. },
  70. [WM8350_IRQ_UV_DC5] = {
  71. .primary = WM8350_UV_INT,
  72. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  73. .mask = WM8350_UV_DC5_EINT,
  74. },
  75. [WM8350_IRQ_UV_DC6] = {
  76. .primary = WM8350_UV_INT,
  77. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  78. .mask = WM8350_UV_DC6_EINT,
  79. },
  80. [WM8350_IRQ_UV_LDO1] = {
  81. .primary = WM8350_UV_INT,
  82. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  83. .mask = WM8350_UV_LDO1_EINT,
  84. },
  85. [WM8350_IRQ_UV_LDO2] = {
  86. .primary = WM8350_UV_INT,
  87. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  88. .mask = WM8350_UV_LDO2_EINT,
  89. },
  90. [WM8350_IRQ_UV_LDO3] = {
  91. .primary = WM8350_UV_INT,
  92. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  93. .mask = WM8350_UV_LDO3_EINT,
  94. },
  95. [WM8350_IRQ_UV_LDO4] = {
  96. .primary = WM8350_UV_INT,
  97. .reg = WM8350_UNDER_VOLTAGE_INT_OFFSET,
  98. .mask = WM8350_UV_LDO4_EINT,
  99. },
  100. [WM8350_IRQ_CHG_BAT_HOT] = {
  101. .primary = WM8350_CHG_INT,
  102. .reg = WM8350_INT_OFFSET_1,
  103. .mask = WM8350_CHG_BAT_HOT_EINT,
  104. },
  105. [WM8350_IRQ_CHG_BAT_COLD] = {
  106. .primary = WM8350_CHG_INT,
  107. .reg = WM8350_INT_OFFSET_1,
  108. .mask = WM8350_CHG_BAT_COLD_EINT,
  109. },
  110. [WM8350_IRQ_CHG_BAT_FAIL] = {
  111. .primary = WM8350_CHG_INT,
  112. .reg = WM8350_INT_OFFSET_1,
  113. .mask = WM8350_CHG_BAT_FAIL_EINT,
  114. },
  115. [WM8350_IRQ_CHG_TO] = {
  116. .primary = WM8350_CHG_INT,
  117. .reg = WM8350_INT_OFFSET_1,
  118. .mask = WM8350_CHG_TO_EINT,
  119. },
  120. [WM8350_IRQ_CHG_END] = {
  121. .primary = WM8350_CHG_INT,
  122. .reg = WM8350_INT_OFFSET_1,
  123. .mask = WM8350_CHG_END_EINT,
  124. },
  125. [WM8350_IRQ_CHG_START] = {
  126. .primary = WM8350_CHG_INT,
  127. .reg = WM8350_INT_OFFSET_1,
  128. .mask = WM8350_CHG_START_EINT,
  129. },
  130. [WM8350_IRQ_CHG_FAST_RDY] = {
  131. .primary = WM8350_CHG_INT,
  132. .reg = WM8350_INT_OFFSET_1,
  133. .mask = WM8350_CHG_FAST_RDY_EINT,
  134. },
  135. [WM8350_IRQ_CHG_VBATT_LT_3P9] = {
  136. .primary = WM8350_CHG_INT,
  137. .reg = WM8350_INT_OFFSET_1,
  138. .mask = WM8350_CHG_VBATT_LT_3P9_EINT,
  139. },
  140. [WM8350_IRQ_CHG_VBATT_LT_3P1] = {
  141. .primary = WM8350_CHG_INT,
  142. .reg = WM8350_INT_OFFSET_1,
  143. .mask = WM8350_CHG_VBATT_LT_3P1_EINT,
  144. },
  145. [WM8350_IRQ_CHG_VBATT_LT_2P85] = {
  146. .primary = WM8350_CHG_INT,
  147. .reg = WM8350_INT_OFFSET_1,
  148. .mask = WM8350_CHG_VBATT_LT_2P85_EINT,
  149. },
  150. [WM8350_IRQ_RTC_ALM] = {
  151. .primary = WM8350_RTC_INT,
  152. .reg = WM8350_INT_OFFSET_1,
  153. .mask = WM8350_RTC_ALM_EINT,
  154. },
  155. [WM8350_IRQ_RTC_SEC] = {
  156. .primary = WM8350_RTC_INT,
  157. .reg = WM8350_INT_OFFSET_1,
  158. .mask = WM8350_RTC_SEC_EINT,
  159. },
  160. [WM8350_IRQ_RTC_PER] = {
  161. .primary = WM8350_RTC_INT,
  162. .reg = WM8350_INT_OFFSET_1,
  163. .mask = WM8350_RTC_PER_EINT,
  164. },
  165. [WM8350_IRQ_CS1] = {
  166. .primary = WM8350_CS_INT,
  167. .reg = WM8350_INT_OFFSET_2,
  168. .mask = WM8350_CS1_EINT,
  169. },
  170. [WM8350_IRQ_CS2] = {
  171. .primary = WM8350_CS_INT,
  172. .reg = WM8350_INT_OFFSET_2,
  173. .mask = WM8350_CS2_EINT,
  174. },
  175. [WM8350_IRQ_SYS_HYST_COMP_FAIL] = {
  176. .primary = WM8350_SYS_INT,
  177. .reg = WM8350_INT_OFFSET_2,
  178. .mask = WM8350_SYS_HYST_COMP_FAIL_EINT,
  179. },
  180. [WM8350_IRQ_SYS_CHIP_GT115] = {
  181. .primary = WM8350_SYS_INT,
  182. .reg = WM8350_INT_OFFSET_2,
  183. .mask = WM8350_SYS_CHIP_GT115_EINT,
  184. },
  185. [WM8350_IRQ_SYS_CHIP_GT140] = {
  186. .primary = WM8350_SYS_INT,
  187. .reg = WM8350_INT_OFFSET_2,
  188. .mask = WM8350_SYS_CHIP_GT140_EINT,
  189. },
  190. [WM8350_IRQ_SYS_WDOG_TO] = {
  191. .primary = WM8350_SYS_INT,
  192. .reg = WM8350_INT_OFFSET_2,
  193. .mask = WM8350_SYS_WDOG_TO_EINT,
  194. },
  195. [WM8350_IRQ_AUXADC_DATARDY] = {
  196. .primary = WM8350_AUXADC_INT,
  197. .reg = WM8350_INT_OFFSET_2,
  198. .mask = WM8350_AUXADC_DATARDY_EINT,
  199. },
  200. [WM8350_IRQ_AUXADC_DCOMP4] = {
  201. .primary = WM8350_AUXADC_INT,
  202. .reg = WM8350_INT_OFFSET_2,
  203. .mask = WM8350_AUXADC_DCOMP4_EINT,
  204. },
  205. [WM8350_IRQ_AUXADC_DCOMP3] = {
  206. .primary = WM8350_AUXADC_INT,
  207. .reg = WM8350_INT_OFFSET_2,
  208. .mask = WM8350_AUXADC_DCOMP3_EINT,
  209. },
  210. [WM8350_IRQ_AUXADC_DCOMP2] = {
  211. .primary = WM8350_AUXADC_INT,
  212. .reg = WM8350_INT_OFFSET_2,
  213. .mask = WM8350_AUXADC_DCOMP2_EINT,
  214. },
  215. [WM8350_IRQ_AUXADC_DCOMP1] = {
  216. .primary = WM8350_AUXADC_INT,
  217. .reg = WM8350_INT_OFFSET_2,
  218. .mask = WM8350_AUXADC_DCOMP1_EINT,
  219. },
  220. [WM8350_IRQ_USB_LIMIT] = {
  221. .primary = WM8350_USB_INT,
  222. .reg = WM8350_INT_OFFSET_2,
  223. .mask = WM8350_USB_LIMIT_EINT,
  224. .primary_only = 1,
  225. },
  226. [WM8350_IRQ_WKUP_OFF_STATE] = {
  227. .primary = WM8350_WKUP_INT,
  228. .reg = WM8350_COMPARATOR_INT_OFFSET,
  229. .mask = WM8350_WKUP_OFF_STATE_EINT,
  230. },
  231. [WM8350_IRQ_WKUP_HIB_STATE] = {
  232. .primary = WM8350_WKUP_INT,
  233. .reg = WM8350_COMPARATOR_INT_OFFSET,
  234. .mask = WM8350_WKUP_HIB_STATE_EINT,
  235. },
  236. [WM8350_IRQ_WKUP_CONV_FAULT] = {
  237. .primary = WM8350_WKUP_INT,
  238. .reg = WM8350_COMPARATOR_INT_OFFSET,
  239. .mask = WM8350_WKUP_CONV_FAULT_EINT,
  240. },
  241. [WM8350_IRQ_WKUP_WDOG_RST] = {
  242. .primary = WM8350_WKUP_INT,
  243. .reg = WM8350_COMPARATOR_INT_OFFSET,
  244. .mask = WM8350_WKUP_WDOG_RST_EINT,
  245. },
  246. [WM8350_IRQ_WKUP_GP_PWR_ON] = {
  247. .primary = WM8350_WKUP_INT,
  248. .reg = WM8350_COMPARATOR_INT_OFFSET,
  249. .mask = WM8350_WKUP_GP_PWR_ON_EINT,
  250. },
  251. [WM8350_IRQ_WKUP_ONKEY] = {
  252. .primary = WM8350_WKUP_INT,
  253. .reg = WM8350_COMPARATOR_INT_OFFSET,
  254. .mask = WM8350_WKUP_ONKEY_EINT,
  255. },
  256. [WM8350_IRQ_WKUP_GP_WAKEUP] = {
  257. .primary = WM8350_WKUP_INT,
  258. .reg = WM8350_COMPARATOR_INT_OFFSET,
  259. .mask = WM8350_WKUP_GP_WAKEUP_EINT,
  260. },
  261. [WM8350_IRQ_CODEC_JCK_DET_L] = {
  262. .primary = WM8350_CODEC_INT,
  263. .reg = WM8350_COMPARATOR_INT_OFFSET,
  264. .mask = WM8350_CODEC_JCK_DET_L_EINT,
  265. },
  266. [WM8350_IRQ_CODEC_JCK_DET_R] = {
  267. .primary = WM8350_CODEC_INT,
  268. .reg = WM8350_COMPARATOR_INT_OFFSET,
  269. .mask = WM8350_CODEC_JCK_DET_R_EINT,
  270. },
  271. [WM8350_IRQ_CODEC_MICSCD] = {
  272. .primary = WM8350_CODEC_INT,
  273. .reg = WM8350_COMPARATOR_INT_OFFSET,
  274. .mask = WM8350_CODEC_MICSCD_EINT,
  275. },
  276. [WM8350_IRQ_CODEC_MICD] = {
  277. .primary = WM8350_CODEC_INT,
  278. .reg = WM8350_COMPARATOR_INT_OFFSET,
  279. .mask = WM8350_CODEC_MICD_EINT,
  280. },
  281. [WM8350_IRQ_EXT_USB_FB] = {
  282. .primary = WM8350_EXT_INT,
  283. .reg = WM8350_COMPARATOR_INT_OFFSET,
  284. .mask = WM8350_EXT_USB_FB_EINT,
  285. },
  286. [WM8350_IRQ_EXT_WALL_FB] = {
  287. .primary = WM8350_EXT_INT,
  288. .reg = WM8350_COMPARATOR_INT_OFFSET,
  289. .mask = WM8350_EXT_WALL_FB_EINT,
  290. },
  291. [WM8350_IRQ_EXT_BAT_FB] = {
  292. .primary = WM8350_EXT_INT,
  293. .reg = WM8350_COMPARATOR_INT_OFFSET,
  294. .mask = WM8350_EXT_BAT_FB_EINT,
  295. },
  296. [WM8350_IRQ_GPIO(0)] = {
  297. .primary = WM8350_GP_INT,
  298. .reg = WM8350_GPIO_INT_OFFSET,
  299. .mask = WM8350_GP0_EINT,
  300. },
  301. [WM8350_IRQ_GPIO(1)] = {
  302. .primary = WM8350_GP_INT,
  303. .reg = WM8350_GPIO_INT_OFFSET,
  304. .mask = WM8350_GP1_EINT,
  305. },
  306. [WM8350_IRQ_GPIO(2)] = {
  307. .primary = WM8350_GP_INT,
  308. .reg = WM8350_GPIO_INT_OFFSET,
  309. .mask = WM8350_GP2_EINT,
  310. },
  311. [WM8350_IRQ_GPIO(3)] = {
  312. .primary = WM8350_GP_INT,
  313. .reg = WM8350_GPIO_INT_OFFSET,
  314. .mask = WM8350_GP3_EINT,
  315. },
  316. [WM8350_IRQ_GPIO(4)] = {
  317. .primary = WM8350_GP_INT,
  318. .reg = WM8350_GPIO_INT_OFFSET,
  319. .mask = WM8350_GP4_EINT,
  320. },
  321. [WM8350_IRQ_GPIO(5)] = {
  322. .primary = WM8350_GP_INT,
  323. .reg = WM8350_GPIO_INT_OFFSET,
  324. .mask = WM8350_GP5_EINT,
  325. },
  326. [WM8350_IRQ_GPIO(6)] = {
  327. .primary = WM8350_GP_INT,
  328. .reg = WM8350_GPIO_INT_OFFSET,
  329. .mask = WM8350_GP6_EINT,
  330. },
  331. [WM8350_IRQ_GPIO(7)] = {
  332. .primary = WM8350_GP_INT,
  333. .reg = WM8350_GPIO_INT_OFFSET,
  334. .mask = WM8350_GP7_EINT,
  335. },
  336. [WM8350_IRQ_GPIO(8)] = {
  337. .primary = WM8350_GP_INT,
  338. .reg = WM8350_GPIO_INT_OFFSET,
  339. .mask = WM8350_GP8_EINT,
  340. },
  341. [WM8350_IRQ_GPIO(9)] = {
  342. .primary = WM8350_GP_INT,
  343. .reg = WM8350_GPIO_INT_OFFSET,
  344. .mask = WM8350_GP9_EINT,
  345. },
  346. [WM8350_IRQ_GPIO(10)] = {
  347. .primary = WM8350_GP_INT,
  348. .reg = WM8350_GPIO_INT_OFFSET,
  349. .mask = WM8350_GP10_EINT,
  350. },
  351. [WM8350_IRQ_GPIO(11)] = {
  352. .primary = WM8350_GP_INT,
  353. .reg = WM8350_GPIO_INT_OFFSET,
  354. .mask = WM8350_GP11_EINT,
  355. },
  356. [WM8350_IRQ_GPIO(12)] = {
  357. .primary = WM8350_GP_INT,
  358. .reg = WM8350_GPIO_INT_OFFSET,
  359. .mask = WM8350_GP12_EINT,
  360. },
  361. };
  362. static void wm8350_irq_call_handler(struct wm8350 *wm8350, int irq)
  363. {
  364. mutex_lock(&wm8350->irq_mutex);
  365. if (wm8350->irq[irq].handler)
  366. wm8350->irq[irq].handler(irq, wm8350->irq[irq].data);
  367. else {
  368. dev_err(wm8350->dev, "irq %d nobody cared. now masked.\n",
  369. irq);
  370. wm8350_mask_irq(wm8350, irq);
  371. }
  372. mutex_unlock(&wm8350->irq_mutex);
  373. }
  374. /*
  375. * This is a threaded IRQ handler so can access I2C/SPI. Since all
  376. * interrupts are clear on read the IRQ line will be reasserted and
  377. * the physical IRQ will be handled again if another interrupt is
  378. * asserted while we run - in the normal course of events this is a
  379. * rare occurrence so we save I2C/SPI reads.
  380. */
  381. static irqreturn_t wm8350_irq(int irq, void *irq_data)
  382. {
  383. struct wm8350 *wm8350 = irq_data;
  384. u16 level_one;
  385. u16 sub_reg[WM8350_NUM_IRQ_REGS];
  386. int read_done[WM8350_NUM_IRQ_REGS];
  387. struct wm8350_irq_data *data;
  388. int i;
  389. /* TODO: Use block reads to improve performance? */
  390. level_one = wm8350_reg_read(wm8350, WM8350_SYSTEM_INTERRUPTS)
  391. & ~wm8350_reg_read(wm8350, WM8350_SYSTEM_INTERRUPTS_MASK);
  392. if (!level_one)
  393. return IRQ_NONE;
  394. memset(&read_done, 0, sizeof(read_done));
  395. for (i = 0; i < ARRAY_SIZE(wm8350_irqs); i++) {
  396. data = &wm8350_irqs[i];
  397. if (!(level_one & data->primary))
  398. continue;
  399. if (!read_done[data->reg]) {
  400. sub_reg[data->reg] =
  401. wm8350_reg_read(wm8350, WM8350_INT_STATUS_1 +
  402. data->reg);
  403. sub_reg[data->reg] &=
  404. ~wm8350_reg_read(wm8350,
  405. WM8350_INT_STATUS_1_MASK +
  406. data->reg);
  407. read_done[data->reg] = 1;
  408. }
  409. if (sub_reg[data->reg] & data->mask)
  410. wm8350_irq_call_handler(wm8350, i);
  411. }
  412. return IRQ_HANDLED;
  413. }
  414. int wm8350_register_irq(struct wm8350 *wm8350, int irq,
  415. irq_handler_t handler, unsigned long flags,
  416. const char *name, void *data)
  417. {
  418. if (irq < 0 || irq > WM8350_NUM_IRQ || !handler)
  419. return -EINVAL;
  420. if (wm8350->irq[irq].handler)
  421. return -EBUSY;
  422. mutex_lock(&wm8350->irq_mutex);
  423. wm8350->irq[irq].handler = handler;
  424. wm8350->irq[irq].data = data;
  425. mutex_unlock(&wm8350->irq_mutex);
  426. wm8350_unmask_irq(wm8350, irq);
  427. return 0;
  428. }
  429. EXPORT_SYMBOL_GPL(wm8350_register_irq);
  430. int wm8350_free_irq(struct wm8350 *wm8350, int irq)
  431. {
  432. if (irq < 0 || irq > WM8350_NUM_IRQ)
  433. return -EINVAL;
  434. wm8350_mask_irq(wm8350, irq);
  435. mutex_lock(&wm8350->irq_mutex);
  436. wm8350->irq[irq].handler = NULL;
  437. mutex_unlock(&wm8350->irq_mutex);
  438. return 0;
  439. }
  440. EXPORT_SYMBOL_GPL(wm8350_free_irq);
  441. int wm8350_mask_irq(struct wm8350 *wm8350, int irq)
  442. {
  443. return wm8350_set_bits(wm8350, WM8350_INT_STATUS_1_MASK +
  444. wm8350_irqs[irq].reg,
  445. wm8350_irqs[irq].mask);
  446. }
  447. EXPORT_SYMBOL_GPL(wm8350_mask_irq);
  448. int wm8350_unmask_irq(struct wm8350 *wm8350, int irq)
  449. {
  450. return wm8350_clear_bits(wm8350, WM8350_INT_STATUS_1_MASK +
  451. wm8350_irqs[irq].reg,
  452. wm8350_irqs[irq].mask);
  453. }
  454. EXPORT_SYMBOL_GPL(wm8350_unmask_irq);
  455. int wm8350_irq_init(struct wm8350 *wm8350, int irq,
  456. struct wm8350_platform_data *pdata)
  457. {
  458. int ret;
  459. int flags = IRQF_ONESHOT;
  460. if (!irq) {
  461. dev_err(wm8350->dev, "No IRQ configured\n");
  462. return -EINVAL;
  463. }
  464. wm8350_reg_write(wm8350, WM8350_SYSTEM_INTERRUPTS_MASK, 0xFFFF);
  465. wm8350_reg_write(wm8350, WM8350_INT_STATUS_1_MASK, 0xFFFF);
  466. wm8350_reg_write(wm8350, WM8350_INT_STATUS_2_MASK, 0xFFFF);
  467. wm8350_reg_write(wm8350, WM8350_UNDER_VOLTAGE_INT_STATUS_MASK, 0xFFFF);
  468. wm8350_reg_write(wm8350, WM8350_GPIO_INT_STATUS_MASK, 0xFFFF);
  469. wm8350_reg_write(wm8350, WM8350_COMPARATOR_INT_STATUS_MASK, 0xFFFF);
  470. mutex_init(&wm8350->irq_mutex);
  471. wm8350->chip_irq = irq;
  472. if (pdata && pdata->irq_high) {
  473. flags |= IRQF_TRIGGER_HIGH;
  474. wm8350_set_bits(wm8350, WM8350_SYSTEM_CONTROL_1,
  475. WM8350_IRQ_POL);
  476. } else {
  477. flags |= IRQF_TRIGGER_LOW;
  478. wm8350_clear_bits(wm8350, WM8350_SYSTEM_CONTROL_1,
  479. WM8350_IRQ_POL);
  480. }
  481. ret = request_threaded_irq(irq, NULL, wm8350_irq, flags,
  482. "wm8350", wm8350);
  483. if (ret != 0)
  484. dev_err(wm8350->dev, "Failed to request IRQ: %d\n", ret);
  485. return ret;
  486. }
  487. int wm8350_irq_exit(struct wm8350 *wm8350)
  488. {
  489. free_irq(wm8350->chip_irq, wm8350);
  490. return 0;
  491. }