linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
* [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc
@ 2021-09-23 16:47 Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 1/5] Documentation/kcov: Include types.h in the example Sebastian Andrzej Siewior
                   ` (5 more replies)
  0 siblings, 6 replies; 7+ messages in thread
From: Sebastian Andrzej Siewior @ 2021-09-23 16:47 UTC (permalink / raw)
  To: kasan-dev, linux-kernel
  Cc: Dmitry Vyukov, Andrey Konovalov, Thomas Gleixner, Steven Rostedt,
	Marco Elver, Clark Williams, Andrew Morton

This is a repost of my initial series [0] with collected tags. 

The last patch in series is follow-up to address the PREEMPT_RT issue
within in kcov reported by Clark [1].
Patches 1-3 are smaller things that I noticed while staring at it.
Patch 4 is small change which makes replacement in #5 simpler / more
obvious.
I tested this with the snippets in examples, Marco Elver used to run syzkaller
for a few hours.

[0] https://lkml.kernel.org/r/20210830172627.267989-1-bigeasy@linutronix.de
[1] https://lkml.kernel.org/r/20210809155909.333073de@theseus.lan



^ permalink raw reply	[flat|nested] 7+ messages in thread

* [PATCH v2 1/5] Documentation/kcov: Include types.h in the example.
  2021-09-23 16:47 [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Sebastian Andrzej Siewior
@ 2021-09-23 16:47 ` Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 2/5] Documentation/kcov: Define `ip' " Sebastian Andrzej Siewior
                   ` (4 subsequent siblings)
  5 siblings, 0 replies; 7+ messages in thread
From: Sebastian Andrzej Siewior @ 2021-09-23 16:47 UTC (permalink / raw)
  To: kasan-dev, linux-kernel
  Cc: Dmitry Vyukov, Andrey Konovalov, Thomas Gleixner, Steven Rostedt,
	Marco Elver, Clark Williams, Andrew Morton,
	Sebastian Andrzej Siewior

The first example code has includes at the top, the following two
example share that part. The last example (remote coverage collection)
requires the linux/types.h header file due its __aligned_u64 usage.

Add the linux/types.h to the top most example and a comment that the
header files from above are required as it is done in the second
example.

Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Acked-by: Dmitry Vyukov <dvyukov@google.com>
Acked-by: Marco Elver <elver@google.com>
Tested-by: Marco Elver <elver@google.com>
Link: https://lore.kernel.org/r/20210830172627.267989-2-bigeasy@linutronix.de
---
 Documentation/dev-tools/kcov.rst | 3 +++
 1 file changed, 3 insertions(+)

diff --git a/Documentation/dev-tools/kcov.rst b/Documentation/dev-tools/kcov.rst
index d2c4c27e1702d..347f3b6de8d40 100644
--- a/Documentation/dev-tools/kcov.rst
+++ b/Documentation/dev-tools/kcov.rst
@@ -50,6 +50,7 @@ The following program demonstrates coverage collection from within a test
     #include <sys/mman.h>
     #include <unistd.h>
     #include <fcntl.h>
+    #include <linux/types.h>
 
     #define KCOV_INIT_TRACE			_IOR('c', 1, unsigned long)
     #define KCOV_ENABLE			_IO('c', 100)
@@ -251,6 +252,8 @@ selectively from different subsystems.
 
 .. code-block:: c
 
+    /* Same includes and defines as above. */
+
     struct kcov_remote_arg {
 	__u32		trace_mode;
 	__u32		area_size;
-- 
2.33.0


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* [PATCH v2 2/5] Documentation/kcov: Define `ip' in the example.
  2021-09-23 16:47 [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 1/5] Documentation/kcov: Include types.h in the example Sebastian Andrzej Siewior
@ 2021-09-23 16:47 ` Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 3/5] kcov: Allocate per-CPU memory on the relevant node Sebastian Andrzej Siewior
                   ` (3 subsequent siblings)
  5 siblings, 0 replies; 7+ messages in thread
From: Sebastian Andrzej Siewior @ 2021-09-23 16:47 UTC (permalink / raw)
  To: kasan-dev, linux-kernel
  Cc: Dmitry Vyukov, Andrey Konovalov, Thomas Gleixner, Steven Rostedt,
	Marco Elver, Clark Williams, Andrew Morton,
	Sebastian Andrzej Siewior

The example code uses the variable `ip' but never declares it.

Declare `ip' as a 64bit variable which is the same type as the array
from which it loads its value.

Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Acked-by: Dmitry Vyukov <dvyukov@google.com>
Acked-by: Marco Elver <elver@google.com>
Tested-by: Marco Elver <elver@google.com>
Link: https://lore.kernel.org/r/20210830172627.267989-3-bigeasy@linutronix.de
---
 Documentation/dev-tools/kcov.rst | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/Documentation/dev-tools/kcov.rst b/Documentation/dev-tools/kcov.rst
index 347f3b6de8d40..d83c9ab494275 100644
--- a/Documentation/dev-tools/kcov.rst
+++ b/Documentation/dev-tools/kcov.rst
@@ -178,6 +178,8 @@ Comparison operands collection
 	/* Read number of comparisons collected. */
 	n = __atomic_load_n(&cover[0], __ATOMIC_RELAXED);
 	for (i = 0; i < n; i++) {
+		uint64_t ip;
+
 		type = cover[i * KCOV_WORDS_PER_CMP + 1];
 		/* arg1 and arg2 - operands of the comparison. */
 		arg1 = cover[i * KCOV_WORDS_PER_CMP + 2];
-- 
2.33.0


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* [PATCH v2 3/5] kcov: Allocate per-CPU memory on the relevant node.
  2021-09-23 16:47 [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 1/5] Documentation/kcov: Include types.h in the example Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 2/5] Documentation/kcov: Define `ip' " Sebastian Andrzej Siewior
@ 2021-09-23 16:47 ` Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 4/5] kcov: Avoid enable+disable interrupts if !in_task() Sebastian Andrzej Siewior
                   ` (2 subsequent siblings)
  5 siblings, 0 replies; 7+ messages in thread
From: Sebastian Andrzej Siewior @ 2021-09-23 16:47 UTC (permalink / raw)
  To: kasan-dev, linux-kernel
  Cc: Dmitry Vyukov, Andrey Konovalov, Thomas Gleixner, Steven Rostedt,
	Marco Elver, Clark Williams, Andrew Morton,
	Sebastian Andrzej Siewior

During boot kcov allocates per-CPU memory which is used later if remote/
softirq processing is enabled.

Allocate the per-CPU memory on the CPU local node to avoid cross node
memory access.

Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Acked-by: Dmitry Vyukov <dvyukov@google.com>
Acked-by: Marco Elver <elver@google.com>
Tested-by: Marco Elver <elver@google.com>
Link: https://lore.kernel.org/r/20210830172627.267989-4-bigeasy@linutronix.de
---
 kernel/kcov.c | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/kernel/kcov.c b/kernel/kcov.c
index 80bfe71bbe13e..4f910231d99a2 100644
--- a/kernel/kcov.c
+++ b/kernel/kcov.c
@@ -1034,8 +1034,8 @@ static int __init kcov_init(void)
 	int cpu;
 
 	for_each_possible_cpu(cpu) {
-		void *area = vmalloc(CONFIG_KCOV_IRQ_AREA_SIZE *
-				sizeof(unsigned long));
+		void *area = vmalloc_node(CONFIG_KCOV_IRQ_AREA_SIZE *
+				sizeof(unsigned long), cpu_to_node(cpu));
 		if (!area)
 			return -ENOMEM;
 		per_cpu_ptr(&kcov_percpu_data, cpu)->irq_area = area;
-- 
2.33.0


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* [PATCH v2 4/5] kcov: Avoid enable+disable interrupts if !in_task().
  2021-09-23 16:47 [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Sebastian Andrzej Siewior
                   ` (2 preceding siblings ...)
  2021-09-23 16:47 ` [PATCH v2 3/5] kcov: Allocate per-CPU memory on the relevant node Sebastian Andrzej Siewior
@ 2021-09-23 16:47 ` Sebastian Andrzej Siewior
  2021-09-23 16:47 ` [PATCH v2 5/5] kcov: Replace local_irq_save() with a local_lock_t Sebastian Andrzej Siewior
  2021-10-03 18:11 ` [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Andrey Konovalov
  5 siblings, 0 replies; 7+ messages in thread
From: Sebastian Andrzej Siewior @ 2021-09-23 16:47 UTC (permalink / raw)
  To: kasan-dev, linux-kernel
  Cc: Dmitry Vyukov, Andrey Konovalov, Thomas Gleixner, Steven Rostedt,
	Marco Elver, Clark Williams, Andrew Morton,
	Sebastian Andrzej Siewior

kcov_remote_start() may need to allocate memory in the in_task() case
(otherwise per-CPU memory has been pre-allocated) and therefore requires
enabled interrupts.
The interrupts are enabled before checking if the allocation is required
so if no allocation is required then the interrupts are needlessly
enabled and disabled again.

Enable interrupts only if memory allocation is performed.

Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Acked-by: Dmitry Vyukov <dvyukov@google.com>
Acked-by: Marco Elver <elver@google.com>
Tested-by: Marco Elver <elver@google.com>
Link: https://lore.kernel.org/r/20210830172627.267989-5-bigeasy@linutronix.de
---
 kernel/kcov.c | 6 +++---
 1 file changed, 3 insertions(+), 3 deletions(-)

diff --git a/kernel/kcov.c b/kernel/kcov.c
index 4f910231d99a2..620dc4ffeb685 100644
--- a/kernel/kcov.c
+++ b/kernel/kcov.c
@@ -869,19 +869,19 @@ void kcov_remote_start(u64 handle)
 		size = CONFIG_KCOV_IRQ_AREA_SIZE;
 		area = this_cpu_ptr(&kcov_percpu_data)->irq_area;
 	}
-	spin_unlock_irqrestore(&kcov_remote_lock, flags);
+	spin_unlock(&kcov_remote_lock);
 
 	/* Can only happen when in_task(). */
 	if (!area) {
+		local_irqrestore(flags);
 		area = vmalloc(size * sizeof(unsigned long));
 		if (!area) {
 			kcov_put(kcov);
 			return;
 		}
+		local_irq_save(flags);
 	}
 
-	local_irq_save(flags);
-
 	/* Reset coverage size. */
 	*(u64 *)area = 0;
 
-- 
2.33.0


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* [PATCH v2 5/5] kcov: Replace local_irq_save() with a local_lock_t.
  2021-09-23 16:47 [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Sebastian Andrzej Siewior
                   ` (3 preceding siblings ...)
  2021-09-23 16:47 ` [PATCH v2 4/5] kcov: Avoid enable+disable interrupts if !in_task() Sebastian Andrzej Siewior
@ 2021-09-23 16:47 ` Sebastian Andrzej Siewior
  2021-10-03 18:11 ` [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Andrey Konovalov
  5 siblings, 0 replies; 7+ messages in thread
From: Sebastian Andrzej Siewior @ 2021-09-23 16:47 UTC (permalink / raw)
  To: kasan-dev, linux-kernel
  Cc: Dmitry Vyukov, Andrey Konovalov, Thomas Gleixner, Steven Rostedt,
	Marco Elver, Clark Williams, Andrew Morton,
	Sebastian Andrzej Siewior

The kcov code mixes local_irq_save() and spin_lock() in
kcov_remote_{start|end}(). This creates a warning on PREEMPT_RT because
local_irq_save() disables interrupts and spin_lock_t is turned into a
sleeping lock which can not be acquired in a section with disabled
interrupts.

The kcov_remote_lock is used to synchronize the access to the hash-list
kcov_remote_map. The local_irq_save() block protects access to the
per-CPU data kcov_percpu_data.

There no compelling reason to change the lock type to raw_spin_lock_t to
make it work with local_irq_save(). Changing it would require to move
memory allocation (in kcov_remote_add()) and deallocation outside of the
locked section.
Adding an unlimited amount of entries to the hashlist will increase the
IRQ-off time during lookup. It could be argued that this is debug code
and the latency does not matter. There is however no need to do so and
it would allow to use this facility in an RT enabled build.

Using a local_lock_t instead of local_irq_save() has the befit of adding
a protection scope within the source which makes it obvious what is
protected. On a !PREEMPT_RT && !LOCKDEP build the local_lock_irqsave()
maps directly to local_irq_save() so there is overhead at runtime.

Replace the local_irq_save() section with a local_lock_t.

Reported-by: Clark Williams <williams@redhat.com>
Signed-off-by: Sebastian Andrzej Siewior <bigeasy@linutronix.de>
Acked-by: Dmitry Vyukov <dvyukov@google.com>
Acked-by: Marco Elver <elver@google.com>
Tested-by: Marco Elver <elver@google.com>
Link: https://lore.kernel.org/r/20210830172627.267989-6-bigeasy@linutronix.de
---
 kernel/kcov.c | 30 +++++++++++++++++-------------
 1 file changed, 17 insertions(+), 13 deletions(-)

diff --git a/kernel/kcov.c b/kernel/kcov.c
index 620dc4ffeb685..36ca640c4f8e7 100644
--- a/kernel/kcov.c
+++ b/kernel/kcov.c
@@ -88,6 +88,7 @@ static struct list_head kcov_remote_areas = LIST_HEAD_INIT(kcov_remote_areas);
 
 struct kcov_percpu_data {
 	void			*irq_area;
+	local_lock_t		lock;
 
 	unsigned int		saved_mode;
 	unsigned int		saved_size;
@@ -96,7 +97,9 @@ struct kcov_percpu_data {
 	int			saved_sequence;
 };
 
-static DEFINE_PER_CPU(struct kcov_percpu_data, kcov_percpu_data);
+static DEFINE_PER_CPU(struct kcov_percpu_data, kcov_percpu_data) = {
+	.lock = INIT_LOCAL_LOCK(lock),
+};
 
 /* Must be called with kcov_remote_lock locked. */
 static struct kcov_remote *kcov_remote_find(u64 handle)
@@ -824,7 +827,7 @@ void kcov_remote_start(u64 handle)
 	if (!in_task() && !in_serving_softirq())
 		return;
 
-	local_irq_save(flags);
+	local_lock_irqsave(&kcov_percpu_data.lock, flags);
 
 	/*
 	 * Check that kcov_remote_start() is not called twice in background
@@ -832,7 +835,7 @@ void kcov_remote_start(u64 handle)
 	 */
 	mode = READ_ONCE(t->kcov_mode);
 	if (WARN_ON(in_task() && kcov_mode_enabled(mode))) {
-		local_irq_restore(flags);
+		local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 		return;
 	}
 	/*
@@ -841,14 +844,15 @@ void kcov_remote_start(u64 handle)
 	 * happened while collecting coverage from a background thread.
 	 */
 	if (WARN_ON(in_serving_softirq() && t->kcov_softirq)) {
-		local_irq_restore(flags);
+		local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 		return;
 	}
 
 	spin_lock(&kcov_remote_lock);
 	remote = kcov_remote_find(handle);
 	if (!remote) {
-		spin_unlock_irqrestore(&kcov_remote_lock, flags);
+		spin_unlock(&kcov_remote_lock);
+		local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 		return;
 	}
 	kcov_debug("handle = %llx, context: %s\n", handle,
@@ -873,13 +877,13 @@ void kcov_remote_start(u64 handle)
 
 	/* Can only happen when in_task(). */
 	if (!area) {
-		local_irqrestore(flags);
+		local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 		area = vmalloc(size * sizeof(unsigned long));
 		if (!area) {
 			kcov_put(kcov);
 			return;
 		}
-		local_irq_save(flags);
+		local_lock_irqsave(&kcov_percpu_data.lock, flags);
 	}
 
 	/* Reset coverage size. */
@@ -891,7 +895,7 @@ void kcov_remote_start(u64 handle)
 	}
 	kcov_start(t, kcov, size, area, mode, sequence);
 
-	local_irq_restore(flags);
+	local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 
 }
 EXPORT_SYMBOL(kcov_remote_start);
@@ -965,12 +969,12 @@ void kcov_remote_stop(void)
 	if (!in_task() && !in_serving_softirq())
 		return;
 
-	local_irq_save(flags);
+	local_lock_irqsave(&kcov_percpu_data.lock, flags);
 
 	mode = READ_ONCE(t->kcov_mode);
 	barrier();
 	if (!kcov_mode_enabled(mode)) {
-		local_irq_restore(flags);
+		local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 		return;
 	}
 	/*
@@ -978,12 +982,12 @@ void kcov_remote_stop(void)
 	 * actually found the remote handle and started collecting coverage.
 	 */
 	if (in_serving_softirq() && !t->kcov_softirq) {
-		local_irq_restore(flags);
+		local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 		return;
 	}
 	/* Make sure that kcov_softirq is only set when in softirq. */
 	if (WARN_ON(!in_serving_softirq() && t->kcov_softirq)) {
-		local_irq_restore(flags);
+		local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 		return;
 	}
 
@@ -1013,7 +1017,7 @@ void kcov_remote_stop(void)
 		spin_unlock(&kcov_remote_lock);
 	}
 
-	local_irq_restore(flags);
+	local_unlock_irqrestore(&kcov_percpu_data.lock, flags);
 
 	/* Get in kcov_remote_start(). */
 	kcov_put(kcov);
-- 
2.33.0


^ permalink raw reply related	[flat|nested] 7+ messages in thread

* Re: [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc
  2021-09-23 16:47 [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Sebastian Andrzej Siewior
                   ` (4 preceding siblings ...)
  2021-09-23 16:47 ` [PATCH v2 5/5] kcov: Replace local_irq_save() with a local_lock_t Sebastian Andrzej Siewior
@ 2021-10-03 18:11 ` Andrey Konovalov
  5 siblings, 0 replies; 7+ messages in thread
From: Andrey Konovalov @ 2021-10-03 18:11 UTC (permalink / raw)
  To: Sebastian Andrzej Siewior
  Cc: kasan-dev, LKML, Dmitry Vyukov, Thomas Gleixner, Steven Rostedt,
	Marco Elver, Clark Williams, Andrew Morton

On Thu, Sep 23, 2021 at 6:47 PM Sebastian Andrzej Siewior
<bigeasy@linutronix.de> wrote:
>
> This is a repost of my initial series [0] with collected tags.
>
> The last patch in series is follow-up to address the PREEMPT_RT issue
> within in kcov reported by Clark [1].
> Patches 1-3 are smaller things that I noticed while staring at it.
> Patch 4 is small change which makes replacement in #5 simpler / more
> obvious.
> I tested this with the snippets in examples, Marco Elver used to run syzkaller
> for a few hours.
>
> [0] https://lkml.kernel.org/r/20210830172627.267989-1-bigeasy@linutronix.de
> [1] https://lkml.kernel.org/r/20210809155909.333073de@theseus.lan
>

Reviewed-by: Andrey Konovalov <andreyknvl@gmail.com>

for the series.

^ permalink raw reply	[flat|nested] 7+ messages in thread

end of thread, other threads:[~2021-10-03 18:11 UTC | newest]

Thread overview: 7+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2021-09-23 16:47 [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Sebastian Andrzej Siewior
2021-09-23 16:47 ` [PATCH v2 1/5] Documentation/kcov: Include types.h in the example Sebastian Andrzej Siewior
2021-09-23 16:47 ` [PATCH v2 2/5] Documentation/kcov: Define `ip' " Sebastian Andrzej Siewior
2021-09-23 16:47 ` [PATCH v2 3/5] kcov: Allocate per-CPU memory on the relevant node Sebastian Andrzej Siewior
2021-09-23 16:47 ` [PATCH v2 4/5] kcov: Avoid enable+disable interrupts if !in_task() Sebastian Andrzej Siewior
2021-09-23 16:47 ` [PATCH v2 5/5] kcov: Replace local_irq_save() with a local_lock_t Sebastian Andrzej Siewior
2021-10-03 18:11 ` [PATCH v2 0/5] kcov: PREEMPT_RT fixup + misc Andrey Konovalov

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).