Commit 0b479289 authored by Dave Young's avatar Dave Young Committed by James Toy

When syslog is not possible, at the same time there's no serial/net

console available, it will be hard to read the printk messages.  For
example oops/panic/warning messages in shutdown phase.

Add a printk delay feature, we can make each printk message delay some
milliseconds.

Setting the delay by proc/sysctl interface: /proc/sys/kernel/printk_delay

The value range from 0 - 10000, default value is 0
Signed-off-by: default avatarDave Young <hidave.darkstar@gmail.com>
Acked-by: default avatarIngo Molnar <mingo@elte.hu>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
parent 5c55f85a
...@@ -313,6 +313,14 @@ send before ratelimiting kicks in. ...@@ -313,6 +313,14 @@ send before ratelimiting kicks in.
============================================================== ==============================================================
printk_delay:
Delay each printk message in printk_delay milliseconds
Value from 0 - 10000 is allowed.
==============================================================
randomize-va-space: randomize-va-space:
This option can be used to select the type of process address This option can be used to select the type of process address
......
...@@ -246,6 +246,8 @@ extern int printk_ratelimit(void); ...@@ -246,6 +246,8 @@ extern int printk_ratelimit(void);
extern bool printk_timed_ratelimit(unsigned long *caller_jiffies, extern bool printk_timed_ratelimit(unsigned long *caller_jiffies,
unsigned int interval_msec); unsigned int interval_msec);
extern int printk_delay_msec;
/* /*
* Print a one-time message (analogous to WARN_ONCE() et al): * Print a one-time message (analogous to WARN_ONCE() et al):
*/ */
......
...@@ -163,6 +163,7 @@ enum ...@@ -163,6 +163,7 @@ enum
KERN_MAX_LOCK_DEPTH=74, KERN_MAX_LOCK_DEPTH=74,
KERN_NMI_WATCHDOG=75, /* int: enable/disable nmi watchdog */ KERN_NMI_WATCHDOG=75, /* int: enable/disable nmi watchdog */
KERN_PANIC_ON_NMI=76, /* int: whether we will panic on an unrecovered */ KERN_PANIC_ON_NMI=76, /* int: whether we will panic on an unrecovered */
KERN_PRINTK_DELAY = 77, /* int: tune printk delay*/
}; };
......
...@@ -653,6 +653,19 @@ static int recursion_bug; ...@@ -653,6 +653,19 @@ static int recursion_bug;
static int new_text_line = 1; static int new_text_line = 1;
static char printk_buf[1024]; static char printk_buf[1024];
int printk_delay_msec;
static inline void printk_delay(void)
{
if (unlikely(printk_delay_msec)) {
int m = printk_delay_msec;
while (m--) {
mdelay(1);
touch_nmi_watchdog();
}
}
}
asmlinkage int vprintk(const char *fmt, va_list args) asmlinkage int vprintk(const char *fmt, va_list args)
{ {
int printed_len = 0; int printed_len = 0;
...@@ -662,6 +675,7 @@ asmlinkage int vprintk(const char *fmt, va_list args) ...@@ -662,6 +675,7 @@ asmlinkage int vprintk(const char *fmt, va_list args)
char *p; char *p;
boot_delay_msec(); boot_delay_msec();
printk_delay();
preempt_disable(); preempt_disable();
/* This stops the holder of console_sem just where we want him */ /* This stops the holder of console_sem just where we want him */
......
...@@ -105,6 +105,7 @@ static int __maybe_unused one = 1; ...@@ -105,6 +105,7 @@ static int __maybe_unused one = 1;
static int __maybe_unused two = 2; static int __maybe_unused two = 2;
static unsigned long one_ul = 1; static unsigned long one_ul = 1;
static int one_hundred = 100; static int one_hundred = 100;
static int ten_thousand = 10000;
/* this is needed for the proc_doulongvec_minmax of vm_dirty_bytes */ /* this is needed for the proc_doulongvec_minmax of vm_dirty_bytes */
static unsigned long dirty_bytes_min = 2 * PAGE_SIZE; static unsigned long dirty_bytes_min = 2 * PAGE_SIZE;
...@@ -713,6 +714,17 @@ static struct ctl_table kern_table[] = { ...@@ -713,6 +714,17 @@ static struct ctl_table kern_table[] = {
.mode = 0644, .mode = 0644,
.proc_handler = &proc_dointvec, .proc_handler = &proc_dointvec,
}, },
{
.ctl_name = KERN_PRINTK_DELAY,
.procname = "printk_delay",
.data = &printk_delay_msec,
.maxlen = sizeof(int),
.mode = 0644,
.proc_handler = &proc_dointvec_minmax,
.strategy = &sysctl_intvec,
.extra1 = &zero,
.extra2 = &ten_thousand,
},
#endif #endif
{ {
.ctl_name = KERN_NGROUPS_MAX, .ctl_name = KERN_NGROUPS_MAX,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment