Re: KASAN: use-after-free Read in si470x_int_in_callback (2)

From: Alan Stern
Date: Wed Nov 27 2019 - 13:07:16 EST


On Wed, 27 Nov 2019, syzbot wrote:

> Hello,
>
> syzbot has tested the proposed patch but the reproducer still triggered
> crash:
> INFO: rcu detected stall in dummy_timer
>
> radio-si470x 5-1:0.0: non-zero urb status (-71)
> radio-si470x 3-1:0.0: non-zero urb status (-71)
> rcu: INFO: rcu_sched self-detected stall on CPU
> rcu: 1-....: (8213 ticks this GP) idle=4f6/1/0x4000000000000004

Almost the same as Oliver's patch, but this one stops when the
interrupt-IN URB gets an unrecognized error status.

Alan Stern

#syz test: https://github.com/google/kasan.git 22be26f7

Index: usb-devel/drivers/media/radio/si470x/radio-si470x-usb.c
===================================================================
--- usb-devel.orig/drivers/media/radio/si470x/radio-si470x-usb.c
+++ usb-devel/drivers/media/radio/si470x/radio-si470x-usb.c
@@ -370,15 +370,14 @@ static void si470x_int_in_callback(struc
unsigned char tmpbuf[3];

if (urb->status) {
- if (urb->status == -ENOENT ||
+ if (!(urb->status == -ENOENT ||
urb->status == -ECONNRESET ||
- urb->status == -ESHUTDOWN) {
- return;
- } else {
+ urb->status == -ESHUTDOWN))
dev_warn(&radio->intf->dev,
- "non-zero urb status (%d)\n", urb->status);
- goto resubmit; /* Maybe we can recover. */
- }
+ "unrecognized urb status (%d)\n",
+ urb->status);
+ radio->int_in_running = 0;
+ return;
}

/* Sometimes the device returns len 0 packets */
@@ -542,6 +541,8 @@ static int si470x_start_usb(struct si470
radio->int_in_running = 0;
}
radio->status_rssi_auto_update = radio->int_in_running;
+ if (retval < 0)
+ return retval;

/* start radio */
retval = si470x_start(radio);
@@ -734,7 +735,8 @@ static int si470x_usb_driver_probe(struc
/* start radio */
retval = si470x_start_usb(radio);
if (retval < 0)
- goto err_buf;
+ /* the urb may be running even after an error */
+ goto err_all;

/* set initial frequency */
si470x_set_freq(radio, 87.5 * FREQ_MUL); /* available in all regions */
@@ -749,7 +751,7 @@ static int si470x_usb_driver_probe(struc

return 0;
err_all:
- usb_kill_urb(radio->int_in_urb);
+ usb_poison_urb(radio->int_in_urb);
err_buf:
kfree(radio->buffer);
err_ctrl:
@@ -824,7 +826,7 @@ static void si470x_usb_driver_disconnect
mutex_lock(&radio->lock);
v4l2_device_disconnect(&radio->v4l2_dev);
video_unregister_device(&radio->videodev);
- usb_kill_urb(radio->int_in_urb);
+ usb_poison_urb(radio->int_in_urb);
usb_set_intfdata(intf, NULL);
mutex_unlock(&radio->lock);
v4l2_device_put(&radio->v4l2_dev);