1234567891011121314151617181920212223242526272829303132333435363738394041424344454647 |
- // SPDX-License-Identifier: GPL-2.0-or-later
- /*
- * printk_safe.c - Safe printk for printk-deadlock-prone contexts
- */
- #include <linux/preempt.h>
- #include <linux/kdb.h>
- #include <linux/smp.h>
- #include <linux/cpumask.h>
- #include <linux/printk.h>
- #include <linux/kprobes.h>
- #include "internal.h"
- static DEFINE_PER_CPU(int, printk_context);
- /* Can be preempted by NMI. */
- void __printk_safe_enter(void)
- {
- this_cpu_inc(printk_context);
- }
- /* Can be preempted by NMI. */
- void __printk_safe_exit(void)
- {
- this_cpu_dec(printk_context);
- }
- asmlinkage int vprintk(const char *fmt, va_list args)
- {
- #ifdef CONFIG_KGDB_KDB
- /* Allow to pass printk() to kdb but avoid a recursion. */
- if (unlikely(kdb_trap_printk && kdb_printf_cpu < 0))
- return vkdb_printf(KDB_MSGSRC_PRINTK, fmt, args);
- #endif
- /*
- * Use the main logbuf even in NMI. But avoid calling console
- * drivers that might have their own locks.
- */
- if (this_cpu_read(printk_context) || in_nmi())
- return vprintk_deferred(fmt, args);
- /* No obstacles. */
- return vprintk_default(fmt, args);
- }
- EXPORT_SYMBOL(vprintk);
|