sysrq.c 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Magic System Request Key Hacks
  9. *
  10. */
  11. #include <kernel.h>
  12. #include <drivers/uart.h>
  13. #include <soc.h>
  14. struct sysrq_key_op {
  15. char key;
  16. char reserved[3];
  17. void (*handler)(int);
  18. char *help_msg;
  19. char *action_msg;
  20. };
  21. #if CONFIG_KERNEL_SHOW_STACK
  22. extern void show_stack(void);
  23. static void sysrq_handle_show_stack(int key)
  24. {
  25. show_stack();
  26. }
  27. #endif
  28. static void sysrq_handle_change_jtag(int key)
  29. {
  30. jtag_set();
  31. }
  32. #if defined(CONFIG_INIT_STACKS) && defined(CONFIG_THREAD_STACK_INFO) && defined(CONFIG_THREAD_MONITOR)
  33. static void sysrq_stack_dump(const struct k_thread *thread, void *user_data)
  34. {
  35. unsigned int pcnt;
  36. size_t unused;
  37. size_t size = thread->stack_info.size;
  38. const char *tname;
  39. int ret;
  40. ret = k_thread_stack_space_get(thread, &unused);
  41. if (ret) {
  42. printk("Unable to determine unused stack size (%d)\n", ret);
  43. return;
  44. }
  45. tname = k_thread_name_get((struct k_thread *)thread);
  46. /* Calculate the real size reserved for the stack */
  47. pcnt = ((size - unused) * 100U) / size;
  48. printk("%p %-10s (real size %u):\tunused %u\tusage %u / %u (%u %%)\n",
  49. thread,
  50. tname ? tname : "NA",
  51. size, unused, size - unused, size, pcnt);
  52. }
  53. static void sysrq_handle_stack_dump(int key)
  54. {
  55. k_thread_foreach(sysrq_stack_dump, NULL);
  56. }
  57. #endif
  58. static const struct sysrq_key_op sysrq_key_table[] = {
  59. #ifdef CONFIG_KERNEL_SHOW_STACK
  60. {
  61. .key = 't',
  62. .handler = sysrq_handle_show_stack,
  63. .help_msg = "show-thread-states(t)",
  64. .action_msg = "Show State:",
  65. },
  66. #endif
  67. {
  68. .key = 'j',
  69. .handler = sysrq_handle_change_jtag,
  70. .help_msg = "change jtag to grp1 (j)",
  71. .action_msg = "jtag:",
  72. },
  73. #if defined(CONFIG_INIT_STACKS) && defined(CONFIG_THREAD_STACK_INFO) && defined(CONFIG_THREAD_MONITOR)
  74. {
  75. .key = 's',
  76. .handler = sysrq_handle_stack_dump,
  77. .help_msg = "show thread stack usage (s)",
  78. .action_msg = "stack usage:",
  79. },
  80. #endif
  81. };
  82. /* magic sysrq key: CTLR + 'b' 'r' 'e' 'a' 'k' */
  83. static const char sysrq_breakbuf[] = {0x02, 0x12, 0x05, 0x01, 0x0b};
  84. static int sysrq_break_idx;
  85. static struct sysrq_key_op* sysrq_get_key_op(char key)
  86. {
  87. int i, cnt;
  88. cnt = ARRAY_SIZE(sysrq_key_table);
  89. for (i = 0; i < cnt; i++) {
  90. if (key == sysrq_key_table[i].key) {
  91. return (struct sysrq_key_op*)&sysrq_key_table[i];
  92. }
  93. }
  94. return NULL;
  95. }
  96. static void sysrq_print_help(void)
  97. {
  98. int i;
  99. printk("HELP : quit(q) \n");
  100. for (i = 0; i < ARRAY_SIZE(sysrq_key_table); i++) {
  101. printk("%s \n", sysrq_key_table[i].help_msg);
  102. }
  103. printk("\n");
  104. }
  105. extern void printk_dma_switch(int sw_dma);
  106. static void handle_sysrq_main(const struct device * port)
  107. {
  108. struct sysrq_key_op* op_p;
  109. char key;
  110. #ifdef CONFIG_ACTIONS_PRINTK_DMA
  111. printk_dma_switch(0);
  112. #endif
  113. for(;;) {
  114. printk("Wait SYSRQ Key:\n");
  115. sysrq_print_help();
  116. uart_poll_in(port, &key);
  117. if (key == 'q') {
  118. printk("sysrq: exit\n");
  119. break;
  120. }
  121. op_p = sysrq_get_key_op(key);
  122. if (op_p) {
  123. printk("%s\n", op_p->action_msg);
  124. op_p->handler(key);
  125. } else {
  126. sysrq_print_help();
  127. }
  128. }
  129. #ifdef CONFIG_ACTIONS_PRINTK_DMA
  130. printk_dma_switch(1);
  131. #endif
  132. }
  133. void uart_handle_sysrq_char(const struct device * port, char c)
  134. {
  135. if (c == sysrq_breakbuf[sysrq_break_idx]) {
  136. sysrq_break_idx++;
  137. if (sysrq_break_idx == sizeof(sysrq_breakbuf)) {
  138. sysrq_break_idx = 0;
  139. handle_sysrq_main(port);
  140. }
  141. } else {
  142. sysrq_break_idx = 0;
  143. }
  144. }