cregit-Linux how code gets into the kernel

Release 4.11 kernel/panic.c

Directory: kernel
/*
 *  linux/kernel/panic.c
 *
 *  Copyright (C) 1991, 1992  Linus Torvalds
 */

/*
 * This function is used through-out the kernel (including mm and fs)
 * to indicate a major problem.
 */
#include <linux/debug_locks.h>
#include <linux/sched/debug.h>
#include <linux/interrupt.h>
#include <linux/kmsg_dump.h>
#include <linux/kallsyms.h>
#include <linux/notifier.h>
#include <linux/module.h>
#include <linux/random.h>
#include <linux/ftrace.h>
#include <linux/reboot.h>
#include <linux/delay.h>
#include <linux/kexec.h>
#include <linux/sched.h>
#include <linux/sysrq.h>
#include <linux/init.h>
#include <linux/nmi.h>
#include <linux/console.h>
#include <linux/bug.h>


#define PANIC_TIMER_STEP 100

#define PANIC_BLINK_SPD 18


int panic_on_oops = CONFIG_PANIC_ON_OOPS_VALUE;

static unsigned long tainted_mask;

static int pause_on_oops;

static int pause_on_oops_flag;
static DEFINE_SPINLOCK(pause_on_oops_lock);

bool crash_kexec_post_notifiers;

int panic_on_warn __read_mostly;


int panic_timeout = CONFIG_PANIC_TIMEOUT;

EXPORT_SYMBOL_GPL(panic_timeout);


ATOMIC_NOTIFIER_HEAD(panic_notifier_list);


EXPORT_SYMBOL(panic_notifier_list);


static long no_blink(int state) { return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Tamuki Shoichi12100.00%1100.00%
Total12100.00%1100.00%

/* Returns how long it waited in ms */ long (*panic_blink)(int state); EXPORT_SYMBOL(panic_blink); /* * Stop ourself in panic -- architecture code may override this */
void __weak panic_smp_self_stop(void) { while (1) cpu_relax(); }

Contributors

PersonTokensPropCommitsCommitProp
Michael Holzheu15100.00%1100.00%
Total15100.00%1100.00%

/* * Stop ourselves in NMI context if another CPU has already panicked. Arch code * may override this to prepare for crash dumping, e.g. save regs info. */
void __weak nmi_panic_self_stop(struct pt_regs *regs) { panic_smp_self_stop(); }

Contributors

PersonTokensPropCommitsCommitProp
Hidehiro Kawai14100.00%1100.00%
Total14100.00%1100.00%

/* * Stop other CPUs in panic. Architecture dependent code may override this * with more suitable version. For example, if the architecture supports * crash dump, it should save registers of each stopped CPU and disable * per-CPU features such as virtualization extensions. */
void __weak crash_smp_send_stop(void) { static int cpus_stopped; /* * This function can be called twice in panic path, but obviously * we execute this only once. */ if (cpus_stopped) return; /* * Note smp_send_stop is the usual smp shutdown function, which * unfortunately means it may not be hardened to work in a panic * situation. */ smp_send_stop(); cpus_stopped = 1; }

Contributors

PersonTokensPropCommitsCommitProp
Hidehiro Kawai26100.00%1100.00%
Total26100.00%1100.00%

atomic_t panic_cpu = ATOMIC_INIT(PANIC_CPU_INVALID); /* * A variant of panic() called from NMI context. We return if we've already * panicked on this CPU. If another CPU already panicked, loop in * nmi_panic_self_stop() which can provide architecture dependent code such * as saving register state for crash dump. */
void nmi_panic(struct pt_regs *regs, const char *msg) { int old_cpu, cpu; cpu = raw_smp_processor_id(); old_cpu = atomic_cmpxchg(&panic_cpu, PANIC_CPU_INVALID, cpu); if (old_cpu == PANIC_CPU_INVALID) panic("%s", msg); else if (old_cpu != cpu) nmi_panic_self_stop(regs); }

Contributors

PersonTokensPropCommitsCommitProp
Hidehiro Kawai62100.00%1100.00%
Total62100.00%1100.00%

EXPORT_SYMBOL(nmi_panic); /** * panic - halt the system * @fmt: The text string to print * * Display a message, then perform cleanups. * * This function never returns. */
void panic(const char *fmt, ...) { static char buf[1024]; va_list args; long i, i_next = 0; int state = 0; int old_cpu, this_cpu; bool _crash_kexec_post_notifiers = crash_kexec_post_notifiers; /* * Disable local interrupts. This will prevent panic_smp_self_stop * from deadlocking the first cpu that invokes the panic, since * there is nothing to prevent an interrupt handler (that runs * after setting panic_cpu) from invoking panic() again. */ local_irq_disable(); /* * It's possible to come here directly from a panic-assertion and * not have preempt disabled. Some functions called from here want * preempt to be disabled. No point enabling it later though... * * Only one CPU is allowed to execute the panic code from here. For * multiple parallel invocations of panic, all other CPUs either * stop themself or will wait until they are stopped by the 1st CPU * with smp_send_stop(). * * `old_cpu == PANIC_CPU_INVALID' means this is the 1st CPU which * comes here, so go ahead. * `old_cpu == this_cpu' means we came from nmi_panic() which sets * panic_cpu to this CPU. In this case, this is also the 1st CPU. */ this_cpu = raw_smp_processor_id(); old_cpu = atomic_cmpxchg(&panic_cpu, PANIC_CPU_INVALID, this_cpu); if (old_cpu != PANIC_CPU_INVALID && old_cpu != this_cpu) panic_smp_self_stop(); console_verbose(); bust_spinlocks(1); va_start(args, fmt); vsnprintf(buf, sizeof(buf), fmt, args); va_end(args); pr_emerg("Kernel panic - not syncing: %s\n", buf); #ifdef CONFIG_DEBUG_BUGVERBOSE /* * Avoid nested stack-dumping if a panic occurs during oops processing */ if (!test_taint(TAINT_DIE) && oops_in_progress <= 1) dump_stack(); #endif /* * If we have crashed and we have a crash kernel loaded let it handle * everything else. * If we want to run this after calling panic_notifiers, pass * the "crash_kexec_post_notifiers" option to the kernel. * * Bypass the panic_cpu check and call __crash_kexec directly. */ if (!_crash_kexec_post_notifiers) { printk_safe_flush_on_panic(); __crash_kexec(NULL); /* * Note smp_send_stop is the usual smp shutdown function, which * unfortunately means it may not be hardened to work in a * panic situation. */ smp_send_stop(); } else { /* * If we want to do crash dump after notifier calls and * kmsg_dump, we will need architecture dependent extra * works in addition to stopping other CPUs. */ crash_smp_send_stop(); } /* * Run any panic handlers, including those that might need to * add information to the kmsg dump output. */ atomic_notifier_call_chain(&panic_notifier_list, 0, buf); /* Call flush even twice. It tries harder with a single online CPU */ printk_safe_flush_on_panic(); kmsg_dump(KMSG_DUMP_PANIC); /* * If you doubt kdump always works fine in any situation, * "crash_kexec_post_notifiers" offers you a chance to run * panic_notifiers and dumping kmsg before kdump. * Note: since some panic_notifiers can make crashed kernel * more unstable, it can increase risks of the kdump failure too. * * Bypass the panic_cpu check and call __crash_kexec directly. */ if (_crash_kexec_post_notifiers) __crash_kexec(NULL); bust_spinlocks(0); /* * We may have ended up stopping the CPU holding the lock (in * smp_send_stop()) while still having some valuable data in the console * buffer. Try to acquire the lock then release it regardless of the * result. The release will also print the buffers out. Locks debug * should be disabled to avoid reporting bad unlock balance when * panic() is not being callled from OOPS. */ debug_locks_off(); console_flush_on_panic(); if (!panic_blink) panic_blink = no_blink; if (panic_timeout > 0) { /* * Delay timeout seconds before rebooting the machine. * We can't use the "normal" timers since we just panicked. */ pr_emerg("Rebooting in %d seconds..\n", panic_timeout); for (i = 0; i < panic_timeout * 1000; i += PANIC_TIMER_STEP) { touch_nmi_watchdog(); if (i >= i_next) { i += panic_blink(state ^= 1); i_next = i + 3600 / PANIC_BLINK_SPD; } mdelay(PANIC_TIMER_STEP); } } if (panic_timeout != 0) { /* * This will not be a clean reboot, with everything * shutting down. But if there is a chance of * rebooting the system it will be rebooted. */ emergency_restart(); } #ifdef __sparc__ { extern int stop_a_enabled; /* Make sure the user can actually press Stop-A (L1-A) */ stop_a_enabled = 1; pr_emerg("Press Stop-A (L1-A) from sun keyboard or send break\n" "twice on console to return to the boot prom\n"); } #endif #if defined(CONFIG_S390) { unsigned long caller; caller = (unsigned long)__builtin_return_address(0); disabled_wait(caller); } #endif pr_emerg("---[ end Kernel panic - not syncing: %s\n", buf); local_irq_enable(); for (i = 0; ; i += PANIC_TIMER_STEP) { touch_softlockup_watchdog(); if (i >= i_next) { i += panic_blink(state ^= 1); i_next = i + 3600 / PANIC_BLINK_SPD; } mdelay(PANIC_TIMER_STEP); } }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds (pre-git)10325.37%816.33%
Tamuki Shoichi9122.41%12.04%
Hidehiro Kawai5012.32%48.16%
Ingo Molnar297.14%36.12%
Andrew Morton225.42%24.08%
Fabian Frederick102.46%12.04%
Andi Kleen102.46%24.08%
Hugh Dickins81.97%12.04%
Masami Hiramatsu81.97%12.04%
Jason Wessel71.72%12.04%
Petr Mladek61.48%12.04%
Vitaly Kuznetsov61.48%24.08%
Kees Cook61.48%12.04%
Linus Torvalds51.23%12.04%
Aaro Koskinen51.23%12.04%
Eric W. Biedermann51.23%24.08%
Michael Holzheu51.23%12.04%
Anton Blanchard40.99%24.08%
Alexander Nyberg30.74%12.04%
Seiji Aguchi30.74%12.04%
Daisuke Hatayama30.74%12.04%
Jan Beulich30.74%12.04%
Vikram Mulukutla30.74%12.04%
Vijay Kumar20.49%12.04%
Sergey Senozhatsky20.49%12.04%
Alan Stern10.25%12.04%
Jeff Dike10.25%12.04%
Tom 'spot' Callaway10.25%12.04%
Jiri Slaby10.25%12.04%
Tejun Heo10.25%12.04%
Martin Schwidefsky10.25%12.04%
Christian Bornträger10.25%12.04%
Total406100.00%49100.00%

EXPORT_SYMBOL(panic); /* * TAINT_FORCED_RMMOD could be a per-module flag but the module * is being removed anyway. */ const struct taint_flag taint_flags[TAINT_FLAGS_COUNT] = { { 'P', 'G', true }, /* TAINT_PROPRIETARY_MODULE */ { 'F', ' ', true }, /* TAINT_FORCED_MODULE */ { 'S', ' ', false }, /* TAINT_CPU_OUT_OF_SPEC */ { 'R', ' ', false }, /* TAINT_FORCED_RMMOD */ { 'M', ' ', false }, /* TAINT_MACHINE_CHECK */ { 'B', ' ', false }, /* TAINT_BAD_PAGE */ { 'U', ' ', false }, /* TAINT_USER */ { 'D', ' ', false }, /* TAINT_DIE */ { 'A', ' ', false }, /* TAINT_OVERRIDDEN_ACPI_TABLE */ { 'W', ' ', false }, /* TAINT_WARN */ { 'C', ' ', true }, /* TAINT_CRAP */ { 'I', ' ', false }, /* TAINT_FIRMWARE_WORKAROUND */ { 'O', ' ', true }, /* TAINT_OOT_MODULE */ { 'E', ' ', true }, /* TAINT_UNSIGNED_MODULE */ { 'L', ' ', false }, /* TAINT_SOFTLOCKUP */ { 'K', ' ', true }, /* TAINT_LIVEPATCH */ }; /** * print_tainted - return a string to represent the kernel taint state. * * 'P' - Proprietary module has been loaded. * 'F' - Module has been forcibly loaded. * 'S' - SMP with CPUs not designed for SMP. * 'R' - User forced a module unload. * 'M' - System experienced a machine check exception. * 'B' - System has hit bad_page. * 'U' - Userspace-defined naughtiness. * 'D' - Kernel has oopsed before * 'A' - ACPI table overridden. * 'W' - Taint on warning. * 'C' - modules from drivers/staging are loaded. * 'I' - Working around severe firmware bug. * 'O' - Out-of-tree module has been loaded. * 'E' - Unsigned module has been loaded. * 'L' - A soft lockup has previously occurred. * 'K' - Kernel has been live patched. * * The string is overwritten by the next call to print_tainted(). */
const char *print_tainted(void) { static char buf[TAINT_FLAGS_COUNT + sizeof("Tainted: ")]; if (tainted_mask) { char *s; int i; s = buf + sprintf(buf, "Tainted: "); for (i = 0; i < TAINT_FLAGS_COUNT; i++) { const struct taint_flag *t = &taint_flags[i]; *s++ = test_bit(i, &tainted_mask) ? t->c_true : t->c_false; } *s = 0; } else snprintf(buf, sizeof(buf), "Not tainted"); return buf; }

Contributors

PersonTokensPropCommitsCommitProp
Andi Kleen8676.11%125.00%
Linus Torvalds2017.70%125.00%
Petr Mladek54.42%125.00%
Larry Finger21.77%125.00%
Total113100.00%4100.00%


int test_taint(unsigned flag) { return test_bit(flag, &tainted_mask); }

Contributors

PersonTokensPropCommitsCommitProp
Andi Kleen17100.00%2100.00%
Total17100.00%2100.00%

EXPORT_SYMBOL(test_taint);
unsigned long get_taint(void) { return tainted_mask; }

Contributors

PersonTokensPropCommitsCommitProp
Andi Kleen872.73%150.00%
Andrew Morton327.27%150.00%
Total11100.00%2100.00%

/** * add_taint: add a taint flag if not already set. * @flag: one of the TAINT_* constants. * @lockdep_ok: whether lock debugging is still OK. * * If something bad has gone wrong, you'll want @lockdebug_ok = false, but for * some notewortht-but-not-corrupting cases, it can be set to true. */
void add_taint(unsigned flag, enum lockdep_ok lockdep_ok) { if (lockdep_ok == LOCKDEP_NOW_UNRELIABLE && __debug_locks_off()) pr_warn("Disabling lock debugging due to kernel taint\n"); set_bit(flag, &tainted_mask); }

Contributors

PersonTokensPropCommitsCommitProp
Andi Kleen1132.35%222.22%
Rusty Russell823.53%111.11%
Andrew Morton514.71%111.11%
Frédéric Weisbecker514.71%111.11%
Peter Zijlstra25.88%111.11%
Fabian Frederick12.94%111.11%
Arjan van de Ven12.94%111.11%
Ingo Molnar12.94%111.11%
Total34100.00%9100.00%

EXPORT_SYMBOL(add_taint);
static void spin_msec(int msecs) { int i; for (i = 0; i < msecs; i++) { touch_nmi_watchdog(); mdelay(1); } }

Contributors

PersonTokensPropCommitsCommitProp
Andrew Morton35100.00%1100.00%
Total35100.00%1100.00%

/* * It just happens that oops_enter() and oops_exit() are identically * implemented... */
static void do_oops_enter_exit(void) { unsigned long flags; static int spin_counter; if (!pause_on_oops) return; spin_lock_irqsave(&pause_on_oops_lock, flags); if (pause_on_oops_flag == 0) { /* This CPU may now print the oops message */ pause_on_oops_flag = 1; } else { /* We need to stall this CPU */ if (!spin_counter) { /* This CPU gets to do the counting */ spin_counter = pause_on_oops; do { spin_unlock(&pause_on_oops_lock); spin_msec(MSEC_PER_SEC); spin_lock(&pause_on_oops_lock); } while (--spin_counter); pause_on_oops_flag = 0; } else { /* This CPU waits for a different one */ while (spin_counter) { spin_unlock(&pause_on_oops_lock); spin_msec(1); spin_lock(&pause_on_oops_lock); } } } spin_unlock_irqrestore(&pause_on_oops_lock, flags); }

Contributors

PersonTokensPropCommitsCommitProp
Andrew Morton124100.00%1100.00%
Total124100.00%1100.00%

/* * Return true if the calling CPU is allowed to print oops-related info. * This is a bit racy.. */
int oops_may_print(void) { return pause_on_oops_flag == 0; }

Contributors

PersonTokensPropCommitsCommitProp
Andrew Morton12100.00%1100.00%
Total12100.00%1100.00%

/* * Called when the architecture enters its oops handler, before it prints * anything. If this is the first CPU to oops, and it's oopsing the first * time then let it proceed. * * This is all enabled by the pause_on_oops kernel boot option. We do all * this to ensure that oopses don't scroll off the screen. It has the * side-effect of preventing later-oopsing CPUs from mucking up the display, * too. * * It turns out that the CPU which is allowed to print ends up pausing for * the right duration, whereas all the other CPUs pause for twice as long: * once in oops_enter(), once in oops_exit(). */
void oops_enter(void) { tracing_off(); /* can't trust the integrity of the kernel anymore: */ debug_locks_off(); do_oops_enter_exit(); }

Contributors

PersonTokensPropCommitsCommitProp
Andrew Morton1058.82%125.00%
Arjan van de Ven317.65%125.00%
Thomas Gleixner317.65%125.00%
Ingo Molnar15.88%125.00%
Total17100.00%4100.00%

/* * 64-bit random ID for oopses: */ static u64 oops_id;
static int init_oops_id(void) { if (!oops_id) get_random_bytes(&oops_id, sizeof(oops_id)); else oops_id++; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Arjan van de Ven31100.00%2100.00%
Total31100.00%2100.00%

late_initcall(init_oops_id);
void print_oops_end_marker(void) { init_oops_id(); pr_warn("---[ end trace %016llx ]---\n", (unsigned long long)oops_id); }

Contributors

PersonTokensPropCommitsCommitProp
Arjan van de Ven1568.18%250.00%
Andrew Morton627.27%125.00%
Fabian Frederick14.55%125.00%
Total22100.00%4100.00%

/* * Called when the architecture exits its oops handler, after printing * everything. */
void oops_exit(void) { do_oops_enter_exit(); print_oops_end_marker(); kmsg_dump(KMSG_DUMP_OOPS); }

Contributors

PersonTokensPropCommitsCommitProp
Arjan van de Ven1372.22%150.00%
Simon Kågström527.78%150.00%
Total18100.00%2100.00%

struct warn_args { const char *fmt; va_list args; };
void __warn(const char *file, int line, void *caller, unsigned taint, struct pt_regs *regs, struct warn_args *args) { disable_trace_on_warning(); pr_warn("------------[ cut here ]------------\n"); if (file) pr_warn("WARNING: CPU: %d PID: %d at %s:%d %pS\n", raw_smp_processor_id(), current->pid, file, line, caller); else pr_warn("WARNING: CPU: %d PID: %d at %pS\n", raw_smp_processor_id(), current->pid, caller); if (args) vprintk(args->fmt, args->args); if (panic_on_warn) { /* * This thread may hit another WARN() in the panic path. * Resetting this prevents additional WARN() from panicking the * system on this thread. Other threads are blocked by the * panic_mutex in panic(). */ panic_on_warn = 0; panic("panic_on_warn set ...\n"); } print_modules(); if (regs) show_regs(regs); else dump_stack(); print_oops_end_marker(); /* Just a warning, don't kill lockdep. */ add_taint(taint, LOCKDEP_STILL_OK); }

Contributors

PersonTokensPropCommitsCommitProp
Arjan van de Ven5138.35%222.22%
Josh Poimboeuf3727.82%111.11%
Prarit Bhargava1612.03%111.11%
Linus Torvalds107.52%111.11%
Alex Thorlton96.77%111.11%
Ben Hutchings43.01%111.11%
Steven Rostedt32.26%111.11%
Rusty Russell32.26%111.11%
Total133100.00%9100.00%

#ifdef WANT_WARN_ON_SLOWPATH
void warn_slowpath_fmt(const char *file, int line, const char *fmt, ...) { struct warn_args args; args.fmt = fmt; va_start(args.args, fmt); __warn(file, line, __builtin_return_address(0), TAINT_WARN, NULL, &args); va_end(args.args); }

Contributors

PersonTokensPropCommitsCommitProp
Linus Torvalds5990.77%133.33%
Josh Poimboeuf46.15%133.33%
Ben Hutchings23.08%133.33%
Total65100.00%3100.00%

EXPORT_SYMBOL(warn_slowpath_fmt);
void warn_slowpath_fmt_taint(const char *file, int line, unsigned taint, const char *fmt, ...) { struct warn_args args; args.fmt = fmt; va_start(args.args, fmt); __warn(file, line, __builtin_return_address(0), taint, NULL, &args); va_end(args.args); }

Contributors

PersonTokensPropCommitsCommitProp
Ben Hutchings6494.12%150.00%
Josh Poimboeuf45.88%150.00%
Total68100.00%2100.00%

EXPORT_SYMBOL(warn_slowpath_fmt_taint);
void warn_slowpath_null(const char *file, int line) { __warn(file, line, __builtin_return_address(0), TAINT_WARN, NULL, NULL); }

Contributors

PersonTokensPropCommitsCommitProp
Andi Kleen2064.52%125.00%
Linus Torvalds619.35%125.00%
Josh Poimboeuf39.68%125.00%
Ben Hutchings26.45%125.00%
Total31100.00%4100.00%

EXPORT_SYMBOL(warn_slowpath_null); #endif #ifdef CONFIG_CC_STACKPROTECTOR /* * Called when gcc's -fstack-protector feature is used, and * gcc detects corruption of the on-stack canary value */
__visible void __stack_chk_fail(void) { panic("stack-protector: Kernel stack is corrupted in: %p\n", __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
Arjan van de Ven1161.11%133.33%
Ingo Molnar633.33%133.33%
Andi Kleen15.56%133.33%
Total18100.00%3100.00%

EXPORT_SYMBOL(__stack_chk_fail); #endif core_param(panic, panic_timeout, int, 0644); core_param(pause_on_oops, pause_on_oops, int, 0644); core_param(panic_on_warn, panic_on_warn, int, 0644); core_param(crash_kexec_post_notifiers, crash_kexec_post_notifiers, bool, 0644);
static int __init oops_setup(char *s) { if (!s) return -EINVAL; if (!strcmp(s, "panic")) panic_on_oops = 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Olaf Hering37100.00%1100.00%
Total37100.00%1100.00%

early_param("oops", oops_setup);

Overall Contributors

PersonTokensPropCommitsCommitProp
Andrew Morton24614.04%43.67%
Andi Kleen22913.07%65.50%
Hidehiro Kawai1759.99%65.50%
Arjan van de Ven1578.96%87.34%
Linus Torvalds1216.91%43.67%
Linus Torvalds (pre-git)1206.85%1311.93%
Tamuki Shoichi1136.45%10.92%
Ben Hutchings895.08%32.75%
Petr Mladek673.82%21.83%
Ingo Molnar553.14%65.50%
Josh Poimboeuf553.14%10.92%
Olaf Hering442.51%10.92%
Rusty Russell352.00%32.75%
Prarit Bhargava311.77%10.92%
Michael Holzheu211.20%10.92%
Masami Hiramatsu150.86%10.92%
Fabian Frederick120.68%10.92%
Arnaldo Carvalho de Melo120.68%10.92%
Alex Thorlton90.51%10.92%
Vitaly Kuznetsov90.51%21.83%
Hugh Dickins80.46%10.92%
Simon Kågström80.46%10.92%
Eric W. Biedermann70.40%21.83%
Seth Jennings70.40%10.92%
Jason Wessel70.40%10.92%
Joshua Hunt60.34%10.92%
Kees Cook60.34%10.92%
Steven Rostedt60.34%10.92%
Mathieu Desnoyers50.29%10.92%
Huang Ying50.29%10.92%
Frédéric Weisbecker50.29%10.92%
Alan Stern50.29%10.92%
Aaro Koskinen50.29%10.92%
Nicholas Piggin40.23%21.83%
Anton Blanchard40.23%21.83%
Vikram Mulukutla30.17%10.92%
Nur Hussein30.17%10.92%
Thomas Gleixner30.17%10.92%
Seiji Aguchi30.17%10.92%
Alexander Nyberg30.17%10.92%
Daisuke Hatayama30.17%10.92%
Jan Beulich30.17%10.92%
Jason Baron20.11%10.92%
Pavel Emelyanov20.11%10.92%
Sergey Senozhatsky20.11%10.92%
Dave Jones20.11%10.92%
Éric Piel20.11%10.92%
Kyle McMartin20.11%10.92%
Larry Finger20.11%10.92%
Theodore Y. Ts'o20.11%10.92%
Peter Zijlstra20.11%10.92%
Vijay Kumar20.11%10.92%
Tom 'spot' Callaway10.06%10.92%
Jiri Slaby10.06%10.92%
Christian Bornträger10.06%10.92%
Martin Schwidefsky10.06%10.92%
Tejun Heo10.06%10.92%
Heiko Carstens10.06%10.92%
Jeff Dike10.06%10.92%
Greg Kroah-Hartman10.06%10.92%
Total1752100.00%109100.00%
Directory: kernel
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.