mcount.S 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171
  1. /*
  2. * Low-level ftrace handling
  3. *
  4. * Copyright (C) 2009 Michal Simek <monstr@monstr.eu>
  5. * Copyright (C) 2009 PetaLogix
  6. *
  7. * This file is subject to the terms and conditions of the GNU General
  8. * Public License. See the file COPYING in the main directory of this
  9. * archive for more details.
  10. */
  11. #include <linux/linkage.h>
  12. #define NOALIGN_ENTRY(name) .globl name; name:
  13. /* FIXME MS: I think that I don't need to save all regs */
  14. #define SAVE_REGS \
  15. addik r1, r1, -120; \
  16. swi r2, r1, 4; \
  17. swi r3, r1, 8; \
  18. swi r4, r1, 12; \
  19. swi r5, r1, 116; \
  20. swi r6, r1, 16; \
  21. swi r7, r1, 20; \
  22. swi r8, r1, 24; \
  23. swi r9, r1, 28; \
  24. swi r10, r1, 32; \
  25. swi r11, r1, 36; \
  26. swi r12, r1, 40; \
  27. swi r13, r1, 44; \
  28. swi r14, r1, 48; \
  29. swi r16, r1, 52; \
  30. swi r17, r1, 56; \
  31. swi r18, r1, 60; \
  32. swi r19, r1, 64; \
  33. swi r20, r1, 68; \
  34. swi r21, r1, 72; \
  35. swi r22, r1, 76; \
  36. swi r23, r1, 80; \
  37. swi r24, r1, 84; \
  38. swi r25, r1, 88; \
  39. swi r26, r1, 92; \
  40. swi r27, r1, 96; \
  41. swi r28, r1, 100; \
  42. swi r29, r1, 104; \
  43. swi r30, r1, 108; \
  44. swi r31, r1, 112;
  45. #define RESTORE_REGS \
  46. lwi r2, r1, 4; \
  47. lwi r3, r1, 8; \
  48. lwi r4, r1, 12; \
  49. lwi r5, r1, 116; \
  50. lwi r6, r1, 16; \
  51. lwi r7, r1, 20; \
  52. lwi r8, r1, 24; \
  53. lwi r9, r1, 28; \
  54. lwi r10, r1, 32; \
  55. lwi r11, r1, 36; \
  56. lwi r12, r1, 40; \
  57. lwi r13, r1, 44; \
  58. lwi r14, r1, 48; \
  59. lwi r16, r1, 52; \
  60. lwi r17, r1, 56; \
  61. lwi r18, r1, 60; \
  62. lwi r19, r1, 64; \
  63. lwi r20, r1, 68; \
  64. lwi r21, r1, 72; \
  65. lwi r22, r1, 76; \
  66. lwi r23, r1, 80; \
  67. lwi r24, r1, 84; \
  68. lwi r25, r1, 88; \
  69. lwi r26, r1, 92; \
  70. lwi r27, r1, 96; \
  71. lwi r28, r1, 100; \
  72. lwi r29, r1, 104; \
  73. lwi r30, r1, 108; \
  74. lwi r31, r1, 112; \
  75. addik r1, r1, 120;
  76. ENTRY(ftrace_stub)
  77. rtsd r15, 8;
  78. nop;
  79. ENTRY(_mcount)
  80. #ifdef CONFIG_DYNAMIC_FTRACE
  81. ENTRY(ftrace_caller)
  82. /* MS: It is just barrier which is removed from C code */
  83. rtsd r15, 8
  84. nop
  85. #endif /* CONFIG_DYNAMIC_FTRACE */
  86. SAVE_REGS
  87. swi r15, r1, 0;
  88. /* MS: HAVE_FUNCTION_TRACE_MCOUNT_TEST begin of checking */
  89. lwi r5, r0, function_trace_stop;
  90. bneid r5, end;
  91. nop;
  92. /* MS: HAVE_FUNCTION_TRACE_MCOUNT_TEST end of checking */
  93. #ifdef CONFIG_FUNCTION_GRAPH_TRACER
  94. #ifndef CONFIG_DYNAMIC_FTRACE
  95. lwi r5, r0, ftrace_graph_return;
  96. addik r6, r0, ftrace_stub; /* asm implementation */
  97. cmpu r5, r5, r6; /* ftrace_graph_return != ftrace_stub */
  98. beqid r5, end_graph_tracer;
  99. nop;
  100. lwi r6, r0, ftrace_graph_entry;
  101. addik r5, r0, ftrace_graph_entry_stub; /* implemented in C */
  102. cmpu r5, r5, r6; /* ftrace_graph_entry != ftrace_graph_entry_stub */
  103. beqid r5, end_graph_tracer;
  104. nop;
  105. #else /* CONFIG_DYNAMIC_FTRACE */
  106. NOALIGN_ENTRY(ftrace_call_graph)
  107. /* MS: jump over graph function - replaced from C code */
  108. bri end_graph_tracer
  109. #endif /* CONFIG_DYNAMIC_FTRACE */
  110. addik r5, r1, 120; /* MS: load parent addr */
  111. addik r6, r15, 0; /* MS: load current function addr */
  112. bralid r15, prepare_ftrace_return;
  113. nop;
  114. /* MS: graph was taken that's why - can jump over function trace */
  115. brid end;
  116. nop;
  117. end_graph_tracer:
  118. #endif /* CONFIG_FUNCTION_GRAPH_TRACER */
  119. #ifndef CONFIG_DYNAMIC_FTRACE
  120. /* MS: test function trace if is taken or not */
  121. lwi r20, r0, ftrace_trace_function;
  122. addik r6, r0, ftrace_stub;
  123. cmpu r5, r20, r6; /* ftrace_trace_function != ftrace_stub */
  124. beqid r5, end; /* MS: not taken -> jump over */
  125. nop;
  126. #else /* CONFIG_DYNAMIC_FTRACE */
  127. NOALIGN_ENTRY(ftrace_call)
  128. /* instruction for setup imm FUNC_part1, addik r20, r0, FUNC_part2 */
  129. nop
  130. nop
  131. #endif /* CONFIG_DYNAMIC_FTRACE */
  132. /* static normal trace */
  133. lwi r6, r1, 120; /* MS: load parent addr */
  134. addik r5, r15, 0; /* MS: load current function addr */
  135. /* MS: here is dependency on previous code */
  136. brald r15, r20; /* MS: jump to ftrace handler */
  137. nop;
  138. end:
  139. lwi r15, r1, 0;
  140. RESTORE_REGS
  141. rtsd r15, 8; /* MS: jump back */
  142. nop;
  143. #ifdef CONFIG_FUNCTION_GRAPH_TRACER
  144. ENTRY(return_to_handler)
  145. nop; /* MS: just barrier for rtsd r15, 8 */
  146. nop;
  147. SAVE_REGS
  148. swi r15, r1, 0;
  149. /* MS: find out returning address */
  150. bralid r15, ftrace_return_to_handler;
  151. nop;
  152. /* MS: return value from ftrace_return_to_handler is my returning addr
  153. * must be before restore regs because I have to restore r3 content */
  154. addik r15, r3, 0;
  155. RESTORE_REGS
  156. rtsd r15, 8; /* MS: jump back */
  157. nop;
  158. #endif /* CONFIG_FUNCTION_TRACER */