Received: by 2002:a05:6a10:8c0a:0:0:0:0 with SMTP id go10csp455806pxb; Wed, 27 Jan 2021 11:53:13 -0800 (PST) X-Google-Smtp-Source: ABdhPJyHiB6t612q/FhJqgn8Y+fZhCpSvxxyN35+cyboKLYQX7vsPl/YxwobtbcH/765XklphRvP X-Received: by 2002:a17:907:9879:: with SMTP id ko25mr8162033ejc.524.1611777193631; Wed, 27 Jan 2021 11:53:13 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1611777193; cv=none; d=google.com; s=arc-20160816; b=eUcbreqJZE/NMOCp65u1KSabY0711puBIoy2fIbo5l5Xw6WTaj0rIsJTNCZFW+Tm4U uqlVmXXDeq2u51jwBwsTC+Pq/mfASwzJfR0wfjkCT7+qdTTwh81ILbE/csDst0VYLqE+ HUdIJNr25z3CHmFekuDZUYH8McepGqB9EOSRaGiIFuHZ1IR+RVNxSJkQ03ey+6Rf7ecS 7wVdFywa24USoNReG8stnvMurJM5yrKRlSQR3Ss5wErPkAJhUB7klgpZZz8RGA6j+GVn GWd5P6sFdzLGhxbmszbcBUtAlVeGQZOlzaAVXwUCguQTIdMuNnJ6q6fuCKDQRIWpcnzx QEaw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:mime-version :references:in-reply-to:message-id:date:subject:cc:to:dkim-signature :dkim-signature:from; bh=NWAD//HRgAQT7jAM3o6b5dUhjIuyj5O8XAiKIq7uDyU=; b=d8zh1BfX0/MaYOUIXMTTNPwJff9tOLM+oj/EIpc5rZyHZtJJACTVYyOfJ9cKPLXoqY dnXrN/9+4Ids5dTsTd6/7NUwKj01HWKf7wRIqawzD9Xgc67GT9ZtEKRFIid5SWLejqsH YJjfAuRpp/Gi4DCjQMR+/2VB2eToSOJD7bCWo465fIQztpZJFseXvd5vGNMjjMXJrL1D mMvBoKqra3aBkxSAczIg/YyhpfxSDEDeO0DsgzU3ADaNzV7aiQBVF2DJ3pO4C52/SFQE yGrpmhPG5tNeSWYNm1b2gS6Jq7DpVJMuRbU3HdNfUqdzt/ToSReX39yCSyr5Evu+KKgm dsvQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linutronix.de header.s=2020 header.b=EoiiqRQv; dkim=neutral (no key) header.i=@linutronix.de header.b=4YVV7lgC; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=linutronix.de Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id jt18si1221523ejb.633.2021.01.27.11.52.48; Wed, 27 Jan 2021 11:53:13 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@linutronix.de header.s=2020 header.b=EoiiqRQv; dkim=neutral (no key) header.i=@linutronix.de header.b=4YVV7lgC; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=linutronix.de Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S233918AbhA0Da0 (ORCPT + 99 others); Tue, 26 Jan 2021 22:30:26 -0500 Received: from Galois.linutronix.de ([193.142.43.55]:51962 "EHLO galois.linutronix.de" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1728910AbhAZVQl (ORCPT ); Tue, 26 Jan 2021 16:16:41 -0500 From: John Ogness DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linutronix.de; s=2020; t=1611695754; h=from:from:reply-to:subject:subject:date:date:message-id:message-id: to:to:cc:cc:mime-version:mime-version: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=NWAD//HRgAQT7jAM3o6b5dUhjIuyj5O8XAiKIq7uDyU=; b=EoiiqRQv5sxFcUWlvmP1KkJ4Z7kUd1xOyhlemMOK0MXE0vA3EW+EgiDPj5q2Xs02J5RlTi Tnoxe+Bx77fd54ohcKUtlcESfmJUI4pQVzxvSXUp2gsa/ODVx2B8n5C+9A6lPC+VOWEYw2 +IvvOuYERRAu2fsdXgJI6nzTgBpdsuHNzgdNUH+OjtPf99jc4Zabb8G7DslCb3MknyxNBj hcfr7PZn8LBXAAHug1yvnXyB36B+HZ9wFA3zEljpJmcfJ+d3FMEkea85AvdPbL97z9KThy fRTVSzRmQQr71ZwHNTRjsc7pcrihY4k/bWSj8QQ+h1mh3Qt2A4fJ5qyRATBrvA== DKIM-Signature: v=1; a=ed25519-sha256; c=relaxed/relaxed; d=linutronix.de; s=2020e; t=1611695754; h=from:from:reply-to:subject:subject:date:date:message-id:message-id: to:to:cc:cc:mime-version:mime-version: content-transfer-encoding:content-transfer-encoding: in-reply-to:in-reply-to:references:references; bh=NWAD//HRgAQT7jAM3o6b5dUhjIuyj5O8XAiKIq7uDyU=; b=4YVV7lgCYpqk0f1DRSuMlC5yU8mQLn/eccDwCZF3mAboq20rpme1Y7aZ5qxEVCWeWRE5BG WJqC2pxn6/UOf2Cg== To: Petr Mladek Cc: Sergey Senozhatsky , Sergey Senozhatsky , Steven Rostedt , Thomas Gleixner , linux-kernel@vger.kernel.org Subject: [PATCH printk-rework 05/12] printk: use seqcount_latch for clear_seq Date: Tue, 26 Jan 2021 22:21:44 +0106 Message-Id: <20210126211551.26536-6-john.ogness@linutronix.de> In-Reply-To: <20210126211551.26536-1-john.ogness@linutronix.de> References: <20210126211551.26536-1-john.ogness@linutronix.de> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org kmsg_dump_rewind_nolock() locklessly reads @clear_seq. However, this is not done atomically. Since @clear_seq is 64-bit, this cannot be an atomic operation for all platforms. Therefore, use a seqcount_latch to allow readers to always read a consistent value. Signed-off-by: John Ogness --- kernel/printk/printk.c | 58 ++++++++++++++++++++++++++++++++++++------ 1 file changed, 50 insertions(+), 8 deletions(-) diff --git a/kernel/printk/printk.c b/kernel/printk/printk.c index 5faf9c0db171..d41fd3306f4e 100644 --- a/kernel/printk/printk.c +++ b/kernel/printk/printk.c @@ -402,8 +402,21 @@ static u64 console_seq; static u64 exclusive_console_stop_seq; static unsigned long console_dropped; -/* the next printk record to read after the last 'clear' command */ -static u64 clear_seq; +struct latched_seq { + seqcount_latch_t latch; + u64 val[2]; +}; + +/* + * The next printk record to read after the last 'clear' command. There are + * two copies (updated with seqcount_latch) so that reads can locklessly + * access a valid value. Writers are synchronized by @logbuf_lock. + */ +static struct latched_seq clear_seq = { + .latch = SEQCNT_LATCH_ZERO(clear_seq.latch), + .val[0] = 0, + .val[1] = 0, +}; #ifdef CONFIG_PRINTK_CALLER #define PREFIX_MAX 48 @@ -452,6 +465,31 @@ bool printk_percpu_data_ready(void) return __printk_percpu_data_ready; } +/* Must be called under logbuf_lock. */ +static void latched_seq_write(struct latched_seq *ls, u64 val) +{ + raw_write_seqcount_latch(&ls->latch); + ls->val[0] = val; + raw_write_seqcount_latch(&ls->latch); + ls->val[1] = val; +} + +/* Can be called from any context. */ +static u64 latched_seq_read_nolock(struct latched_seq *ls) +{ + unsigned int seq; + unsigned int idx; + u64 val; + + do { + seq = raw_read_seqcount_latch(&ls->latch); + idx = seq & 0x1; + val = ls->val[idx]; + } while (read_seqcount_latch_retry(&ls->latch, seq)); + + return val; +} + /* Return log buffer address */ char *log_buf_addr_get(void) { @@ -797,7 +835,7 @@ static loff_t devkmsg_llseek(struct file *file, loff_t offset, int whence) * like issued by 'dmesg -c'. Reading /dev/kmsg itself * changes no global state, and does not clear anything. */ - user->seq = clear_seq; + user->seq = latched_seq_read_nolock(&clear_seq); break; case SEEK_END: /* after the last record */ @@ -955,6 +993,9 @@ void log_buf_vmcoreinfo_setup(void) VMCOREINFO_SIZE(atomic_long_t); VMCOREINFO_TYPE_OFFSET(atomic_long_t, counter); + + VMCOREINFO_STRUCT_SIZE(latched_seq); + VMCOREINFO_OFFSET(latched_seq, val); } #endif @@ -1543,7 +1584,8 @@ static int syslog_print_all(char __user *buf, int size, bool clear) * Find first record that fits, including all following records, * into the user-provided buffer for this dump. */ - seq = find_first_fitting_seq(clear_seq, -1, size, &info, true, time); + seq = find_first_fitting_seq(latched_seq_read_nolock(&clear_seq), -1, + size, &info, true, time); prb_rec_init_rd(&r, &info, text, CONSOLE_LOG_MAX); @@ -1570,7 +1612,7 @@ static int syslog_print_all(char __user *buf, int size, bool clear) } if (clear) - clear_seq = seq; + latched_seq_write(&clear_seq, seq); logbuf_unlock_irq(); kfree(text); @@ -1580,7 +1622,7 @@ static int syslog_print_all(char __user *buf, int size, bool clear) static void syslog_clear(void) { logbuf_lock_irq(); - clear_seq = prb_next_seq(prb); + latched_seq_write(&clear_seq, prb_next_seq(prb)); logbuf_unlock_irq(); } @@ -3311,7 +3353,7 @@ void kmsg_dump(enum kmsg_dump_reason reason) dumper->active = true; logbuf_lock_irqsave(flags); - dumper->cur_seq = clear_seq; + dumper->cur_seq = latched_seq_read_nolock(&clear_seq); dumper->next_seq = prb_next_seq(prb); logbuf_unlock_irqrestore(flags); @@ -3504,7 +3546,7 @@ EXPORT_SYMBOL_GPL(kmsg_dump_get_buffer); */ void kmsg_dump_rewind_nolock(struct kmsg_dumper *dumper) { - dumper->cur_seq = clear_seq; + dumper->cur_seq = latched_seq_read_nolock(&clear_seq); dumper->next_seq = prb_next_seq(prb); } -- 2.20.1