max77686-irq.c 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. /*
  2. * max77686-irq.c - Interrupt controller support for MAX77686
  3. *
  4. * Copyright (C) 2012 Samsung Electronics Co.Ltd
  5. * Chiwoong Byun <woong.byun@samsung.com>
  6. *
  7. * This program is free software; you can redistribute it and/or modify
  8. * it under the terms of the GNU General Public License as published by
  9. * the Free Software Foundation; either version 2 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * This program is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU General Public License
  18. * along with this program; if not, write to the Free Software
  19. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  20. *
  21. * This driver is based on max8997-irq.c
  22. */
  23. #include <linux/err.h>
  24. #include <linux/irq.h>
  25. #include <linux/interrupt.h>
  26. #include <linux/gpio.h>
  27. #include <linux/mfd/max77686.h>
  28. #include <linux/mfd/max77686-private.h>
  29. #include <linux/irqdomain.h>
  30. #include <linux/regmap.h>
  31. enum {
  32. MAX77686_DEBUG_IRQ_INFO = 1 << 0,
  33. MAX77686_DEBUG_IRQ_MASK = 1 << 1,
  34. MAX77686_DEBUG_IRQ_INT = 1 << 2,
  35. };
  36. static int debug_mask = 0;
  37. module_param(debug_mask, int, 0);
  38. MODULE_PARM_DESC(debug_mask, "Set debug_mask : 0x0=off 0x1=IRQ_INFO 0x2=IRQ_MASK 0x4=IRQ_INI)");
  39. static const u8 max77686_mask_reg[] = {
  40. [PMIC_INT1] = MAX77686_REG_INT1MSK,
  41. [PMIC_INT2] = MAX77686_REG_INT2MSK,
  42. [RTC_INT] = MAX77686_RTC_INTM,
  43. };
  44. static struct regmap *max77686_get_regmap(struct max77686_dev *max77686,
  45. enum max77686_irq_source src)
  46. {
  47. switch (src) {
  48. case PMIC_INT1 ... PMIC_INT2:
  49. return max77686->regmap;
  50. case RTC_INT:
  51. return max77686->rtc_regmap;
  52. default:
  53. return ERR_PTR(-EINVAL);
  54. }
  55. }
  56. struct max77686_irq_data {
  57. int mask;
  58. enum max77686_irq_source group;
  59. };
  60. #define DECLARE_IRQ(idx, _group, _mask) \
  61. [(idx)] = { .group = (_group), .mask = (_mask) }
  62. static const struct max77686_irq_data max77686_irqs[] = {
  63. DECLARE_IRQ(MAX77686_PMICIRQ_PWRONF, PMIC_INT1, 1 << 0),
  64. DECLARE_IRQ(MAX77686_PMICIRQ_PWRONR, PMIC_INT1, 1 << 1),
  65. DECLARE_IRQ(MAX77686_PMICIRQ_JIGONBF, PMIC_INT1, 1 << 2),
  66. DECLARE_IRQ(MAX77686_PMICIRQ_JIGONBR, PMIC_INT1, 1 << 3),
  67. DECLARE_IRQ(MAX77686_PMICIRQ_ACOKBF, PMIC_INT1, 1 << 4),
  68. DECLARE_IRQ(MAX77686_PMICIRQ_ACOKBR, PMIC_INT1, 1 << 5),
  69. DECLARE_IRQ(MAX77686_PMICIRQ_ONKEY1S, PMIC_INT1, 1 << 6),
  70. DECLARE_IRQ(MAX77686_PMICIRQ_MRSTB, PMIC_INT1, 1 << 7),
  71. DECLARE_IRQ(MAX77686_PMICIRQ_140C, PMIC_INT2, 1 << 0),
  72. DECLARE_IRQ(MAX77686_PMICIRQ_120C, PMIC_INT2, 1 << 1),
  73. DECLARE_IRQ(MAX77686_RTCIRQ_RTC60S, RTC_INT, 1 << 0),
  74. DECLARE_IRQ(MAX77686_RTCIRQ_RTCA1, RTC_INT, 1 << 1),
  75. DECLARE_IRQ(MAX77686_RTCIRQ_RTCA2, RTC_INT, 1 << 2),
  76. DECLARE_IRQ(MAX77686_RTCIRQ_SMPL, RTC_INT, 1 << 3),
  77. DECLARE_IRQ(MAX77686_RTCIRQ_RTC1S, RTC_INT, 1 << 4),
  78. DECLARE_IRQ(MAX77686_RTCIRQ_WTSR, RTC_INT, 1 << 5),
  79. };
  80. static void max77686_irq_lock(struct irq_data *data)
  81. {
  82. struct max77686_dev *max77686 = irq_get_chip_data(data->irq);
  83. if (debug_mask & MAX77686_DEBUG_IRQ_MASK)
  84. pr_info("%s\n", __func__);
  85. mutex_lock(&max77686->irqlock);
  86. }
  87. static void max77686_irq_sync_unlock(struct irq_data *data)
  88. {
  89. struct max77686_dev *max77686 = irq_get_chip_data(data->irq);
  90. int i;
  91. for (i = 0; i < MAX77686_IRQ_GROUP_NR; i++) {
  92. u8 mask_reg = max77686_mask_reg[i];
  93. struct regmap *map = max77686_get_regmap(max77686, i);
  94. if (debug_mask & MAX77686_DEBUG_IRQ_MASK)
  95. pr_debug("%s: mask_reg[%d]=0x%x, cur=0x%x\n",
  96. __func__, i, mask_reg, max77686->irq_masks_cur[i]);
  97. if (mask_reg == MAX77686_REG_INVALID ||
  98. IS_ERR_OR_NULL(map))
  99. continue;
  100. max77686->irq_masks_cache[i] = max77686->irq_masks_cur[i];
  101. regmap_write(map, max77686_mask_reg[i],
  102. max77686->irq_masks_cur[i]);
  103. }
  104. mutex_unlock(&max77686->irqlock);
  105. }
  106. static const inline struct max77686_irq_data *to_max77686_irq(int irq)
  107. {
  108. struct irq_data *data = irq_get_irq_data(irq);
  109. return &max77686_irqs[data->hwirq];
  110. }
  111. static void max77686_irq_mask(struct irq_data *data)
  112. {
  113. struct max77686_dev *max77686 = irq_get_chip_data(data->irq);
  114. const struct max77686_irq_data *irq_data = to_max77686_irq(data->irq);
  115. max77686->irq_masks_cur[irq_data->group] |= irq_data->mask;
  116. if (debug_mask & MAX77686_DEBUG_IRQ_MASK)
  117. pr_info("%s: group=%d, cur=0x%x\n",
  118. __func__, irq_data->group,
  119. max77686->irq_masks_cur[irq_data->group]);
  120. }
  121. static void max77686_irq_unmask(struct irq_data *data)
  122. {
  123. struct max77686_dev *max77686 = irq_get_chip_data(data->irq);
  124. const struct max77686_irq_data *irq_data = to_max77686_irq(data->irq);
  125. max77686->irq_masks_cur[irq_data->group] &= ~irq_data->mask;
  126. if (debug_mask & MAX77686_DEBUG_IRQ_MASK)
  127. pr_info("%s: group=%d, cur=0x%x\n",
  128. __func__, irq_data->group,
  129. max77686->irq_masks_cur[irq_data->group]);
  130. }
  131. static struct irq_chip max77686_irq_chip = {
  132. .name = "max77686",
  133. .irq_bus_lock = max77686_irq_lock,
  134. .irq_bus_sync_unlock = max77686_irq_sync_unlock,
  135. .irq_mask = max77686_irq_mask,
  136. .irq_unmask = max77686_irq_unmask,
  137. };
  138. static irqreturn_t max77686_irq_thread(int irq, void *data)
  139. {
  140. struct max77686_dev *max77686 = data;
  141. unsigned int irq_reg[MAX77686_IRQ_GROUP_NR] = {};
  142. unsigned int irq_src;
  143. int ret;
  144. int i, cur_irq;
  145. ret = regmap_read(max77686->regmap, MAX77686_REG_INTSRC, &irq_src);
  146. if (ret < 0) {
  147. dev_err(max77686->dev, "Failed to read interrupt source: %d\n",
  148. ret);
  149. return IRQ_NONE;
  150. }
  151. if (debug_mask & MAX77686_DEBUG_IRQ_INT)
  152. pr_info("%s: irq_src=0x%x\n", __func__, irq_src);
  153. if (irq_src == MAX77686_IRQSRC_PMIC) {
  154. ret = regmap_bulk_read(max77686->regmap,
  155. MAX77686_REG_INT1, irq_reg, 2);
  156. if (ret < 0) {
  157. dev_err(max77686->dev, "Failed to read interrupt source: %d\n",
  158. ret);
  159. return IRQ_NONE;
  160. }
  161. if (debug_mask & MAX77686_DEBUG_IRQ_INT)
  162. pr_info("%s: int1=0x%x, int2=0x%x\n", __func__,
  163. irq_reg[PMIC_INT1], irq_reg[PMIC_INT2]);
  164. }
  165. if (irq_src & MAX77686_IRQSRC_RTC) {
  166. ret = regmap_read(max77686->rtc_regmap,
  167. MAX77686_RTC_INT, &irq_reg[RTC_INT]);
  168. if (ret < 0) {
  169. dev_err(max77686->dev, "Failed to read interrupt source: %d\n",
  170. ret);
  171. return IRQ_NONE;
  172. }
  173. if (debug_mask & MAX77686_DEBUG_IRQ_INT)
  174. pr_info("%s: rtc int=0x%x\n", __func__,
  175. irq_reg[RTC_INT]);
  176. }
  177. for (i = 0; i < MAX77686_IRQ_GROUP_NR; i++)
  178. irq_reg[i] &= ~max77686->irq_masks_cur[i];
  179. for (i = 0; i < MAX77686_IRQ_NR; i++) {
  180. if (irq_reg[max77686_irqs[i].group] & max77686_irqs[i].mask) {
  181. cur_irq = irq_find_mapping(max77686->irq_domain, i);
  182. if (cur_irq)
  183. handle_nested_irq(cur_irq);
  184. }
  185. }
  186. return IRQ_HANDLED;
  187. }
  188. static int max77686_irq_domain_map(struct irq_domain *d, unsigned int irq,
  189. irq_hw_number_t hw)
  190. {
  191. struct max77686_dev *max77686 = d->host_data;
  192. irq_set_chip_data(irq, max77686);
  193. irq_set_chip_and_handler(irq, &max77686_irq_chip, handle_edge_irq);
  194. irq_set_nested_thread(irq, 1);
  195. #ifdef CONFIG_ARM
  196. set_irq_flags(irq, IRQF_VALID);
  197. #else
  198. irq_set_noprobe(irq);
  199. #endif
  200. return 0;
  201. }
  202. static struct irq_domain_ops max77686_irq_domain_ops = {
  203. .map = max77686_irq_domain_map,
  204. };
  205. int max77686_irq_init(struct max77686_dev *max77686)
  206. {
  207. struct irq_domain *domain;
  208. int i;
  209. int ret;
  210. int val;
  211. struct regmap *map;
  212. mutex_init(&max77686->irqlock);
  213. if (max77686->irq_gpio && !max77686->irq) {
  214. max77686->irq = gpio_to_irq(max77686->irq_gpio);
  215. if (debug_mask & MAX77686_DEBUG_IRQ_INT) {
  216. ret = gpio_request(max77686->irq_gpio, "pmic_irq");
  217. if (ret < 0) {
  218. dev_err(max77686->dev,
  219. "Failed to request gpio %d with ret:"
  220. "%d\n", max77686->irq_gpio, ret);
  221. return IRQ_NONE;
  222. }
  223. gpio_direction_input(max77686->irq_gpio);
  224. val = gpio_get_value(max77686->irq_gpio);
  225. gpio_free(max77686->irq_gpio);
  226. pr_info("%s: gpio_irq=%x\n", __func__, val);
  227. }
  228. }
  229. if (!max77686->irq) {
  230. dev_err(max77686->dev, "irq is not specified\n");
  231. return -ENODEV;
  232. }
  233. /* Mask individual interrupt sources */
  234. for (i = 0; i < MAX77686_IRQ_GROUP_NR; i++) {
  235. max77686->irq_masks_cur[i] = 0xff;
  236. max77686->irq_masks_cache[i] = 0xff;
  237. map = max77686_get_regmap(max77686, i);
  238. if (IS_ERR_OR_NULL(map))
  239. continue;
  240. if (max77686_mask_reg[i] == MAX77686_REG_INVALID)
  241. continue;
  242. regmap_write(map, max77686_mask_reg[i], 0xff);
  243. }
  244. domain = irq_domain_add_linear(NULL, MAX77686_IRQ_NR,
  245. &max77686_irq_domain_ops, max77686);
  246. if (!domain) {
  247. dev_err(max77686->dev, "could not create irq domain\n");
  248. return -ENODEV;
  249. }
  250. max77686->irq_domain = domain;
  251. ret = request_threaded_irq(max77686->irq, NULL, max77686_irq_thread,
  252. IRQF_TRIGGER_FALLING | IRQF_ONESHOT,
  253. "max77686-irq", max77686);
  254. if (ret)
  255. dev_err(max77686->dev, "Failed to request IRQ %d: %d\n",
  256. max77686->irq, ret);
  257. if (debug_mask & MAX77686_DEBUG_IRQ_INFO)
  258. pr_info("%s-\n", __func__);
  259. return 0;
  260. }
  261. void max77686_irq_exit(struct max77686_dev *max77686)
  262. {
  263. if (max77686->irq)
  264. free_irq(max77686->irq, max77686);
  265. }