diff options
Diffstat (limited to 'kernel/trace/trace_kprobe.c')
-rw-r--r-- | kernel/trace/trace_kprobe.c | 42 |
1 files changed, 19 insertions, 23 deletions
diff --git a/kernel/trace/trace_kprobe.c b/kernel/trace/trace_kprobe.c index cbc0870dcf5d..ea0db8eee570 100644 --- a/kernel/trace/trace_kprobe.c +++ b/kernel/trace/trace_kprobe.c | |||
@@ -347,20 +347,15 @@ static struct trace_probe *find_probe_event(const char *event) | |||
347 | return NULL; | 347 | return NULL; |
348 | } | 348 | } |
349 | 349 | ||
350 | static void __unregister_trace_probe(struct trace_probe *tp) | 350 | /* Unregister a trace_probe and probe_event: call with locking probe_lock */ |
351 | static void unregister_trace_probe(struct trace_probe *tp) | ||
351 | { | 352 | { |
352 | if (probe_is_return(tp)) | 353 | if (probe_is_return(tp)) |
353 | unregister_kretprobe(&tp->rp); | 354 | unregister_kretprobe(&tp->rp); |
354 | else | 355 | else |
355 | unregister_kprobe(&tp->rp.kp); | 356 | unregister_kprobe(&tp->rp.kp); |
356 | } | ||
357 | |||
358 | /* Unregister a trace_probe and probe_event: call with locking probe_lock */ | ||
359 | static void unregister_trace_probe(struct trace_probe *tp) | ||
360 | { | ||
361 | unregister_probe_event(tp); | ||
362 | __unregister_trace_probe(tp); | ||
363 | list_del(&tp->list); | 357 | list_del(&tp->list); |
358 | unregister_probe_event(tp); | ||
364 | } | 359 | } |
365 | 360 | ||
366 | /* Register a trace_probe and probe_event */ | 361 | /* Register a trace_probe and probe_event */ |
@@ -371,6 +366,19 @@ static int register_trace_probe(struct trace_probe *tp) | |||
371 | 366 | ||
372 | mutex_lock(&probe_lock); | 367 | mutex_lock(&probe_lock); |
373 | 368 | ||
369 | /* register as an event */ | ||
370 | old_tp = find_probe_event(tp->call.name); | ||
371 | if (old_tp) { | ||
372 | /* delete old event */ | ||
373 | unregister_trace_probe(old_tp); | ||
374 | free_trace_probe(old_tp); | ||
375 | } | ||
376 | ret = register_probe_event(tp); | ||
377 | if (ret) { | ||
378 | pr_warning("Faild to register probe event(%d)\n", ret); | ||
379 | goto end; | ||
380 | } | ||
381 | |||
374 | if (probe_is_return(tp)) | 382 | if (probe_is_return(tp)) |
375 | ret = register_kretprobe(&tp->rp); | 383 | ret = register_kretprobe(&tp->rp); |
376 | else | 384 | else |
@@ -384,21 +392,9 @@ static int register_trace_probe(struct trace_probe *tp) | |||
384 | tp->rp.kp.addr); | 392 | tp->rp.kp.addr); |
385 | ret = -EINVAL; | 393 | ret = -EINVAL; |
386 | } | 394 | } |
387 | goto end; | 395 | unregister_probe_event(tp); |
388 | } | 396 | } else |
389 | /* register as an event */ | 397 | list_add_tail(&tp->list, &probe_list); |
390 | old_tp = find_probe_event(tp->call.name); | ||
391 | if (old_tp) { | ||
392 | /* delete old event */ | ||
393 | unregister_trace_probe(old_tp); | ||
394 | free_trace_probe(old_tp); | ||
395 | } | ||
396 | ret = register_probe_event(tp); | ||
397 | if (ret) { | ||
398 | pr_warning("Faild to register probe event(%d)\n", ret); | ||
399 | __unregister_trace_probe(tp); | ||
400 | } | ||
401 | list_add_tail(&tp->list, &probe_list); | ||
402 | end: | 398 | end: |
403 | mutex_unlock(&probe_lock); | 399 | mutex_unlock(&probe_lock); |
404 | return ret; | 400 | return ret; |