diff options
Diffstat (limited to 'kernel/trace/trace_irqsoff.c')
| -rw-r--r-- | kernel/trace/trace_irqsoff.c | 61 |
1 files changed, 40 insertions, 21 deletions
diff --git a/kernel/trace/trace_irqsoff.c b/kernel/trace/trace_irqsoff.c index 9c74071c10e0..7c2e326bbc8b 100644 --- a/kernel/trace/trace_irqsoff.c +++ b/kernel/trace/trace_irqsoff.c | |||
| @@ -353,15 +353,28 @@ void trace_preempt_off(unsigned long a0, unsigned long a1) | |||
| 353 | } | 353 | } |
| 354 | #endif /* CONFIG_PREEMPT_TRACER */ | 354 | #endif /* CONFIG_PREEMPT_TRACER */ |
| 355 | 355 | ||
| 356 | /* | ||
| 357 | * save_tracer_enabled is used to save the state of the tracer_enabled | ||
| 358 | * variable when we disable it when we open a trace output file. | ||
| 359 | */ | ||
| 360 | static int save_tracer_enabled; | ||
| 361 | |||
| 356 | static void start_irqsoff_tracer(struct trace_array *tr) | 362 | static void start_irqsoff_tracer(struct trace_array *tr) |
| 357 | { | 363 | { |
| 358 | register_ftrace_function(&trace_ops); | 364 | register_ftrace_function(&trace_ops); |
| 359 | tracer_enabled = 1; | 365 | if (tracing_is_enabled()) { |
| 366 | tracer_enabled = 1; | ||
| 367 | save_tracer_enabled = 1; | ||
| 368 | } else { | ||
| 369 | tracer_enabled = 0; | ||
| 370 | save_tracer_enabled = 0; | ||
| 371 | } | ||
| 360 | } | 372 | } |
| 361 | 373 | ||
| 362 | static void stop_irqsoff_tracer(struct trace_array *tr) | 374 | static void stop_irqsoff_tracer(struct trace_array *tr) |
| 363 | { | 375 | { |
| 364 | tracer_enabled = 0; | 376 | tracer_enabled = 0; |
| 377 | save_tracer_enabled = 0; | ||
| 365 | unregister_ftrace_function(&trace_ops); | 378 | unregister_ftrace_function(&trace_ops); |
| 366 | } | 379 | } |
| 367 | 380 | ||
| @@ -370,53 +383,55 @@ static void __irqsoff_tracer_init(struct trace_array *tr) | |||
| 370 | irqsoff_trace = tr; | 383 | irqsoff_trace = tr; |
| 371 | /* make sure that the tracer is visible */ | 384 | /* make sure that the tracer is visible */ |
| 372 | smp_wmb(); | 385 | smp_wmb(); |
| 373 | 386 | start_irqsoff_tracer(tr); | |
| 374 | if (tr->ctrl) | ||
| 375 | start_irqsoff_tracer(tr); | ||
| 376 | } | 387 | } |
| 377 | 388 | ||
| 378 | static void irqsoff_tracer_reset(struct trace_array *tr) | 389 | static void irqsoff_tracer_reset(struct trace_array *tr) |
| 379 | { | 390 | { |
| 380 | if (tr->ctrl) | 391 | stop_irqsoff_tracer(tr); |
| 381 | stop_irqsoff_tracer(tr); | ||
| 382 | } | 392 | } |
| 383 | 393 | ||
| 384 | static void irqsoff_tracer_ctrl_update(struct trace_array *tr) | 394 | static void irqsoff_tracer_start(struct trace_array *tr) |
| 385 | { | 395 | { |
| 386 | if (tr->ctrl) | 396 | tracer_enabled = 1; |
| 387 | start_irqsoff_tracer(tr); | 397 | save_tracer_enabled = 1; |
| 388 | else | 398 | } |
| 389 | stop_irqsoff_tracer(tr); | 399 | |
| 400 | static void irqsoff_tracer_stop(struct trace_array *tr) | ||
| 401 | { | ||
| 402 | tracer_enabled = 0; | ||
| 403 | save_tracer_enabled = 0; | ||
| 390 | } | 404 | } |
| 391 | 405 | ||
| 392 | static void irqsoff_tracer_open(struct trace_iterator *iter) | 406 | static void irqsoff_tracer_open(struct trace_iterator *iter) |
| 393 | { | 407 | { |
| 394 | /* stop the trace while dumping */ | 408 | /* stop the trace while dumping */ |
| 395 | if (iter->tr->ctrl) | 409 | tracer_enabled = 0; |
| 396 | stop_irqsoff_tracer(iter->tr); | ||
| 397 | } | 410 | } |
| 398 | 411 | ||
| 399 | static void irqsoff_tracer_close(struct trace_iterator *iter) | 412 | static void irqsoff_tracer_close(struct trace_iterator *iter) |
| 400 | { | 413 | { |
| 401 | if (iter->tr->ctrl) | 414 | /* restart tracing */ |
| 402 | start_irqsoff_tracer(iter->tr); | 415 | tracer_enabled = save_tracer_enabled; |
| 403 | } | 416 | } |
| 404 | 417 | ||
| 405 | #ifdef CONFIG_IRQSOFF_TRACER | 418 | #ifdef CONFIG_IRQSOFF_TRACER |
| 406 | static void irqsoff_tracer_init(struct trace_array *tr) | 419 | static int irqsoff_tracer_init(struct trace_array *tr) |
| 407 | { | 420 | { |
| 408 | trace_type = TRACER_IRQS_OFF; | 421 | trace_type = TRACER_IRQS_OFF; |
| 409 | 422 | ||
| 410 | __irqsoff_tracer_init(tr); | 423 | __irqsoff_tracer_init(tr); |
| 424 | return 0; | ||
| 411 | } | 425 | } |
| 412 | static struct tracer irqsoff_tracer __read_mostly = | 426 | static struct tracer irqsoff_tracer __read_mostly = |
| 413 | { | 427 | { |
| 414 | .name = "irqsoff", | 428 | .name = "irqsoff", |
| 415 | .init = irqsoff_tracer_init, | 429 | .init = irqsoff_tracer_init, |
| 416 | .reset = irqsoff_tracer_reset, | 430 | .reset = irqsoff_tracer_reset, |
| 431 | .start = irqsoff_tracer_start, | ||
| 432 | .stop = irqsoff_tracer_stop, | ||
| 417 | .open = irqsoff_tracer_open, | 433 | .open = irqsoff_tracer_open, |
| 418 | .close = irqsoff_tracer_close, | 434 | .close = irqsoff_tracer_close, |
| 419 | .ctrl_update = irqsoff_tracer_ctrl_update, | ||
| 420 | .print_max = 1, | 435 | .print_max = 1, |
| 421 | #ifdef CONFIG_FTRACE_SELFTEST | 436 | #ifdef CONFIG_FTRACE_SELFTEST |
| 422 | .selftest = trace_selftest_startup_irqsoff, | 437 | .selftest = trace_selftest_startup_irqsoff, |
| @@ -428,11 +443,12 @@ static struct tracer irqsoff_tracer __read_mostly = | |||
| 428 | #endif | 443 | #endif |
| 429 | 444 | ||
| 430 | #ifdef CONFIG_PREEMPT_TRACER | 445 | #ifdef CONFIG_PREEMPT_TRACER |
| 431 | static void preemptoff_tracer_init(struct trace_array *tr) | 446 | static int preemptoff_tracer_init(struct trace_array *tr) |
| 432 | { | 447 | { |
| 433 | trace_type = TRACER_PREEMPT_OFF; | 448 | trace_type = TRACER_PREEMPT_OFF; |
| 434 | 449 | ||
| 435 | __irqsoff_tracer_init(tr); | 450 | __irqsoff_tracer_init(tr); |
| 451 | return 0; | ||
| 436 | } | 452 | } |
| 437 | 453 | ||
| 438 | static struct tracer preemptoff_tracer __read_mostly = | 454 | static struct tracer preemptoff_tracer __read_mostly = |
| @@ -440,9 +456,10 @@ static struct tracer preemptoff_tracer __read_mostly = | |||
| 440 | .name = "preemptoff", | 456 | .name = "preemptoff", |
| 441 | .init = preemptoff_tracer_init, | 457 | .init = preemptoff_tracer_init, |
| 442 | .reset = irqsoff_tracer_reset, | 458 | .reset = irqsoff_tracer_reset, |
| 459 | .start = irqsoff_tracer_start, | ||
| 460 | .stop = irqsoff_tracer_stop, | ||
| 443 | .open = irqsoff_tracer_open, | 461 | .open = irqsoff_tracer_open, |
| 444 | .close = irqsoff_tracer_close, | 462 | .close = irqsoff_tracer_close, |
| 445 | .ctrl_update = irqsoff_tracer_ctrl_update, | ||
| 446 | .print_max = 1, | 463 | .print_max = 1, |
| 447 | #ifdef CONFIG_FTRACE_SELFTEST | 464 | #ifdef CONFIG_FTRACE_SELFTEST |
| 448 | .selftest = trace_selftest_startup_preemptoff, | 465 | .selftest = trace_selftest_startup_preemptoff, |
| @@ -456,11 +473,12 @@ static struct tracer preemptoff_tracer __read_mostly = | |||
| 456 | #if defined(CONFIG_IRQSOFF_TRACER) && \ | 473 | #if defined(CONFIG_IRQSOFF_TRACER) && \ |
| 457 | defined(CONFIG_PREEMPT_TRACER) | 474 | defined(CONFIG_PREEMPT_TRACER) |
| 458 | 475 | ||
| 459 | static void preemptirqsoff_tracer_init(struct trace_array *tr) | 476 | static int preemptirqsoff_tracer_init(struct trace_array *tr) |
| 460 | { | 477 | { |
| 461 | trace_type = TRACER_IRQS_OFF | TRACER_PREEMPT_OFF; | 478 | trace_type = TRACER_IRQS_OFF | TRACER_PREEMPT_OFF; |
| 462 | 479 | ||
| 463 | __irqsoff_tracer_init(tr); | 480 | __irqsoff_tracer_init(tr); |
| 481 | return 0; | ||
| 464 | } | 482 | } |
| 465 | 483 | ||
| 466 | static struct tracer preemptirqsoff_tracer __read_mostly = | 484 | static struct tracer preemptirqsoff_tracer __read_mostly = |
| @@ -468,9 +486,10 @@ static struct tracer preemptirqsoff_tracer __read_mostly = | |||
| 468 | .name = "preemptirqsoff", | 486 | .name = "preemptirqsoff", |
| 469 | .init = preemptirqsoff_tracer_init, | 487 | .init = preemptirqsoff_tracer_init, |
| 470 | .reset = irqsoff_tracer_reset, | 488 | .reset = irqsoff_tracer_reset, |
| 489 | .start = irqsoff_tracer_start, | ||
| 490 | .stop = irqsoff_tracer_stop, | ||
| 471 | .open = irqsoff_tracer_open, | 491 | .open = irqsoff_tracer_open, |
| 472 | .close = irqsoff_tracer_close, | 492 | .close = irqsoff_tracer_close, |
| 473 | .ctrl_update = irqsoff_tracer_ctrl_update, | ||
| 474 | .print_max = 1, | 493 | .print_max = 1, |
| 475 | #ifdef CONFIG_FTRACE_SELFTEST | 494 | #ifdef CONFIG_FTRACE_SELFTEST |
| 476 | .selftest = trace_selftest_startup_preemptirqsoff, | 495 | .selftest = trace_selftest_startup_preemptirqsoff, |
