ALSA: hda: Fix potential race in unsol event handler
authorTakashi Iwai <tiwai@suse.de>
Sat, 16 May 2020 06:25:56 +0000 (08:25 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 1 Oct 2020 18:40:13 +0000 (20:40 +0200)
[ Upstream commit c637fa151259c0f74665fde7cba5b7eac1417ae5 ]

The unsol event handling code has a loop retrieving the read/write
indices and the arrays without locking while the append to the array
may happen concurrently.  This may lead to some inconsistency.
Although there hasn't been any proof of this bad results, it's still
safer to protect the racy accesses.

This patch adds the spinlock protection around the unsol handling loop
for addressing it.  Here we take bus->reg_lock as the writer side
snd_hdac_bus_queue_event() is also protected by that lock.

Link: https://lore.kernel.org/r/20200516062556.30951-1-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Sasha Levin <sashal@kernel.org>
sound/hda/hdac_bus.c

index 0e81ea89a5965a49a80e52c6405d9a456cc9decf..e3f68a76d90eb242d1e655b1388ddf3940fead84 100644 (file)
@@ -155,6 +155,7 @@ static void process_unsol_events(struct work_struct *work)
        struct hdac_driver *drv;
        unsigned int rp, caddr, res;
 
+       spin_lock_irq(&bus->reg_lock);
        while (bus->unsol_rp != bus->unsol_wp) {
                rp = (bus->unsol_rp + 1) % HDA_UNSOL_QUEUE_SIZE;
                bus->unsol_rp = rp;
@@ -166,10 +167,13 @@ static void process_unsol_events(struct work_struct *work)
                codec = bus->caddr_tbl[caddr & 0x0f];
                if (!codec || !codec->dev.driver)
                        continue;
+               spin_unlock_irq(&bus->reg_lock);
                drv = drv_to_hdac_driver(codec->dev.driver);
                if (drv->unsol_event)
                        drv->unsol_event(codec, res);
+               spin_lock_irq(&bus->reg_lock);
        }
+       spin_unlock_irq(&bus->reg_lock);
 }
 
 /**