Skip to content

Commit

Permalink
ftrace: Have init/main.c call ftrace directly to free init memory
Browse files Browse the repository at this point in the history
Relying on free_reserved_area() to call ftrace to free init memory proved to
not be sufficient. The issue is that on x86, when debug_pagealloc is
enabled, the init memory is not freed, but simply set as not present. Since
ftrace was uninformed of this, starting function tracing still tries to
update pages that are not present according to the page tables, causing
ftrace to bug, as well as killing the kernel itself.

Instead of relying on free_reserved_area(), have init/main.c call ftrace
directly just before it frees the init memory. Then it needs to use
__init_begin and __init_end to know where the init memory location is.
Looking at all archs (and testing what I can), it appears that this should
work for each of them.

Reported-by: kernel test robot <[email protected]>
Reported-by: Fengguang Wu <[email protected]>
Signed-off-by: Steven Rostedt (VMware) <[email protected]>
  • Loading branch information
rostedt committed Apr 3, 2017
1 parent 5bd8462 commit b80f0f6
Show file tree
Hide file tree
Showing 4 changed files with 8 additions and 9 deletions.
6 changes: 3 additions & 3 deletions include/linux/ftrace.h
Original file line number Diff line number Diff line change
Expand Up @@ -147,9 +147,9 @@ struct ftrace_ops_hash {
struct mutex regex_lock;
};

void ftrace_free_mem(void *start, void *end);
void ftrace_free_init_mem(void);
#else
static inline void ftrace_free_mem(void *start, void *end) { }
static inline void ftrace_free_init_mem(void) { }
#endif

/*
Expand Down Expand Up @@ -266,7 +266,7 @@ static inline int ftrace_nr_registered_ops(void)
}
static inline void clear_ftrace_function(void) { }
static inline void ftrace_kill(void) { }
static inline void ftrace_free_mem(void *start, void *end) { }
static inline void ftrace_free_init_mem(void) { }
#endif /* CONFIG_FUNCTION_TRACER */

#ifdef CONFIG_STACK_TRACER
Expand Down
1 change: 1 addition & 0 deletions init/main.c
Original file line number Diff line number Diff line change
Expand Up @@ -962,6 +962,7 @@ static int __ref kernel_init(void *unused)
kernel_init_freeable();
/* need to finish all async __init code before freeing the memory */
async_synchronize_full();
ftrace_free_init_mem();
free_initmem();
mark_readonly();
system_state = SYSTEM_RUNNING;
Expand Down
7 changes: 4 additions & 3 deletions kernel/trace/ftrace.c
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@

#include <trace/events/sched.h>

#include <asm/sections.h>
#include <asm/setup.h>

#include "trace_output.h"
Expand Down Expand Up @@ -5279,10 +5280,10 @@ void ftrace_module_init(struct module *mod)
}
#endif /* CONFIG_MODULES */

void ftrace_free_mem(void *start_ptr, void *end_ptr)
void __init ftrace_free_init_mem(void)
{
unsigned long start = (unsigned long)start_ptr;
unsigned long end = (unsigned long)end_ptr;
unsigned long start = (unsigned long)(&__init_begin);
unsigned long end = (unsigned long)(&__init_end);
struct ftrace_page **last_pg = &ftrace_pages_start;
struct ftrace_page *pg;
struct dyn_ftrace *rec;
Expand Down
3 changes: 0 additions & 3 deletions mm/page_alloc.c
Original file line number Diff line number Diff line change
Expand Up @@ -6606,9 +6606,6 @@ unsigned long free_reserved_area(void *start, void *end, int poison, char *s)
void *pos;
unsigned long pages = 0;

/* This may be .init text, inform ftrace to remove it */
ftrace_free_mem(start, end);

start = (void *)PAGE_ALIGN((unsigned long)start);
end = (void *)((unsigned long)end & PAGE_MASK);
for (pos = start; pos < end; pos += PAGE_SIZE, pages++) {
Expand Down

0 comments on commit b80f0f6

Please sign in to comment.