/* * trace_ksym.c - Kernel Symbol Tracer * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * Copyright (C) IBM Corporation, 2009 */ #include <linux/kallsyms.h> #include <linux/uaccess.h> #include <linux/debugfs.h> #include <linux/ftrace.h> #include <linux/module.h> #include <linux/slab.h> #include <linux/fs.h> #include "trace_output.h" #include "trace.h" #include <linux/hw_breakpoint.h> #include <asm/hw_breakpoint.h> #include <asm/atomic.h> #define KSYM_TRACER_OP_LEN 3 /* rw- */ struct trace_ksym { struct perf_event **ksym_hbp; struct perf_event_attr attr; #ifdef CONFIG_PROFILE_KSYM_TRACER atomic64_t counter; #endif struct hlist_node ksym_hlist; }; static struct trace_array *ksym_trace_array; static unsigned int ksym_tracing_enabled; static HLIST_HEAD(ksym_filter_head); static DEFINE_MUTEX(ksym_tracer_mutex); #ifdef CONFIG_PROFILE_KSYM_TRACER #define MAX_UL_INT 0xffffffff void ksym_collect_stats(unsigned long hbp_hit_addr) { struct hlist_node *node; struct trace_ksym *entry; rcu_read_lock(); hlist_for_each_entry_rcu(entry, node, &ksym_filter_head, ksym_hlist) { if (entry->attr.bp_addr == hbp_hit_addr) { atomic64_inc(&entry->counter); break; } } rcu_read_unlock(); } #endif /* CONFIG_PROFILE_KSYM_TRACER */ void ksym_hbp_handler(struct perf_event *hbp, int nmi, struct perf_sample_data *data, struct pt_regs *regs) { struct ring_buffer_event *event; struct ksym_trace_entry *entry; struct ring_buffer *buffer; int pc; if (!ksym_tracing_enabled) return; buffer = ksym_trace_array->buffer; pc = preempt_count(); event = trace_buffer_lock_reserve(buffer, TRACE_KSYM, sizeof(*entry), 0, pc); if (!event) return; entry = ring_buffer_event_data(event); entry->ip = instruction_pointer(regs); entry->type = hw_breakpoint_type(hbp); entry->addr = hw_breakpoint_addr(hbp); strlcpy(entry->cmd, current->comm, TASK_COMM_LEN); #ifdef CONFIG_PROFILE_KSYM_TRACER ksym_collect_stats(hw_breakpoint_addr(hbp)); #endif /* CONFIG_PROFILE_KSYM_TRACER */ trace_buffer_unlock_commit(buffer, event, 0, pc); } /* Valid access types are represented as * * rw- : Set Read/Write Access Breakpoint * -w- : Set Write Access Breakpoint * --- : Clear Breakpoints * --x : Set Execution Break points (Not available yet) * */ static int ksym_trace_get_access_type(char *str) { int access = 0; if (str[0] == 'r') access |= HW_BREAKPOINT_R; if (str[1] == 'w') access |= HW_BREAKPOINT_W; if (str[2] == 'x') access |= HW_BREAKPOINT_X; switch (access) { case HW_BREAKPOINT_R: case HW_BREAKPOINT_W: case HW_BREAKPOINT_W | HW_BREAKPOINT_R: return access; default: return -EINVAL; } } /* * There can be several possible malformed requests and we attempt to capture * all of them. We enumerate some of the rules * 1. We will not allow kernel symbols with ':' since it is used as a delimiter. * i.e. multiple ':' symbols disallowed. Possible uses are of the form * <module>:<ksym_name>:<op>. * 2. No delimiter symbol ':' in the input string * 3. Spurious operator symbols or symbols not in their respective positions * 4. <ksym_name>:--- i.e. clear breakpoint request when ksym_name not in file * 5. Kernel symbol not a part of /proc/kallsyms * 6. Duplicate requests */ static int parse_ksym_trace_str(char *input_string, char **ksymname, unsigned long *addr) { int ret; *ksymname = strsep(&input_string, ":"); *addr = kallsyms_lookup_name(*ksymname); /* Check for malformed request: (2), (1) and (5) */ if ((!input_string) || (strlen(input_string) != KSYM_TRACER_OP_LEN) || (*addr == 0)) return -EINVAL;; ret = ksym_trace_get_access_type(input_string); return ret; } int process_new_ksym_entry(char *ksymname, int op, unsigned long addr) { struct trace_ksym *entry; int ret = -ENOMEM; entry = kzalloc(sizeof(struct trace_ksym), GFP_KERNEL); if (!entry) return -ENOMEM; hw_breakpoint_init(&entry->attr); entry->attr.bp_type = op; entry->attr.bp_addr = addr; entry->attr.bp_len = HW_BREAKPOINT_LEN_4; entry->ksym_hbp = register_wide_hw_breakpoint(&entry->attr, ksym_hbp_handler); if (IS_ERR(entry->ksym_hbp)) { ret = PTR_ERR(entry->ksym_hbp); if (ret == -ENOSPC) { printk(KERN_ERR "ksym_tracer: Maximum limit reached." " No new requests for tracing can be accepted now.\n"); } else { printk(KERN_INFO "ksym_tracer request failed. Try again" " later!!\n"); } goto err; } hlist_add_head_rcu(&(entry->ksym_hlist), &ksym_filter_head); return 0; err: kfree(entry); return ret; } static ssize_t ksym_trace_filter_read(struct file *filp, char __user *ubuf, size_t count, loff_t *ppos) { struct trace_ksym *entry; struct hlist_node *node; struct trace_seq *s; ssize_t cnt = 0; int ret; s = kmalloc(sizeof(*s), GFP_KERNEL); if (!s) return -ENOMEM; trace_seq_init(s); mutex_lock(&ksym_tracer_mutex); hlist_for_each_entry(entry, node, &ksym_filter_head, ksym_hlist) { ret = trace_seq_printf(s, "%pS:", (void *)(unsigned long)entry->attr.bp_addr); if (entry->attr.bp_type == HW_BREAKPOINT_R) ret = trace_seq_puts(s, "r--\n"); else if (entry->attr.bp_type == HW_BREAKPOINT_W) ret = trace_seq_puts(s, "-w-\n"); else if (entry->attr.bp_type == (HW_BREAKPOINT_W | HW_BREAKPOINT_R)) ret = trace_seq_puts(s, "rw-\n"); WARN_ON_ONCE(!ret); } cnt = simple_read_from_buffer(ubuf, count, ppos, s->buffer, s->len); mutex_unlock(&ksym_tracer_mutex); kfree(s); return cnt; } static void __ksym_trace_reset(void) { struct trace_ksym *entry; struct hlist_node *node, *node1; mutex_lock(&ksym_tracer_mutex); hlist_for_each_entry_safe(entry, node, node1, &ksym_filter_head, ksym_hlist) { unregister_wide_hw_breakpoint(entry->ksym_hbp); hlist_del_rcu(&(entry->ksym_hlist)); synchronize_rcu(); kfree(entry); } mutex_unlock(&ksym_tracer_mutex); } static ssize_t ksym_trace_filter_write(struct file *file, const char __user *buffer, size_t count, loff_t *ppos) { struct trace_ksym *entry; struct hlist_node *node; char *buf, *input_string, *ksymname = NULL; unsigned long ksym_addr = 0; int ret, op, changed = 0; buf = kzalloc(count + 1, GFP_KERNEL); if (!buf) return -ENOMEM; ret = -EFAULT; if (copy_from_user(buf, buffer, count)) goto out; buf[count] = '\0'; input_string = strstrip(buf); /* * Clear all breakpoints if: * 1: echo > ksym_trace_filter * 2: echo 0 > ksym_trace_filter * 3: echo "*:---" > ksym_trace_filter */ if (!input_string[0] || !strcmp(input_string, "0") || !strcmp(input_string, "*:---")) { __ksym_trace_reset(); ret = 0; goto out; } ret = op = parse_ksym_trace_str(input_string, &ksymname, &ksym_addr); if (ret < 0) goto out; mutex_lock(&ksym_tracer_mutex); ret = -EINVAL; hlist_for_each_entry(entry, node, &ksym_filter_head, ksym_hlist) { if (entry->attr.bp_addr == ksym_addr) { /* Check for malformed request: (6) */ if (entry->attr.bp_type != op) changed = 1; else goto out_unlock; break; } } if (changed) { unregister_wide_hw_breakpoint(entry->ksym_hbp); entry->attr.bp_type = op; ret = 0; if (op > 0) { entry->ksym_hbp = register_wide_hw_breakpoint(&entry->attr, ksym_hbp_handler); if (IS_ERR(entry->ksym_hbp)) ret = PTR_ERR(entry->ksym_hbp); else goto out_unlock; } /* Error or "symbol:---" case: drop it */ hlist_del_rcu(&(entry->ksym_hlist)); synchronize_rcu(); kfree(entry); goto out_unlock; } else { /* Check for malformed request: (4) */ if (op) ret = process_new_ksym_entry(ksymname, op, ksym_addr); } out_unlock: mutex_unlock(&ksym_tracer_mutex); out: kfree(buf); return !ret ? count : ret; } static const struct file_operations ksym_tracing_fops = { .open = tracing_open_generic, .read = ksym_trace_filter_read, .write = ksym_trace_filter_write, }; static void ksym_trace_reset(struct trace_array *tr) { ksym_tracing_enabled = 0; __ksym_trace_reset(); } static int ksym_trace_init(struct trace_array *tr) { int cpu, ret = 0; for_each_online_cpu(cpu) tracing_reset(tr, cpu); ksym_tracing_enabled = 1; ksym_trace_array = tr; return ret; } static void ksym_trace_print_header(struct seq_file *m) { seq_puts(m, "# TASK-PID CPU# Symbol " "Type Function\n"); seq_puts(m, "# | | | " " | |\n"); } static enum print_line_t ksym_trace_output(struct trace_iterator *iter) { struct trace_entry *entry = iter->ent; struct trace_seq *s = &iter->seq; struct ksym_trace_entry *field; char str[KSYM_SYMBOL_LEN]; int ret; if (entry->type != TRACE_KSYM) return TRACE_TYPE_UNHANDLED; trace_assign_type(field, entry); ret = trace_seq_printf(s, "%11s-%-5d [%03d] %pS", field->cmd, entry->pid, iter->cpu, (char *)field->addr); if (!ret) return TRACE_TYPE_PARTIAL_LINE; switch (field->type) { case HW_BREAKPOINT_R: ret = trace_seq_printf(s, " R "); break; case HW_BREAKPOINT_W: ret = trace_seq_printf(s, " W "); break; case HW_BREAKPOINT_R | HW_BREAKPOINT_W: ret = trace_seq_printf(s, " RW "); break; default: return TRACE_TYPE_PARTIAL_LINE; } if (!ret) return TRACE_TYPE_PARTIAL_LINE; sprint_symbol(str, field->ip); ret = trace_seq_printf(s, "%s\n", str); if (!ret) return TRACE_TYPE_PARTIAL_LINE; return TRACE_TYPE_HANDLED; } struct tracer ksym_tracer __read_mostly = { .name = "ksym_tracer", .init = ksym_trace_init, .reset = ksym_trace_reset, #ifdef CONFIG_FTRACE_SELFTEST .selftest = trace_selftest_startup_ksym, #endif .print_header = ksym_trace_print_header, .print_line = ksym_trace_output }; #ifdef CONFIG_PROFILE_KSYM_TRACER static int ksym_profile_show(struct seq_file *m, void *v) { struct hlist_node *node; struct trace_ksym *entry; int access_type = 0; char fn_name[KSYM_NAME_LEN]; seq_puts(m, " Access Type "); seq_puts(m, " Symbol Counter\n"); seq_puts(m, " ----------- "); seq_puts(m, " ------ -------\n"); rcu_read_lock(); hlist_for_each_entry_rcu(entry, node, &ksym_filter_head, ksym_hlist) { access_type = entry->attr.bp_type; switch (access_type) { case HW_BREAKPOINT_R: seq_puts(m, " R "); break; case HW_BREAKPOINT_W: seq_puts(m, " W "); break; case HW_BREAKPOINT_R | HW_BREAKPOINT_W: seq_puts(m, " RW "); break; default: seq_puts(m, " NA "); } if (lookup_symbol_name(entry->attr.bp_addr, fn_name) >= 0) seq_printf(m, " %-36s", fn_name); else seq_printf(m, " %-36s", "<NA>"); seq_printf(m, " %15llu\n", (unsigned long long)atomic64_read(&entry->counter)); } rcu_read_unlock(); return 0; } static int ksym_profile_open(struct inode *node, struct file *file) { return single_open(file, ksym_profile_show, NULL); } static const struct file_operations ksym_profile_fops = { .open = ksym_profile_open, .read = seq_read, .llseek = seq_lseek, .release = single_release, }; #endif /* CONFIG_PROFILE_KSYM_TRACER */ __init static int init_ksym_trace(void) { struct dentry *d_tracer; d_tracer = tracing_init_dentry(); trace_create_file("ksym_trace_filter", 0644, d_tracer, NULL, &ksym_tracing_fops); #ifdef CONFIG_PROFILE_KSYM_TRACER trace_create_file("ksym_profile", 0444, d_tracer, NULL, &ksym_profile_fops); #endif return register_tracer(&ksym_tracer); } device_initcall(init_ksym_trace);