diff options
author | Lai Jiangshan <laijs@cn.fujitsu.com> | 2009-01-11 22:06:18 -0500 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2009-01-15 03:46:40 -0500 |
commit | 6f3b34402e7282cde49dff395d7ea462bf33bf50 (patch) | |
tree | 9f6b01f62bc6065614f3f70aaff11da7f516e383 | |
parent | 428aee1460a75197f0190534b4d610450ee59af1 (diff) |
ring_buffer: reset write when reserve buffer fail
Impact: reset struct buffer_page.write when interrupt storm
if struct buffer_page.write is not reset, any succedent committing
will corrupted ring_buffer:
static inline void
rb_set_commit_to_write(struct ring_buffer_per_cpu *cpu_buffer)
{
......
cpu_buffer->commit_page->commit =
cpu_buffer->commit_page->write;
......
}
when "if (RB_WARN_ON(cpu_buffer, next_page == reader_page))", ring_buffer
is disabled, but some reserved buffers may haven't been committed.
we need reset struct buffer_page.write.
when "if (unlikely(next_page == cpu_buffer->commit_page))", ring_buffer
is still available, we should not corrupt it.
Signed-off-by: Lai Jiangshan <laijs@cn.fujitsu.com>
Signed-off-by: Steven Rostedt <srostedt@redhat.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
-rw-r--r-- | kernel/trace/ring_buffer.c | 10 |
1 files changed, 5 insertions, 5 deletions
diff --git a/kernel/trace/ring_buffer.c b/kernel/trace/ring_buffer.c index 4832ffa5d937..0b9de5a3d699 100644 --- a/kernel/trace/ring_buffer.c +++ b/kernel/trace/ring_buffer.c | |||
@@ -1017,12 +1017,8 @@ __rb_reserve_next(struct ring_buffer_per_cpu *cpu_buffer, | |||
1017 | } | 1017 | } |
1018 | 1018 | ||
1019 | if (next_page == head_page) { | 1019 | if (next_page == head_page) { |
1020 | if (!(buffer->flags & RB_FL_OVERWRITE)) { | 1020 | if (!(buffer->flags & RB_FL_OVERWRITE)) |
1021 | /* reset write */ | ||
1022 | if (tail <= BUF_PAGE_SIZE) | ||
1023 | local_set(&tail_page->write, tail); | ||
1024 | goto out_unlock; | 1021 | goto out_unlock; |
1025 | } | ||
1026 | 1022 | ||
1027 | /* tail_page has not moved yet? */ | 1023 | /* tail_page has not moved yet? */ |
1028 | if (tail_page == cpu_buffer->tail_page) { | 1024 | if (tail_page == cpu_buffer->tail_page) { |
@@ -1097,6 +1093,10 @@ __rb_reserve_next(struct ring_buffer_per_cpu *cpu_buffer, | |||
1097 | return event; | 1093 | return event; |
1098 | 1094 | ||
1099 | out_unlock: | 1095 | out_unlock: |
1096 | /* reset write */ | ||
1097 | if (tail <= BUF_PAGE_SIZE) | ||
1098 | local_set(&tail_page->write, tail); | ||
1099 | |||
1100 | __raw_spin_unlock(&cpu_buffer->lock); | 1100 | __raw_spin_unlock(&cpu_buffer->lock); |
1101 | local_irq_restore(flags); | 1101 | local_irq_restore(flags); |
1102 | return NULL; | 1102 | return NULL; |