]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
ALSA: seq: Improve data consistency at polling
authorTakashi Iwai <tiwai@suse.de>
Fri, 7 Mar 2025 08:42:42 +0000 (09:42 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 4 Jun 2025 12:42:01 +0000 (14:42 +0200)
[ Upstream commit e3cd33ab17c33bd8f1a9df66ec83a15dd8f7afbb ]

snd_seq_poll() calls snd_seq_write_pool_allocated() that reads out a
field in client->pool object, while it can be updated concurrently via
ioctls, as reported by syzbot.  The data race itself is harmless, as
it's merely a poll() call, and the state is volatile.  OTOH, the read
out of poll object info from the caller side is fragile, and we can
leave it better in snd_seq_pool_poll_wait() alone.

A similar pattern is seen in snd_seq_kernel_client_write_poll(), too,
which is called from the OSS sequencer.

This patch drops the pool checks from the caller side and add the
pool->lock in snd_seq_pool_poll_wait() for better data consistency.

Reported-by: syzbot+2d373c9936c00d7e120c@syzkaller.appspotmail.com
Closes: https://lore.kernel.org/67c88903.050a0220.15b4b9.0028.GAE@google.com
Link: https://patch.msgid.link/20250307084246.29271-1-tiwai@suse.de
Signed-off-by: Takashi Iwai <tiwai@suse.de>
Signed-off-by: Sasha Levin <sashal@kernel.org>
sound/core/seq/seq_clientmgr.c
sound/core/seq/seq_memory.c

index 49f6763c3250ddbe6e87386eca4424f5af2d5019..31428cdc0f63d7675e7f55f0606edc85fec24581 100644 (file)
@@ -1169,8 +1169,7 @@ static __poll_t snd_seq_poll(struct file *file, poll_table * wait)
        if (snd_seq_file_flags(file) & SNDRV_SEQ_LFLG_OUTPUT) {
 
                /* check if data is available in the pool */
-               if (!snd_seq_write_pool_allocated(client) ||
-                   snd_seq_pool_poll_wait(client->pool, file, wait))
+               if (snd_seq_pool_poll_wait(client->pool, file, wait))
                        mask |= EPOLLOUT | EPOLLWRNORM;
        }
 
@@ -2584,8 +2583,6 @@ int snd_seq_kernel_client_write_poll(int clientid, struct file *file, poll_table
        if (client == NULL)
                return -ENXIO;
 
-       if (! snd_seq_write_pool_allocated(client))
-               return 1;
        if (snd_seq_pool_poll_wait(client->pool, file, wait))
                return 1;
        return 0;
index b603bb93f896030130504e84eb29ff311ad393d1..692860deec0c3d0498e6428c06f5cee5e7238a47 100644 (file)
@@ -429,6 +429,7 @@ int snd_seq_pool_poll_wait(struct snd_seq_pool *pool, struct file *file,
                           poll_table *wait)
 {
        poll_wait(file, &pool->output_sleep, wait);
+       guard(spinlock_irq)(&pool->lock);
        return snd_seq_output_ok(pool);
 }