Commit 265c831c authored by Steven Rostedt's avatar Steven Rostedt

ftrace: add do_for_each_ftrace_rec and while_for_each_ftrace_rec

Impact: clean up

To iterate over all the functions that dynamic trace knows about
it requires two for loops. One to iterate over the pages and the
other to iterate over the records within the page.

There are several duplications of these loops in ftrace.c. This
patch creates the macros do_for_each_ftrace_rec and
while_for_each_ftrace_rec to handle this logic, and removes the
duplicate code.

While making this change, I also discovered and fixed a small
bug that one of the iterations should exit the loop after it found the
record it was searching for. This used a break when it should have
used a goto, since there were two loops it needed to break out
from.  No real harm was done by this bug since it would only continue
to search the other records, and the code was in a slow path anyway.
Signed-off-by: default avatarSteven Rostedt <srostedt@redhat.com>
parent 0c75a3ed
...@@ -297,6 +297,19 @@ static struct ftrace_page *ftrace_pages; ...@@ -297,6 +297,19 @@ static struct ftrace_page *ftrace_pages;
static struct dyn_ftrace *ftrace_free_records; static struct dyn_ftrace *ftrace_free_records;
/*
* This is a double for. Do not use 'break' to break out of the loop,
* you must use a goto.
*/
#define do_for_each_ftrace_rec(pg, rec) \
for (pg = ftrace_pages_start; pg; pg = pg->next) { \
int _____i; \
for (_____i = 0; _____i < pg->index; _____i++) { \
rec = &pg->records[_____i];
#define while_for_each_ftrace_rec() \
} \
}
#ifdef CONFIG_KPROBES #ifdef CONFIG_KPROBES
...@@ -341,7 +354,6 @@ void ftrace_release(void *start, unsigned long size) ...@@ -341,7 +354,6 @@ void ftrace_release(void *start, unsigned long size)
struct ftrace_page *pg; struct ftrace_page *pg;
unsigned long s = (unsigned long)start; unsigned long s = (unsigned long)start;
unsigned long e = s + size; unsigned long e = s + size;
int i;
if (ftrace_disabled || !start) if (ftrace_disabled || !start)
return; return;
...@@ -349,14 +361,11 @@ void ftrace_release(void *start, unsigned long size) ...@@ -349,14 +361,11 @@ void ftrace_release(void *start, unsigned long size)
/* should not be called from interrupt context */ /* should not be called from interrupt context */
spin_lock(&ftrace_lock); spin_lock(&ftrace_lock);
for (pg = ftrace_pages_start; pg; pg = pg->next) { do_for_each_ftrace_rec(pg, rec) {
for (i = 0; i < pg->index; i++) { if ((rec->ip >= s) && (rec->ip < e))
rec = &pg->records[i]; ftrace_free_rec(rec);
} while_for_each_ftrace_rec();
if ((rec->ip >= s) && (rec->ip < e))
ftrace_free_rec(rec);
}
}
spin_unlock(&ftrace_lock); spin_unlock(&ftrace_lock);
} }
...@@ -523,41 +532,37 @@ __ftrace_replace_code(struct dyn_ftrace *rec, int enable) ...@@ -523,41 +532,37 @@ __ftrace_replace_code(struct dyn_ftrace *rec, int enable)
static void ftrace_replace_code(int enable) static void ftrace_replace_code(int enable)
{ {
int i, failed; int failed;
struct dyn_ftrace *rec; struct dyn_ftrace *rec;
struct ftrace_page *pg; struct ftrace_page *pg;
for (pg = ftrace_pages_start; pg; pg = pg->next) { do_for_each_ftrace_rec(pg, rec) {
for (i = 0; i < pg->index; i++) { /*
rec = &pg->records[i]; * Skip over free records and records that have
* failed.
/* */
* Skip over free records and records that have if (rec->flags & FTRACE_FL_FREE ||
* failed. rec->flags & FTRACE_FL_FAILED)
*/ continue;
if (rec->flags & FTRACE_FL_FREE ||
rec->flags & FTRACE_FL_FAILED)
continue;
/* ignore updates to this record's mcount site */
if (get_kprobe((void *)rec->ip)) {
freeze_record(rec);
continue;
} else {
unfreeze_record(rec);
}
failed = __ftrace_replace_code(rec, enable); /* ignore updates to this record's mcount site */
if (failed && (rec->flags & FTRACE_FL_CONVERTED)) { if (get_kprobe((void *)rec->ip)) {
rec->flags |= FTRACE_FL_FAILED; freeze_record(rec);
if ((system_state == SYSTEM_BOOTING) || continue;
!core_kernel_text(rec->ip)) { } else {
ftrace_free_rec(rec); unfreeze_record(rec);
} else
ftrace_bug(failed, rec->ip);
}
} }
}
failed = __ftrace_replace_code(rec, enable);
if (failed && (rec->flags & FTRACE_FL_CONVERTED)) {
rec->flags |= FTRACE_FL_FAILED;
if ((system_state == SYSTEM_BOOTING) ||
!core_kernel_text(rec->ip)) {
ftrace_free_rec(rec);
} else
ftrace_bug(failed, rec->ip);
}
} while_for_each_ftrace_rec();
} }
static int static int
...@@ -956,22 +961,17 @@ static void ftrace_filter_reset(int enable) ...@@ -956,22 +961,17 @@ static void ftrace_filter_reset(int enable)
struct ftrace_page *pg; struct ftrace_page *pg;
struct dyn_ftrace *rec; struct dyn_ftrace *rec;
unsigned long type = enable ? FTRACE_FL_FILTER : FTRACE_FL_NOTRACE; unsigned long type = enable ? FTRACE_FL_FILTER : FTRACE_FL_NOTRACE;
unsigned i;
/* should not be called from interrupt context */ /* should not be called from interrupt context */
spin_lock(&ftrace_lock); spin_lock(&ftrace_lock);
if (enable) if (enable)
ftrace_filtered = 0; ftrace_filtered = 0;
pg = ftrace_pages_start; do_for_each_ftrace_rec(pg, rec) {
while (pg) { if (rec->flags & FTRACE_FL_FAILED)
for (i = 0; i < pg->index; i++) { continue;
rec = &pg->records[i]; rec->flags &= ~type;
if (rec->flags & FTRACE_FL_FAILED) } while_for_each_ftrace_rec();
continue;
rec->flags &= ~type;
}
pg = pg->next;
}
spin_unlock(&ftrace_lock); spin_unlock(&ftrace_lock);
} }
...@@ -1094,44 +1094,39 @@ ftrace_match(unsigned char *buff, int len, int enable) ...@@ -1094,44 +1094,39 @@ ftrace_match(unsigned char *buff, int len, int enable)
spin_lock(&ftrace_lock); spin_lock(&ftrace_lock);
if (enable) if (enable)
ftrace_filtered = 1; ftrace_filtered = 1;
pg = ftrace_pages_start; do_for_each_ftrace_rec(pg, rec) {
while (pg) { int matched = 0;
for (i = 0; i < pg->index; i++) { char *ptr;
int matched = 0;
char *ptr; if (rec->flags & FTRACE_FL_FAILED)
continue;
rec = &pg->records[i]; kallsyms_lookup(rec->ip, NULL, NULL, NULL, str);
if (rec->flags & FTRACE_FL_FAILED) switch (type) {
continue; case MATCH_FULL:
kallsyms_lookup(rec->ip, NULL, NULL, NULL, str); if (strcmp(str, buff) == 0)
switch (type) { matched = 1;
case MATCH_FULL: break;
if (strcmp(str, buff) == 0) case MATCH_FRONT_ONLY:
matched = 1; if (memcmp(str, buff, match) == 0)
break; matched = 1;
case MATCH_FRONT_ONLY: break;
if (memcmp(str, buff, match) == 0) case MATCH_MIDDLE_ONLY:
matched = 1; if (strstr(str, search))
break; matched = 1;
case MATCH_MIDDLE_ONLY: break;
if (strstr(str, search)) case MATCH_END_ONLY:
matched = 1; ptr = strstr(str, search);
break; if (ptr && (ptr[search_len] == 0))
case MATCH_END_ONLY: matched = 1;
ptr = strstr(str, search); break;
if (ptr && (ptr[search_len] == 0))
matched = 1;
break;
}
if (matched) {
if (not)
rec->flags &= ~flag;
else
rec->flags |= flag;
}
} }
pg = pg->next; if (matched) {
} if (not)
rec->flags &= ~flag;
else
rec->flags |= flag;
}
} while_for_each_ftrace_rec();
spin_unlock(&ftrace_lock); spin_unlock(&ftrace_lock);
} }
...@@ -1452,7 +1447,7 @@ ftrace_set_func(unsigned long *array, int idx, char *buffer) ...@@ -1452,7 +1447,7 @@ ftrace_set_func(unsigned long *array, int idx, char *buffer)
struct dyn_ftrace *rec; struct dyn_ftrace *rec;
struct ftrace_page *pg; struct ftrace_page *pg;
int found = 0; int found = 0;
int i, j; int j;
if (ftrace_disabled) if (ftrace_disabled)
return -ENODEV; return -ENODEV;
...@@ -1460,27 +1455,26 @@ ftrace_set_func(unsigned long *array, int idx, char *buffer) ...@@ -1460,27 +1455,26 @@ ftrace_set_func(unsigned long *array, int idx, char *buffer)
/* should not be called from interrupt context */ /* should not be called from interrupt context */
spin_lock(&ftrace_lock); spin_lock(&ftrace_lock);
for (pg = ftrace_pages_start; pg; pg = pg->next) { do_for_each_ftrace_rec(pg, rec) {
for (i = 0; i < pg->index; i++) {
rec = &pg->records[i]; if (rec->flags & (FTRACE_FL_FAILED | FTRACE_FL_FREE))
continue;
if (rec->flags & (FTRACE_FL_FAILED | FTRACE_FL_FREE))
continue; kallsyms_lookup(rec->ip, NULL, NULL, NULL, str);
if (strcmp(str, buffer) == 0) {
kallsyms_lookup(rec->ip, NULL, NULL, NULL, str); /* Return 1 if we add it to the array */
if (strcmp(str, buffer) == 0) { found = 1;
found = 1; for (j = 0; j < idx; j++)
for (j = 0; j < idx; j++) if (array[j] == rec->ip) {
if (array[j] == rec->ip) { found = 0;
found = 0; break;
break; }
} if (found)
if (found) array[idx] = rec->ip;
array[idx] = rec->ip; goto out;
break;
}
} }
} } while_for_each_ftrace_rec();
out:
spin_unlock(&ftrace_lock); spin_unlock(&ftrace_lock);
return found ? 0 : -EINVAL; return found ? 0 : -EINVAL;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment