Skip to content

Commit e5f20a3

Browse files
tiwaigregkh
authored andcommitted
ALSA: seq: Fix race at creating a queue
commit 4842e98f26dd80be3623c4714a244ba52ea096a8 upstream. When a sequencer queue is created in snd_seq_queue_alloc(),it adds the new queue element to the public list before referencing it. Thus the queue might be deleted before the call of snd_seq_queue_use(), and it results in the use-after-free error, as spotted by syzkaller. The fix is to reference the queue object at the right time. Reported-by: Dmitry Vyukov <dvyukov@google.com> Signed-off-by: Takashi Iwai <tiwai@suse.de> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
1 parent 010ada0 commit e5f20a3

1 file changed

Lines changed: 20 additions & 13 deletions

File tree

sound/core/seq/seq_queue.c

Lines changed: 20 additions & 13 deletions
Original file line numberDiff line numberDiff line change
@@ -181,6 +181,8 @@ void __exit snd_seq_queues_delete(void)
181181
}
182182
}
183183

184+
static void queue_use(struct snd_seq_queue *queue, int client, int use);
185+
184186
/* allocate a new queue -
185187
* return queue index value or negative value for error
186188
*/
@@ -192,11 +194,11 @@ int snd_seq_queue_alloc(int client, int locked, unsigned int info_flags)
192194
if (q == NULL)
193195
return -ENOMEM;
194196
q->info_flags = info_flags;
197+
queue_use(q, client, 1);
195198
if (queue_list_add(q) < 0) {
196199
queue_delete(q);
197200
return -ENOMEM;
198201
}
199-
snd_seq_queue_use(q->queue, client, 1); /* use this queue */
200202
return q->queue;
201203
}
202204

@@ -502,19 +504,9 @@ int snd_seq_queue_timer_set_tempo(int queueid, int client,
502504
return result;
503505
}
504506

505-
506-
/* use or unuse this queue -
507-
* if it is the first client, starts the timer.
508-
* if it is not longer used by any clients, stop the timer.
509-
*/
510-
int snd_seq_queue_use(int queueid, int client, int use)
507+
/* use or unuse this queue */
508+
static void queue_use(struct snd_seq_queue *queue, int client, int use)
511509
{
512-
struct snd_seq_queue *queue;
513-
514-
queue = queueptr(queueid);
515-
if (queue == NULL)
516-
return -EINVAL;
517-
mutex_lock(&queue->timer_mutex);
518510
if (use) {
519511
if (!test_and_set_bit(client, queue->clients_bitmap))
520512
queue->clients++;
@@ -529,6 +521,21 @@ int snd_seq_queue_use(int queueid, int client, int use)
529521
} else {
530522
snd_seq_timer_close(queue);
531523
}
524+
}
525+
526+
/* use or unuse this queue -
527+
* if it is the first client, starts the timer.
528+
* if it is not longer used by any clients, stop the timer.
529+
*/
530+
int snd_seq_queue_use(int queueid, int client, int use)
531+
{
532+
struct snd_seq_queue *queue;
533+
534+
queue = queueptr(queueid);
535+
if (queue == NULL)
536+
return -EINVAL;
537+
mutex_lock(&queue->timer_mutex);
538+
queue_use(queue, client, use);
532539
mutex_unlock(&queue->timer_mutex);
533540
queuefree(queue);
534541
return 0;

0 commit comments

Comments
 (0)