cregit-Linux how code gets into the kernel

Release 4.11 arch/x86/kernel/stacktrace.c

Directory: arch/x86/kernel
/*
 * Stack trace management functions
 *
 *  Copyright (C) 2006-2009 Red Hat, Inc., Ingo Molnar <mingo@redhat.com>
 */
#include <linux/sched.h>
#include <linux/sched/debug.h>
#include <linux/sched/task_stack.h>
#include <linux/stacktrace.h>
#include <linux/export.h>
#include <linux/uaccess.h>
#include <asm/stacktrace.h>
#include <asm/unwind.h>


static int save_stack_address(struct stack_trace *trace, unsigned long addr, bool nosched) { if (nosched && in_sched_functions(addr)) return 0; if (trace->skip > 0) { trace->skip--; return 0; } if (trace->nr_entries >= trace->max_entries) return -1; trace->entries[trace->nr_entries++] = addr; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Andi Kleen2532.47%114.29%
Ingo Molnar1823.38%114.29%
Alexei Starovoitov1114.29%114.29%
Oleg Nesterov1114.29%114.29%
Josh Poimboeuf911.69%114.29%
Vegard Nossum22.60%114.29%
Arjan van de Ven11.30%114.29%
Total77100.00%7100.00%


static void __save_stack_trace(struct stack_trace *trace, struct task_struct *task, struct pt_regs *regs, bool nosched) { struct unwind_state state; unsigned long addr; if (regs) save_stack_address(trace, regs->ip, nosched); for (unwind_start(&state, task, regs, NULL); !unwind_done(&state); unwind_next_frame(&state)) { addr = unwind_get_return_address(&state); if (!addr || save_stack_address(trace, addr, nosched)) break; } if (trace->nr_entries < trace->max_entries) trace->entries[trace->nr_entries++] = ULONG_MAX; }

Contributors

PersonTokensPropCommitsCommitProp
Josh Poimboeuf7561.98%112.50%
Ingo Molnar1613.22%112.50%
Catalin Marinas108.26%112.50%
Arjan van de Ven97.44%225.00%
Oleg Nesterov75.79%112.50%
Andi Kleen32.48%112.50%
Namhyung Kim10.83%112.50%
Total121100.00%8100.00%

/* * Save stack-backtrace addresses into a stack_trace buffer. */
void save_stack_trace(struct stack_trace *trace) { __save_stack_trace(trace, current, NULL, false); }

Contributors

PersonTokensPropCommitsCommitProp
Josh Poimboeuf21100.00%1100.00%
Total21100.00%1100.00%

EXPORT_SYMBOL_GPL(save_stack_trace);
void save_stack_trace_regs(struct pt_regs *regs, struct stack_trace *trace) { __save_stack_trace(trace, current, regs, false); }

Contributors

PersonTokensPropCommitsCommitProp
Vegard Nossum1453.85%125.00%
Soeren Sandmann Pedersen415.38%125.00%
Masami Hiramatsu415.38%125.00%
Josh Poimboeuf415.38%125.00%
Total26100.00%4100.00%


void save_stack_trace_tsk(struct task_struct *tsk, struct stack_trace *trace) { if (!try_get_task_stack(tsk)) return; __save_stack_trace(trace, tsk, NULL, true); put_task_stack(tsk); }

Contributors

PersonTokensPropCommitsCommitProp
Arjan van de Ven2255.00%133.33%
Andrew Lutomirski1435.00%133.33%
Josh Poimboeuf410.00%133.33%
Total40100.00%3100.00%

EXPORT_SYMBOL_GPL(save_stack_trace_tsk); /* Userspace stacktrace - based on kernel/trace/trace_sysprof.c */ struct stack_frame_user { const void __user *next_fp; unsigned long ret_addr; };
static int copy_stack_frame(const void __user *fp, struct stack_frame_user *frame) { int ret; if (!access_ok(VERIFY_READ, fp, sizeof(*frame))) return 0; ret = 1; pagefault_disable(); if (__copy_from_user_inatomic(frame, fp, sizeof(*frame))) ret = 0; pagefault_enable(); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
Török Edwin7098.59%150.00%
Frédéric Weisbecker11.41%150.00%
Total71100.00%2100.00%


static inline void __save_stack_trace_user(struct stack_trace *trace) { const struct pt_regs *regs = task_pt_regs(current); const void __user *fp = (const void __user *)regs->bp; if (trace->nr_entries < trace->max_entries) trace->entries[trace->nr_entries++] = regs->ip; while (trace->nr_entries < trace->max_entries) { struct stack_frame_user frame; frame.next_fp = NULL; frame.ret_addr = 0; if (!copy_stack_frame(fp, &frame)) break; if ((unsigned long)fp < regs->sp) break; if (frame.ret_addr) { trace->entries[trace->nr_entries++] = frame.ret_addr; } if (fp == frame.next_fp) break; fp = frame.next_fp; } }

Contributors

PersonTokensPropCommitsCommitProp
Török Edwin15299.35%266.67%
Frédéric Weisbecker10.65%133.33%
Total153100.00%3100.00%


void save_stack_trace_user(struct stack_trace *trace) { /* * Trace user stack if we are not a kernel thread */ if (current->mm) { __save_stack_trace_user(trace); } if (trace->nr_entries < trace->max_entries) trace->entries[trace->nr_entries++] = ULONG_MAX; }

Contributors

PersonTokensPropCommitsCommitProp
Török Edwin46100.00%2100.00%
Total46100.00%2100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
Török Edwin28647.12%28.70%
Josh Poimboeuf11719.28%14.35%
Ingo Molnar599.72%521.74%
Arjan van de Ven325.27%313.04%
Andi Kleen315.11%14.35%
Oleg Nesterov182.97%14.35%
Vegard Nossum162.64%28.70%
Andrew Lutomirski142.31%14.35%
Alexei Starovoitov111.81%14.35%
Catalin Marinas101.65%14.35%
Masami Hiramatsu40.66%14.35%
Soeren Sandmann Pedersen40.66%14.35%
Frédéric Weisbecker30.49%14.35%
Paul Gortmaker10.16%14.35%
Namhyung Kim10.16%14.35%
Total607100.00%23100.00%
Directory: arch/x86/kernel
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.