modem_notifier.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. /* Copyright (c) 2008-2010, 2012, The Linux Foundation. All rights reserved.
  2. *
  3. * This program is free software; you can redistribute it and/or modify
  4. * it under the terms of the GNU General Public License version 2 and
  5. * only version 2 as published by the Free Software Foundation.
  6. *
  7. * This program is distributed in the hope that it will be useful,
  8. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. * GNU General Public License for more details.
  11. *
  12. */
  13. /*
  14. * Modem Restart Notifier -- Provides notification
  15. * of modem restart events.
  16. */
  17. #include <linux/notifier.h>
  18. #include <linux/init.h>
  19. #include <linux/debugfs.h>
  20. #include <linux/module.h>
  21. #include <linux/workqueue.h>
  22. #include "modem_notifier.h"
  23. #define DEBUG
  24. static struct srcu_notifier_head modem_notifier_list;
  25. static struct workqueue_struct *modem_notifier_wq;
  26. static void notify_work_smsm_init(struct work_struct *work)
  27. {
  28. modem_notify(0, MODEM_NOTIFIER_SMSM_INIT);
  29. }
  30. static DECLARE_WORK(modem_notifier_smsm_init_work, &notify_work_smsm_init);
  31. void modem_queue_smsm_init_notify(void)
  32. {
  33. int ret;
  34. ret = queue_work(modem_notifier_wq, &modem_notifier_smsm_init_work);
  35. if (!ret)
  36. printk(KERN_ERR "%s\n", __func__);
  37. }
  38. EXPORT_SYMBOL(modem_queue_smsm_init_notify);
  39. static void notify_work_start_reset(struct work_struct *work)
  40. {
  41. modem_notify(0, MODEM_NOTIFIER_START_RESET);
  42. }
  43. static DECLARE_WORK(modem_notifier_start_reset_work, &notify_work_start_reset);
  44. void modem_queue_start_reset_notify(void)
  45. {
  46. int ret;
  47. ret = queue_work(modem_notifier_wq, &modem_notifier_start_reset_work);
  48. if (!ret)
  49. printk(KERN_ERR "%s\n", __func__);
  50. }
  51. EXPORT_SYMBOL(modem_queue_start_reset_notify);
  52. static void notify_work_end_reset(struct work_struct *work)
  53. {
  54. modem_notify(0, MODEM_NOTIFIER_END_RESET);
  55. }
  56. static DECLARE_WORK(modem_notifier_end_reset_work, &notify_work_end_reset);
  57. void modem_queue_end_reset_notify(void)
  58. {
  59. int ret;
  60. ret = queue_work(modem_notifier_wq, &modem_notifier_end_reset_work);
  61. if (!ret)
  62. printk(KERN_ERR "%s\n", __func__);
  63. }
  64. EXPORT_SYMBOL(modem_queue_end_reset_notify);
  65. int modem_register_notifier(struct notifier_block *nb)
  66. {
  67. int ret;
  68. ret = srcu_notifier_chain_register(
  69. &modem_notifier_list, nb);
  70. return ret;
  71. }
  72. EXPORT_SYMBOL(modem_register_notifier);
  73. int modem_unregister_notifier(struct notifier_block *nb)
  74. {
  75. int ret;
  76. ret = srcu_notifier_chain_unregister(
  77. &modem_notifier_list, nb);
  78. return ret;
  79. }
  80. EXPORT_SYMBOL(modem_unregister_notifier);
  81. void modem_notify(void *data, unsigned int state)
  82. {
  83. srcu_notifier_call_chain(&modem_notifier_list, state, data);
  84. }
  85. EXPORT_SYMBOL(modem_notify);
  86. #if defined(CONFIG_DEBUG_FS)
  87. static int debug_reset_start(const char __user *buf, int count)
  88. {
  89. modem_queue_start_reset_notify();
  90. return 0;
  91. }
  92. static int debug_reset_end(const char __user *buf, int count)
  93. {
  94. modem_queue_end_reset_notify();
  95. return 0;
  96. }
  97. static ssize_t debug_write(struct file *file, const char __user *buf,
  98. size_t count, loff_t *ppos)
  99. {
  100. int (*fling)(const char __user *buf, int max) = file->private_data;
  101. fling(buf, count);
  102. return count;
  103. }
  104. static int debug_open(struct inode *inode, struct file *file)
  105. {
  106. file->private_data = inode->i_private;
  107. return 0;
  108. }
  109. static const struct file_operations debug_ops = {
  110. .write = debug_write,
  111. .open = debug_open,
  112. };
  113. static void debug_create(const char *name, mode_t mode,
  114. struct dentry *dent,
  115. int (*fling)(const char __user *buf, int max))
  116. {
  117. debugfs_create_file(name, mode, dent, fling, &debug_ops);
  118. }
  119. static void modem_notifier_debugfs_init(void)
  120. {
  121. struct dentry *dent;
  122. dent = debugfs_create_dir("modem_notifier", 0);
  123. if (IS_ERR(dent))
  124. return;
  125. debug_create("reset_start", 0444, dent, debug_reset_start);
  126. debug_create("reset_end", 0444, dent, debug_reset_end);
  127. }
  128. #else
  129. static void modem_notifier_debugfs_init(void) {}
  130. #endif
  131. #if defined(DEBUG)
  132. static int modem_notifier_test_call(struct notifier_block *this,
  133. unsigned long code,
  134. void *_cmd)
  135. {
  136. switch (code) {
  137. case MODEM_NOTIFIER_START_RESET:
  138. printk(KERN_ERR "Notify: start reset\n");
  139. break;
  140. case MODEM_NOTIFIER_END_RESET:
  141. printk(KERN_ERR "Notify: end reset\n");
  142. break;
  143. case MODEM_NOTIFIER_SMSM_INIT:
  144. printk(KERN_ERR "Notify: smsm init\n");
  145. break;
  146. default:
  147. printk(KERN_ERR "Notify: general\n");
  148. break;
  149. }
  150. return NOTIFY_DONE;
  151. }
  152. static struct notifier_block nb = {
  153. .notifier_call = modem_notifier_test_call,
  154. };
  155. static void register_test_notifier(void)
  156. {
  157. modem_register_notifier(&nb);
  158. }
  159. #endif
  160. int __init msm_init_modem_notifier_list(void)
  161. {
  162. static bool registered;
  163. if (registered)
  164. return 0;
  165. registered = true;
  166. srcu_init_notifier_head(&modem_notifier_list);
  167. modem_notifier_debugfs_init();
  168. #if defined(DEBUG)
  169. register_test_notifier();
  170. #endif
  171. /* Create the workqueue */
  172. modem_notifier_wq = create_singlethread_workqueue("modem_notifier");
  173. if (!modem_notifier_wq) {
  174. srcu_cleanup_notifier_head(&modem_notifier_list);
  175. return -ENOMEM;
  176. }
  177. return 0;
  178. }
  179. module_init(msm_init_modem_notifier_list);