aboutsummaryrefslogtreecommitdiffstats
path: root/kernel
diff options
context:
space:
mode:
Diffstat (limited to 'kernel')
-rw-r--r--kernel/trace/ftrace.c95
1 files changed, 34 insertions, 61 deletions
diff --git a/kernel/trace/ftrace.c b/kernel/trace/ftrace.c
index 5b758ea344ce..33bcc71ca09a 100644
--- a/kernel/trace/ftrace.c
+++ b/kernel/trace/ftrace.c
@@ -151,6 +151,34 @@ static void ftrace_test_stop_func(unsigned long ip, unsigned long parent_ip)
151} 151}
152#endif 152#endif
153 153
154static void update_ftrace_function(void)
155{
156 ftrace_func_t func;
157
158 /*
159 * If there's only one function registered, then call that
160 * function directly. Otherwise, we need to iterate over the
161 * registered callers.
162 */
163 if (ftrace_list == &ftrace_list_end ||
164 ftrace_list->next == &ftrace_list_end)
165 func = ftrace_list->func;
166 else
167 func = ftrace_list_func;
168
169 /* If we filter on pids, update to use the pid function */
170 if (!list_empty(&ftrace_pids)) {
171 set_ftrace_pid_function(func);
172 func = ftrace_pid_func;
173 }
174#ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
175 ftrace_trace_function = func;
176#else
177 __ftrace_trace_function = func;
178 ftrace_trace_function = ftrace_test_stop_func;
179#endif
180}
181
154static int __register_ftrace_function(struct ftrace_ops *ops) 182static int __register_ftrace_function(struct ftrace_ops *ops)
155{ 183{
156 ops->next = ftrace_list; 184 ops->next = ftrace_list;
@@ -162,30 +190,8 @@ static int __register_ftrace_function(struct ftrace_ops *ops)
162 */ 190 */
163 rcu_assign_pointer(ftrace_list, ops); 191 rcu_assign_pointer(ftrace_list, ops);
164 192
165 if (ftrace_enabled) { 193 if (ftrace_enabled)
166 ftrace_func_t func; 194 update_ftrace_function();
167
168 if (ops->next == &ftrace_list_end)
169 func = ops->func;
170 else
171 func = ftrace_list_func;
172
173 if (!list_empty(&ftrace_pids)) {
174 set_ftrace_pid_function(func);
175 func = ftrace_pid_func;
176 }
177
178 /*
179 * For one func, simply call it directly.
180 * For more than one func, call the chain.
181 */
182#ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
183 ftrace_trace_function = func;
184#else
185 __ftrace_trace_function = func;
186 ftrace_trace_function = ftrace_test_stop_func;
187#endif
188 }
189 195
190 return 0; 196 return 0;
191} 197}
@@ -213,52 +219,19 @@ static int __unregister_ftrace_function(struct ftrace_ops *ops)
213 219
214 *p = (*p)->next; 220 *p = (*p)->next;
215 221
216 if (ftrace_enabled) { 222 if (ftrace_enabled)
217 /* If we only have one func left, then call that directly */ 223 update_ftrace_function();
218 if (ftrace_list->next == &ftrace_list_end) {
219 ftrace_func_t func = ftrace_list->func;
220
221 if (!list_empty(&ftrace_pids)) {
222 set_ftrace_pid_function(func);
223 func = ftrace_pid_func;
224 }
225#ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
226 ftrace_trace_function = func;
227#else
228 __ftrace_trace_function = func;
229#endif
230 }
231 }
232 224
233 return 0; 225 return 0;
234} 226}
235 227
236static void ftrace_update_pid_func(void) 228static void ftrace_update_pid_func(void)
237{ 229{
238 ftrace_func_t func; 230 /* Only do something if we are tracing something */
239
240 if (ftrace_trace_function == ftrace_stub) 231 if (ftrace_trace_function == ftrace_stub)
241 return; 232 return;
242 233
243#ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST 234 update_ftrace_function();
244 func = ftrace_trace_function;
245#else
246 func = __ftrace_trace_function;
247#endif
248
249 if (!list_empty(&ftrace_pids)) {
250 set_ftrace_pid_function(func);
251 func = ftrace_pid_func;
252 } else {
253 if (func == ftrace_pid_func)
254 func = ftrace_pid_function;
255 }
256
257#ifdef CONFIG_HAVE_FUNCTION_TRACE_MCOUNT_TEST
258 ftrace_trace_function = func;
259#else
260 __ftrace_trace_function = func;
261#endif
262} 235}
263 236
264#ifdef CONFIG_FUNCTION_PROFILER 237#ifdef CONFIG_FUNCTION_PROFILER