提交 19384c03 编写于 作者: S Steven Rostedt 提交者: Thomas Gleixner

ftrace: limit use of check pages

The check_pages function is called often enough that it can cause problems
with trace outputs or even bringing the system to a halt.

This patch limits the check_pages to the places that are most likely to
have problems. The check is made at the flip between the global array and
the max save array, as well as when the size of the buffers changes and
the self tests.

This patch also removes the BUG_ON from check_pages and replaces it with
a WARN_ON and disabling of the tracer.
Signed-off-by: NSteven Rostedt <srostedt@redhat.com>
Cc: pq@iki.fi
Cc: proski@gnu.org
Cc: sandmann@redhat.com
Cc: a.p.zijlstra@chello.nl
Signed-off-by: NThomas Gleixner <tglx@linutronix.de>
上级 c6531cce
...@@ -249,24 +249,32 @@ __update_max_tr(struct trace_array *tr, struct task_struct *tsk, int cpu) ...@@ -249,24 +249,32 @@ __update_max_tr(struct trace_array *tr, struct task_struct *tsk, int cpu)
tracing_record_cmdline(current); tracing_record_cmdline(current);
} }
#define CHECK_COND(cond) \
if (unlikely(cond)) { \
tracing_disabled = 1; \
WARN_ON(1); \
return -1; \
}
/** /**
* check_pages - integrity check of trace buffers * check_pages - integrity check of trace buffers
* *
* As a safty measure we check to make sure the data pages have not * As a safty measure we check to make sure the data pages have not
* been corrupted. TODO: configure to disable this because it adds * been corrupted.
* a bit of overhead.
*/ */
void check_pages(struct trace_array_cpu *data) int check_pages(struct trace_array_cpu *data)
{ {
struct page *page, *tmp; struct page *page, *tmp;
BUG_ON(data->trace_pages.next->prev != &data->trace_pages); CHECK_COND(data->trace_pages.next->prev != &data->trace_pages);
BUG_ON(data->trace_pages.prev->next != &data->trace_pages); CHECK_COND(data->trace_pages.prev->next != &data->trace_pages);
list_for_each_entry_safe(page, tmp, &data->trace_pages, lru) { list_for_each_entry_safe(page, tmp, &data->trace_pages, lru) {
BUG_ON(page->lru.next->prev != &page->lru); CHECK_COND(page->lru.next->prev != &page->lru);
BUG_ON(page->lru.prev->next != &page->lru); CHECK_COND(page->lru.prev->next != &page->lru);
} }
return 0;
} }
/** /**
...@@ -280,7 +288,6 @@ void *head_page(struct trace_array_cpu *data) ...@@ -280,7 +288,6 @@ void *head_page(struct trace_array_cpu *data)
{ {
struct page *page; struct page *page;
check_pages(data);
if (list_empty(&data->trace_pages)) if (list_empty(&data->trace_pages))
return NULL; return NULL;
...@@ -2566,7 +2573,7 @@ tracing_entries_write(struct file *filp, const char __user *ubuf, ...@@ -2566,7 +2573,7 @@ tracing_entries_write(struct file *filp, const char __user *ubuf,
{ {
unsigned long val; unsigned long val;
char buf[64]; char buf[64];
int ret; int i, ret;
if (cnt >= sizeof(buf)) if (cnt >= sizeof(buf))
return -EINVAL; return -EINVAL;
...@@ -2635,8 +2642,15 @@ tracing_entries_write(struct file *filp, const char __user *ubuf, ...@@ -2635,8 +2642,15 @@ tracing_entries_write(struct file *filp, const char __user *ubuf,
trace_free_page(); trace_free_page();
} }
/* check integrity */
for_each_tracing_cpu(i)
check_pages(global_trace.data[i]);
filp->f_pos += cnt; filp->f_pos += cnt;
/* If check pages failed, return ENOMEM */
if (tracing_disabled)
cnt = -ENOMEM;
out: out:
max_tr.entries = global_trace.entries; max_tr.entries = global_trace.entries;
mutex_unlock(&trace_types_lock); mutex_unlock(&trace_types_lock);
......
...@@ -28,6 +28,7 @@ trace_test_buffer_cpu(struct trace_array *tr, struct trace_array_cpu *data) ...@@ -28,6 +28,7 @@ trace_test_buffer_cpu(struct trace_array *tr, struct trace_array_cpu *data)
page = list_entry(data->trace_pages.next, struct page, lru); page = list_entry(data->trace_pages.next, struct page, lru);
entries = page_address(page); entries = page_address(page);
check_pages(data);
if (head_page(data) != entries) if (head_page(data) != entries)
goto failed; goto failed;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册