123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 |
- #include <linux/elf.h>
- #include <linux/ftrace.h>
- #include <linux/memory.h>
- #include <linux/extable.h>
- #include <linux/module.h>
- #include <linux/mutex.h>
- #include <linux/init.h>
- #include <linux/kprobes.h>
- #include <linux/filter.h>
- #include <asm/sections.h>
- #include <linux/uaccess.h>
- DEFINE_MUTEX(text_mutex);
- extern struct exception_table_entry __start___ex_table[];
- extern struct exception_table_entry __stop___ex_table[];
- u32 __initdata __visible main_extable_sort_needed = 1;
- void __init sort_main_extable(void)
- {
- if (main_extable_sort_needed &&
- &__stop___ex_table > &__start___ex_table) {
- pr_notice("Sorting __ex_table...\n");
- sort_extable(__start___ex_table, __stop___ex_table);
- }
- }
- const
- struct exception_table_entry *search_kernel_exception_table(unsigned long addr)
- {
- return search_extable(__start___ex_table,
- __stop___ex_table - __start___ex_table, addr);
- }
- const struct exception_table_entry *search_exception_tables(unsigned long addr)
- {
- const struct exception_table_entry *e;
- e = search_kernel_exception_table(addr);
- if (!e)
- e = search_module_extables(addr);
- if (!e)
- e = search_bpf_extables(addr);
- return e;
- }
- int notrace core_kernel_text(unsigned long addr)
- {
- if (is_kernel_text(addr))
- return 1;
- if (system_state < SYSTEM_FREEING_INITMEM &&
- is_kernel_inittext(addr))
- return 1;
- return 0;
- }
- int __kernel_text_address(unsigned long addr)
- {
- if (kernel_text_address(addr))
- return 1;
-
- if (is_kernel_inittext(addr))
- return 1;
- return 0;
- }
- int kernel_text_address(unsigned long addr)
- {
- bool no_rcu;
- int ret = 1;
- if (core_kernel_text(addr))
- return 1;
-
- no_rcu = !rcu_is_watching();
-
- if (no_rcu)
- ct_nmi_enter();
- if (is_module_text_address(addr))
- goto out;
- if (is_ftrace_trampoline(addr))
- goto out;
- if (is_kprobe_optinsn_slot(addr) || is_kprobe_insn_slot(addr))
- goto out;
- if (is_bpf_text_address(addr))
- goto out;
- ret = 0;
- out:
- if (no_rcu)
- ct_nmi_exit();
- return ret;
- }
- #ifdef CONFIG_HAVE_FUNCTION_DESCRIPTORS
- void *dereference_function_descriptor(void *ptr)
- {
- func_desc_t *desc = ptr;
- void *p;
- if (!get_kernel_nofault(p, (void *)&desc->addr))
- ptr = p;
- return ptr;
- }
- EXPORT_SYMBOL_GPL(dereference_function_descriptor);
- void *dereference_kernel_function_descriptor(void *ptr)
- {
- if (ptr < (void *)__start_opd || ptr >= (void *)__end_opd)
- return ptr;
- return dereference_function_descriptor(ptr);
- }
- #endif
- int func_ptr_is_kernel_text(void *ptr)
- {
- unsigned long addr;
- addr = (unsigned long) dereference_function_descriptor(ptr);
- if (core_kernel_text(addr))
- return 1;
- return is_module_text_address(addr);
- }
|