diff options
Diffstat (limited to 'kernel/trace/trace.c')
-rw-r--r-- | kernel/trace/trace.c | 96 |
1 files changed, 55 insertions, 41 deletions
diff --git a/kernel/trace/trace.c b/kernel/trace/trace.c index 11fde0ad9cb6..d89821283b47 100644 --- a/kernel/trace/trace.c +++ b/kernel/trace/trace.c | |||
@@ -2537,15 +2537,49 @@ static void tracing_spd_release_pipe(struct splice_pipe_desc *spd, | |||
2537 | } | 2537 | } |
2538 | 2538 | ||
2539 | static struct pipe_buf_operations tracing_pipe_buf_ops = { | 2539 | static struct pipe_buf_operations tracing_pipe_buf_ops = { |
2540 | .can_merge = 0, | 2540 | .can_merge = 0, |
2541 | .map = generic_pipe_buf_map, | 2541 | .map = generic_pipe_buf_map, |
2542 | .unmap = generic_pipe_buf_unmap, | 2542 | .unmap = generic_pipe_buf_unmap, |
2543 | .confirm = generic_pipe_buf_confirm, | 2543 | .confirm = generic_pipe_buf_confirm, |
2544 | .release = tracing_pipe_buf_release, | 2544 | .release = tracing_pipe_buf_release, |
2545 | .steal = generic_pipe_buf_steal, | 2545 | .steal = generic_pipe_buf_steal, |
2546 | .get = generic_pipe_buf_get, | 2546 | .get = generic_pipe_buf_get, |
2547 | }; | 2547 | }; |
2548 | 2548 | ||
2549 | static size_t | ||
2550 | tracing_fill_pipe_page(struct page *pages, size_t rem, | ||
2551 | struct trace_iterator *iter) | ||
2552 | { | ||
2553 | size_t count; | ||
2554 | int ret; | ||
2555 | |||
2556 | /* Seq buffer is page-sized, exactly what we need. */ | ||
2557 | for (;;) { | ||
2558 | count = iter->seq.len; | ||
2559 | ret = print_trace_line(iter); | ||
2560 | count = iter->seq.len - count; | ||
2561 | if (rem < count) { | ||
2562 | rem = 0; | ||
2563 | iter->seq.len -= count; | ||
2564 | break; | ||
2565 | } | ||
2566 | if (ret == TRACE_TYPE_PARTIAL_LINE) { | ||
2567 | iter->seq.len -= count; | ||
2568 | break; | ||
2569 | } | ||
2570 | |||
2571 | trace_consume(iter); | ||
2572 | rem -= count; | ||
2573 | if (!find_next_entry_inc(iter)) { | ||
2574 | rem = 0; | ||
2575 | iter->ent = NULL; | ||
2576 | break; | ||
2577 | } | ||
2578 | } | ||
2579 | |||
2580 | return rem; | ||
2581 | } | ||
2582 | |||
2549 | static ssize_t tracing_splice_read_pipe(struct file *filp, | 2583 | static ssize_t tracing_splice_read_pipe(struct file *filp, |
2550 | loff_t *ppos, | 2584 | loff_t *ppos, |
2551 | struct pipe_inode_info *pipe, | 2585 | struct pipe_inode_info *pipe, |
@@ -2556,15 +2590,15 @@ static ssize_t tracing_splice_read_pipe(struct file *filp, | |||
2556 | struct partial_page partial[PIPE_BUFFERS]; | 2590 | struct partial_page partial[PIPE_BUFFERS]; |
2557 | struct trace_iterator *iter = filp->private_data; | 2591 | struct trace_iterator *iter = filp->private_data; |
2558 | struct splice_pipe_desc spd = { | 2592 | struct splice_pipe_desc spd = { |
2559 | .pages = pages, | 2593 | .pages = pages, |
2560 | .partial = partial, | 2594 | .partial = partial, |
2561 | .nr_pages = 0, /* This gets updated below. */ | 2595 | .nr_pages = 0, /* This gets updated below. */ |
2562 | .flags = flags, | 2596 | .flags = flags, |
2563 | .ops = &tracing_pipe_buf_ops, | 2597 | .ops = &tracing_pipe_buf_ops, |
2564 | .spd_release = tracing_spd_release_pipe, | 2598 | .spd_release = tracing_spd_release_pipe, |
2565 | }; | 2599 | }; |
2566 | ssize_t ret; | 2600 | ssize_t ret; |
2567 | size_t count, rem; | 2601 | size_t rem; |
2568 | unsigned int i; | 2602 | unsigned int i; |
2569 | 2603 | ||
2570 | mutex_lock(&trace_types_lock); | 2604 | mutex_lock(&trace_types_lock); |
@@ -2573,45 +2607,25 @@ static ssize_t tracing_splice_read_pipe(struct file *filp, | |||
2573 | ret = iter->trace->splice_read(iter, filp, | 2607 | ret = iter->trace->splice_read(iter, filp, |
2574 | ppos, pipe, len, flags); | 2608 | ppos, pipe, len, flags); |
2575 | if (ret) | 2609 | if (ret) |
2576 | goto out; | 2610 | goto out_err; |
2577 | } | 2611 | } |
2578 | 2612 | ||
2579 | ret = tracing_wait_pipe(filp); | 2613 | ret = tracing_wait_pipe(filp); |
2580 | if (ret <= 0) | 2614 | if (ret <= 0) |
2581 | goto out; | 2615 | goto out_err; |
2582 | 2616 | ||
2583 | if (!iter->ent && !find_next_entry_inc(iter)) { | 2617 | if (!iter->ent && !find_next_entry_inc(iter)) { |
2584 | ret = -EFAULT; | 2618 | ret = -EFAULT; |
2585 | goto out; | 2619 | goto out_err; |
2586 | } | 2620 | } |
2587 | 2621 | ||
2588 | /* Fill as many pages as possible. */ | 2622 | /* Fill as many pages as possible. */ |
2589 | for (i = 0, rem = len; i < PIPE_BUFFERS && rem; i++) { | 2623 | for (i = 0, rem = len; i < PIPE_BUFFERS && rem; i++) { |
2590 | pages[i] = alloc_page(GFP_KERNEL); | 2624 | pages[i] = alloc_page(GFP_KERNEL); |
2625 | if (!pages[i]) | ||
2626 | break; | ||
2591 | 2627 | ||
2592 | /* Seq buffer is page-sized, exactly what we need. */ | 2628 | rem = tracing_fill_pipe_page(pages[i], rem, iter); |
2593 | for (;;) { | ||
2594 | count = iter->seq.len; | ||
2595 | ret = print_trace_line(iter); | ||
2596 | count = iter->seq.len - count; | ||
2597 | if (rem < count) { | ||
2598 | rem = 0; | ||
2599 | iter->seq.len -= count; | ||
2600 | break; | ||
2601 | } | ||
2602 | if (ret == TRACE_TYPE_PARTIAL_LINE) { | ||
2603 | iter->seq.len -= count; | ||
2604 | break; | ||
2605 | } | ||
2606 | |||
2607 | trace_consume(iter); | ||
2608 | rem -= count; | ||
2609 | if (!find_next_entry_inc(iter)) { | ||
2610 | rem = 0; | ||
2611 | iter->ent = NULL; | ||
2612 | break; | ||
2613 | } | ||
2614 | } | ||
2615 | 2629 | ||
2616 | /* Copy the data into the page, so we can start over. */ | 2630 | /* Copy the data into the page, so we can start over. */ |
2617 | ret = trace_seq_to_buffer(&iter->seq, | 2631 | ret = trace_seq_to_buffer(&iter->seq, |
@@ -2633,7 +2647,7 @@ static ssize_t tracing_splice_read_pipe(struct file *filp, | |||
2633 | 2647 | ||
2634 | return splice_to_pipe(pipe, &spd); | 2648 | return splice_to_pipe(pipe, &spd); |
2635 | 2649 | ||
2636 | out: | 2650 | out_err: |
2637 | mutex_unlock(&trace_types_lock); | 2651 | mutex_unlock(&trace_types_lock); |
2638 | 2652 | ||
2639 | return ret; | 2653 | return ret; |