Commit 5341b93d authored by John Ogness's avatar John Ogness Committed by Petr Mladek

printk: wake waiters for safe and NMI contexts

When printk() is called from safe or NMI contexts, it will directly
store the record (vprintk_store()) and then defer the console output.
However, defer_console_output() only causes console printing and does
not wake any waiters of new records.

Wake waiters from defer_console_output() so that they also are aware
of the new records from safe and NMI contexts.

Fixes: 03fc7f9c ("printk/nmi: Prevent deadlock when accessing the main log buffer in NMI")
Signed-off-by: default avatarJohn Ogness <john.ogness@linutronix.de>
Reviewed-by: default avatarPetr Mladek <pmladek@suse.com>
Signed-off-by: default avatarPetr Mladek <pmladek@suse.com>
Link: https://lore.kernel.org/r/20220421212250.565456-6-john.ogness@linutronix.de
parent 938ba408
...@@ -754,7 +754,7 @@ static ssize_t devkmsg_read(struct file *file, char __user *buf, ...@@ -754,7 +754,7 @@ static ssize_t devkmsg_read(struct file *file, char __user *buf,
* prepare_to_wait_event() pairs with the full memory barrier * prepare_to_wait_event() pairs with the full memory barrier
* within wq_has_sleeper(). * within wq_has_sleeper().
* *
* This pairs with wake_up_klogd:A. * This pairs with __wake_up_klogd:A.
*/ */
ret = wait_event_interruptible(log_wait, ret = wait_event_interruptible(log_wait,
prb_read_valid(prb, prb_read_valid(prb,
...@@ -1532,7 +1532,7 @@ static int syslog_print(char __user *buf, int size) ...@@ -1532,7 +1532,7 @@ static int syslog_print(char __user *buf, int size)
* prepare_to_wait_event() pairs with the full memory barrier * prepare_to_wait_event() pairs with the full memory barrier
* within wq_has_sleeper(). * within wq_has_sleeper().
* *
* This pairs with wake_up_klogd:A. * This pairs with __wake_up_klogd:A.
*/ */
len = wait_event_interruptible(log_wait, len = wait_event_interruptible(log_wait,
prb_read_valid(prb, seq, NULL)); /* LMM(syslog_print:A) */ prb_read_valid(prb, seq, NULL)); /* LMM(syslog_print:A) */
...@@ -3332,7 +3332,7 @@ static void wake_up_klogd_work_func(struct irq_work *irq_work) ...@@ -3332,7 +3332,7 @@ static void wake_up_klogd_work_func(struct irq_work *irq_work)
static DEFINE_PER_CPU(struct irq_work, wake_up_klogd_work) = static DEFINE_PER_CPU(struct irq_work, wake_up_klogd_work) =
IRQ_WORK_INIT_LAZY(wake_up_klogd_work_func); IRQ_WORK_INIT_LAZY(wake_up_klogd_work_func);
void wake_up_klogd(void) static void __wake_up_klogd(int val)
{ {
if (!printk_percpu_data_ready()) if (!printk_percpu_data_ready())
return; return;
...@@ -3349,22 +3349,26 @@ void wake_up_klogd(void) ...@@ -3349,22 +3349,26 @@ void wake_up_klogd(void)
* *
* This pairs with devkmsg_read:A and syslog_print:A. * This pairs with devkmsg_read:A and syslog_print:A.
*/ */
if (wq_has_sleeper(&log_wait)) { /* LMM(wake_up_klogd:A) */ if (wq_has_sleeper(&log_wait) || /* LMM(__wake_up_klogd:A) */
this_cpu_or(printk_pending, PRINTK_PENDING_WAKEUP); (val & PRINTK_PENDING_OUTPUT)) {
this_cpu_or(printk_pending, val);
irq_work_queue(this_cpu_ptr(&wake_up_klogd_work)); irq_work_queue(this_cpu_ptr(&wake_up_klogd_work));
} }
preempt_enable(); preempt_enable();
} }
void defer_console_output(void) void wake_up_klogd(void)
{ {
if (!printk_percpu_data_ready()) __wake_up_klogd(PRINTK_PENDING_WAKEUP);
return; }
preempt_disable(); void defer_console_output(void)
this_cpu_or(printk_pending, PRINTK_PENDING_OUTPUT); {
irq_work_queue(this_cpu_ptr(&wake_up_klogd_work)); /*
preempt_enable(); * New messages may have been added directly to the ringbuffer
* using vprintk_store(), so wake any waiters as well.
*/
__wake_up_klogd(PRINTK_PENDING_WAKEUP | PRINTK_PENDING_OUTPUT);
} }
void printk_trigger_flush(void) void printk_trigger_flush(void)
......
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