/* * Memory allocator tracing * * Copyright (C) 2008 Eduard - Gabriel Munteanu * Copyright (C) 2008 Pekka Enberg * Copyright (C) 2008 Frederic Weisbecker */ #include #include #include #include #include #include #include "trace.h" #include "trace_output.h" /* Select an alternative, minimalistic output than the original one */ #define TRACE_KMEM_OPT_MINIMAL 0x1 static struct tracer_opt kmem_opts[] = { /* Default disable the minimalistic output */ { TRACER_OPT(kmem_minimalistic, TRACE_KMEM_OPT_MINIMAL) }, { } }; static struct tracer_flags kmem_tracer_flags = { .val = 0, .opts = kmem_opts }; static struct trace_array *kmemtrace_array; /* Trace allocations */ static inline void kmemtrace_alloc(enum kmemtrace_type_id type_id, unsigned long call_site, const void *ptr, size_t bytes_req, size_t bytes_alloc, gfp_t gfp_flags, int node) { struct ring_buffer_event *event; struct kmemtrace_alloc_entry *entry; struct trace_array *tr = kmemtrace_array; event = ring_buffer_lock_reserve(tr->buffer, sizeof(*entry)); if (!event) return; entry = ring_buffer_event_data(event); tracing_generic_entry_update(&entry->ent, 0, 0); entry->ent.type = TRACE_KMEM_ALLOC; entry->call_site = call_site; entry->ptr = ptr; entry->bytes_req = bytes_req; entry->bytes_alloc = bytes_alloc; entry->gfp_flags = gfp_flags; entry->node = node; ring_buffer_unlock_commit(tr->buffer, event); trace_wake_up(); } static inline void kmemtrace_free(enum kmemtrace_type_id type_id, unsigned long call_site, const void *ptr) { struct ring_buffer_event *event; struct kmemtrace_free_entry *entry; struct trace_array *tr = kmemtrace_array; event = ring_buffer_lock_reserve(tr->buffer, sizeof(*entry)); if (!event) return; entry = ring_buffer_event_data(event); tracing_generic_entry_update(&entry->ent, 0, 0); entry->ent.type = TRACE_KMEM_FREE; entry->type_id = type_id; entry->call_site = call_site; entry->ptr = ptr; ring_buffer_unlock_commit(tr->buffer, event); trace_wake_up(); } static void kmemtrace_kmalloc(unsigned long call_site, const void *ptr, size_t bytes_req, size_t bytes_alloc, gfp_t gfp_flags) { kmemtrace_alloc(KMEMTRACE_TYPE_KMALLOC, call_site, ptr, bytes_req, bytes_alloc, gfp_flags, -1); } static void kmemtrace_kmem_cache_alloc(unsigned long call_site, const void *ptr, size_t bytes_req, size_t bytes_alloc, gfp_t gfp_flags) { kmemtrace_alloc(KMEMTRACE_TYPE_CACHE, call_site, ptr, bytes_req, bytes_alloc, gfp_flags, -1); } static void kmemtrace_kmalloc_node(unsigned long call_site, const void *ptr, size_t bytes_req, size_t bytes_alloc, gfp_t gfp_flags, int node) { kmemtrace_alloc(KMEMTRACE_TYPE_KMALLOC, call_site, ptr, bytes_req, bytes_alloc, gfp_flags, node); } static void kmemtrace_kmem_cache_alloc_node(unsigned long call_site, const void *ptr, size_t bytes_req, size_t bytes_alloc, gfp_t gfp_flags, int node) { kmemtrace_alloc(KMEMTRACE_TYPE_CACHE, call_site, ptr, bytes_req, bytes_alloc, gfp_flags, node); } static void kmemtrace_kfree(unsigned long call_site, const void *ptr) { kmemtrace_free(KMEMTRACE_TYPE_KMALLOC, call_site, ptr); } static void kmemtrace_kmem_cache_free(unsigned long call_site, const void *ptr) { kmemtrace_free(KMEMTRACE_TYPE_CACHE, call_site, ptr); } static int kmemtrace_start_probes(void) { int err; err = register_trace_kmalloc(kmemtrace_kmalloc); if (err) return err; err = register_trace_kmem_cache_alloc(kmemtrace_kmem_cache_alloc); if (err) return err; err = register_trace_kmalloc_node(kmemtrace_kmalloc_node); if (err) return err; err = register_trace_kmem_cache_alloc_node(kmemtrace_kmem_cache_alloc_node); if (err) return err; err = register_trace_kfree(kmemtrace_kfree); if (err) return err; err = register_trace_kmem_cache_free(kmemtrace_kmem_cache_free); return err; } static void kmemtrace_stop_probes(void) { unregister_trace_kmalloc(kmemtrace_kmalloc); unregister_trace_kmem_cache_alloc(kmemtrace_kmem_cache_alloc); unregister_trace_kmalloc_node(kmemtrace_kmalloc_node); unregister_trace_kmem_cache_alloc_node(kmemtrace_kmem_cache_alloc_node); unregister_trace_kfree(kmemtrace_kfree); unregister_trace_kmem_cache_free(kmemtrace_kmem_cache_free); } static int kmem_trace_init(struct trace_array *tr) { int cpu; kmemtrace_array = tr; for_each_cpu_mask(cpu, cpu_possible_map) tracing_reset(tr, cpu); kmemtrace_start_probes(); return 0; } static void kmem_trace_reset(struct trace_array *tr) { kmemtrace_stop_probes(); } static void kmemtrace_headers(struct seq_file *s) { /* Don't need headers for the original kmemtrace output */ if (!(kmem_tracer_flags.val & TRACE_KMEM_OPT_MINIMAL)) return; seq_printf(s, "#\n"); seq_printf(s, "# ALLOC TYPE REQ GIVEN FLAGS " " POINTER NODE CALLER\n"); seq_printf(s, "# FREE | | | | " " | | | |\n"); seq_printf(s, "# |\n\n"); } /* * The two following functions give the original output from kmemtrace, * or something close to....perhaps they need some missing things */ static enum print_line_t kmemtrace_print_alloc_original(struct trace_iterator *iter, struct kmemtrace_alloc_entry *entry) { struct trace_seq *s = &iter->seq; int ret; /* Taken from the old linux/kmemtrace.h */ ret = trace_seq_printf(s, "type_id %d call_site %lu ptr %lu " "bytes_req %lu bytes_alloc %lu gfp_flags %lu node %d\n", entry->type_id, entry->call_site, (unsigned long) entry->ptr, (unsigned long) entry->bytes_req, (unsigned long) entry->bytes_alloc, (unsigned long) entry->gfp_flags, entry->node); if (!ret) return TRACE_TYPE_PARTIAL_LINE; return TRACE_TYPE_HANDLED; } static enum print_line_t kmemtrace_print_free_original(struct trace_iterator *iter, struct kmemtrace_free_entry *entry) { struct trace_seq *s = &iter->seq; int ret; /* Taken from the old linux/kmemtrace.h */ ret = trace_seq_printf(s, "type_id %d call_site %lu ptr %lu\n", entry->type_id, entry->call_site, (unsigned long) entry->ptr); if (!ret) return TRACE_TYPE_PARTIAL_LINE; return TRACE_TYPE_HANDLED; } /* The two other following provide a more minimalistic output */ static enum print_line_t kmemtrace_print_alloc_compress(struct trace_iterator *iter, struct kmemtrace_alloc_entry *entry) { struct trace_seq *s = &iter->seq; int ret; /* Alloc entry */ ret = trace_seq_printf(s, " + "); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Type */ switch (entry->type_id) { case KMEMTRACE_TYPE_KMALLOC: ret = trace_seq_printf(s, "K "); break; case KMEMTRACE_TYPE_CACHE: ret = trace_seq_printf(s, "C "); break; case KMEMTRACE_TYPE_PAGES: ret = trace_seq_printf(s, "P "); break; default: ret = trace_seq_printf(s, "? "); } if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Requested */ ret = trace_seq_printf(s, "%4zu ", entry->bytes_req); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Allocated */ ret = trace_seq_printf(s, "%4zu ", entry->bytes_alloc); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Flags * TODO: would be better to see the name of the GFP flag names */ ret = trace_seq_printf(s, "%08x ", entry->gfp_flags); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Pointer to allocated */ ret = trace_seq_printf(s, "0x%tx ", (ptrdiff_t)entry->ptr); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Node */ ret = trace_seq_printf(s, "%4d ", entry->node); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Call site */ ret = seq_print_ip_sym(s, entry->call_site, 0); if (!ret) return TRACE_TYPE_PARTIAL_LINE; if (!trace_seq_printf(s, "\n")) return TRACE_TYPE_PARTIAL_LINE; return TRACE_TYPE_HANDLED; } static enum print_line_t kmemtrace_print_free_compress(struct trace_iterator *iter, struct kmemtrace_free_entry *entry) { struct trace_seq *s = &iter->seq; int ret; /* Free entry */ ret = trace_seq_printf(s, " - "); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Type */ switch (entry->type_id) { case KMEMTRACE_TYPE_KMALLOC: ret = trace_seq_printf(s, "K "); break; case KMEMTRACE_TYPE_CACHE: ret = trace_seq_printf(s, "C "); break; case KMEMTRACE_TYPE_PAGES: ret = trace_seq_printf(s, "P "); break; default: ret = trace_seq_printf(s, "? "); } if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Skip requested/allocated/flags */ ret = trace_seq_printf(s, " "); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Pointer to allocated */ ret = trace_seq_printf(s, "0x%tx ", (ptrdiff_t)entry->ptr); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Skip node */ ret = trace_seq_printf(s, " "); if (!ret) return TRACE_TYPE_PARTIAL_LINE; /* Call site */ ret = seq_print_ip_sym(s, entry->call_site, 0); if (!ret) return TRACE_TYPE_PARTIAL_LINE; if (!trace_seq_printf(s, "\n")) return TRACE_TYPE_PARTIAL_LINE; return TRACE_TYPE_HANDLED; } static enum print_line_t kmemtrace_print_line(struct trace_iterator *iter) { struct trace_entry *entry = iter->ent; switch (entry->type) { case TRACE_KMEM_ALLOC: { struct kmemtrace_alloc_entry *field; trace_assign_type(field, entry); if (kmem_tracer_flags.val & TRACE_KMEM_OPT_MINIMAL) return kmemtrace_print_alloc_compress(iter, field); else return kmemtrace_print_alloc_original(iter, field); } case TRACE_KMEM_FREE: { struct kmemtrace_free_entry *field; trace_assign_type(field, entry); if (kmem_tracer_flags.val & TRACE_KMEM_OPT_MINIMAL) return kmemtrace_print_free_compress(iter, field); else return kmemtrace_print_free_original(iter, field); } default: return TRACE_TYPE_UNHANDLED; } } static struct tracer kmem_tracer __read_mostly = { .name = "kmemtrace", .init = kmem_trace_init, .reset = kmem_trace_reset, .print_line = kmemtrace_print_line, .print_header = kmemtrace_headers, .flags = &kmem_tracer_flags }; void kmemtrace_init(void) { /* earliest opportunity to start kmem tracing */ } static int __init init_kmem_tracer(void) { return register_tracer(&kmem_tracer); } device_initcall(init_kmem_tracer);