Commit f85d39dd authored by Andrey Konovalov's avatar Andrey Konovalov Committed by Greg Kroah-Hartman

kcov, usb: disable interrupts in kcov_remote_start_usb_softirq

After commit 8fea0c8f ("usb: core: hcd: Convert from tasklet to BH
workqueue"), usb_giveback_urb_bh() runs in the BH workqueue with
interrupts enabled.

Thus, the remote coverage collection section in usb_giveback_urb_bh()->
__usb_hcd_giveback_urb() might be interrupted, and the interrupt handler
might invoke __usb_hcd_giveback_urb() again.

This breaks KCOV, as it does not support nested remote coverage collection
sections within the same context (neither in task nor in softirq).

Update kcov_remote_start/stop_usb_softirq() to disable interrupts for the
duration of the coverage collection section to avoid nested sections in
the softirq context (in addition to such in the task context, which are
already handled).
Reported-by: default avatarTetsuo Handa <penguin-kernel@i-love.sakura.ne.jp>
Closes: https://lore.kernel.org/linux-usb/0f4d1964-7397-485b-bc48-11c01e2fcbca@I-love.SAKURA.ne.jp/
Closes: https://syzkaller.appspot.com/bug?extid=0438378d6f157baae1a2Suggested-by: default avatarAlan Stern <stern@rowland.harvard.edu>
Fixes: 8fea0c8f ("usb: core: hcd: Convert from tasklet to BH workqueue")
Cc: stable@vger.kernel.org
Acked-by: default avatarDmitry Vyukov <dvyukov@google.com>
Signed-off-by: default avatarAndrey Konovalov <andreyknvl@gmail.com>
Link: https://lore.kernel.org/r/20240527173538.4989-1-andrey.konovalov@linux.devSigned-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent e4228cfd
...@@ -1623,6 +1623,7 @@ static void __usb_hcd_giveback_urb(struct urb *urb) ...@@ -1623,6 +1623,7 @@ static void __usb_hcd_giveback_urb(struct urb *urb)
struct usb_hcd *hcd = bus_to_hcd(urb->dev->bus); struct usb_hcd *hcd = bus_to_hcd(urb->dev->bus);
struct usb_anchor *anchor = urb->anchor; struct usb_anchor *anchor = urb->anchor;
int status = urb->unlinked; int status = urb->unlinked;
unsigned long flags;
urb->hcpriv = NULL; urb->hcpriv = NULL;
if (unlikely((urb->transfer_flags & URB_SHORT_NOT_OK) && if (unlikely((urb->transfer_flags & URB_SHORT_NOT_OK) &&
...@@ -1640,13 +1641,14 @@ static void __usb_hcd_giveback_urb(struct urb *urb) ...@@ -1640,13 +1641,14 @@ static void __usb_hcd_giveback_urb(struct urb *urb)
/* pass ownership to the completion handler */ /* pass ownership to the completion handler */
urb->status = status; urb->status = status;
/* /*
* This function can be called in task context inside another remote * Only collect coverage in the softirq context and disable interrupts
* coverage collection section, but kcov doesn't support that kind of * to avoid scenarios with nested remote coverage collection sections
* recursion yet. Only collect coverage in softirq context for now. * that KCOV does not support.
* See the comment next to kcov_remote_start_usb_softirq() for details.
*/ */
kcov_remote_start_usb_softirq((u64)urb->dev->bus->busnum); flags = kcov_remote_start_usb_softirq((u64)urb->dev->bus->busnum);
urb->complete(urb); urb->complete(urb);
kcov_remote_stop_softirq(); kcov_remote_stop_softirq(flags);
usb_anchor_resume_wakeups(anchor); usb_anchor_resume_wakeups(anchor);
atomic_dec(&urb->use_count); atomic_dec(&urb->use_count);
......
...@@ -55,21 +55,47 @@ static inline void kcov_remote_start_usb(u64 id) ...@@ -55,21 +55,47 @@ static inline void kcov_remote_start_usb(u64 id)
/* /*
* The softirq flavor of kcov_remote_*() functions is introduced as a temporary * The softirq flavor of kcov_remote_*() functions is introduced as a temporary
* work around for kcov's lack of nested remote coverage sections support in * workaround for KCOV's lack of nested remote coverage sections support.
* task context. Adding support for nested sections is tracked in: *
* https://bugzilla.kernel.org/show_bug.cgi?id=210337 * Adding support is tracked in https://bugzilla.kernel.org/show_bug.cgi?id=210337.
*
* kcov_remote_start_usb_softirq():
*
* 1. Only collects coverage when called in the softirq context. This allows
* avoiding nested remote coverage collection sections in the task context.
* For example, USB/IP calls usb_hcd_giveback_urb() in the task context
* within an existing remote coverage collection section. Thus, KCOV should
* not attempt to start collecting coverage within the coverage collection
* section in __usb_hcd_giveback_urb() in this case.
*
* 2. Disables interrupts for the duration of the coverage collection section.
* This allows avoiding nested remote coverage collection sections in the
* softirq context (a softirq might occur during the execution of a work in
* the BH workqueue, which runs with in_serving_softirq() > 0).
* For example, usb_giveback_urb_bh() runs in the BH workqueue with
* interrupts enabled, so __usb_hcd_giveback_urb() might be interrupted in
* the middle of its remote coverage collection section, and the interrupt
* handler might invoke __usb_hcd_giveback_urb() again.
*/ */
static inline void kcov_remote_start_usb_softirq(u64 id) static inline unsigned long kcov_remote_start_usb_softirq(u64 id)
{ {
if (in_serving_softirq()) unsigned long flags = 0;
if (in_serving_softirq()) {
local_irq_save(flags);
kcov_remote_start_usb(id); kcov_remote_start_usb(id);
}
return flags;
} }
static inline void kcov_remote_stop_softirq(void) static inline void kcov_remote_stop_softirq(unsigned long flags)
{ {
if (in_serving_softirq()) if (in_serving_softirq()) {
kcov_remote_stop(); kcov_remote_stop();
local_irq_restore(flags);
}
} }
#ifdef CONFIG_64BIT #ifdef CONFIG_64BIT
...@@ -103,8 +129,11 @@ static inline u64 kcov_common_handle(void) ...@@ -103,8 +129,11 @@ static inline u64 kcov_common_handle(void)
} }
static inline void kcov_remote_start_common(u64 id) {} static inline void kcov_remote_start_common(u64 id) {}
static inline void kcov_remote_start_usb(u64 id) {} static inline void kcov_remote_start_usb(u64 id) {}
static inline void kcov_remote_start_usb_softirq(u64 id) {} static inline unsigned long kcov_remote_start_usb_softirq(u64 id)
static inline void kcov_remote_stop_softirq(void) {} {
return 0;
}
static inline void kcov_remote_stop_softirq(unsigned long flags) {}
#endif /* CONFIG_KCOV */ #endif /* CONFIG_KCOV */
#endif /* _LINUX_KCOV_H */ #endif /* _LINUX_KCOV_H */
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