Skip to content

Commit c1cfd90

Browse files
committed
ALSA: rawmidi: Avoid racy info ioctl via ctl device
The rawmidi also allows to obtaining the information via ioctl of ctl API. It means that user can issue an ioctl to the rawmidi device even when it's being removed as long as the control device is present. Although the code has some protection via the global register_mutex, its range is limited to the search of the corresponding rawmidi object, and the mutex is already unlocked at accessing the rawmidi object. This may lead to a use-after-free. For avoiding it, this patch widens the application of register_mutex to the whole snd_rawmidi_info_select() function. We have another mutex per rawmidi object, but this operation isn't very hot path, so it shouldn't matter from the performance POV. Cc: <[email protected]> Signed-off-by: Takashi Iwai <[email protected]>
1 parent 2b4584d commit c1cfd90

File tree

1 file changed

+12
-3
lines changed

1 file changed

+12
-3
lines changed

sound/core/rawmidi.c

+12-3
Original file line numberDiff line numberDiff line change
@@ -579,15 +579,14 @@ static int snd_rawmidi_info_user(struct snd_rawmidi_substream *substream,
579579
return 0;
580580
}
581581

582-
int snd_rawmidi_info_select(struct snd_card *card, struct snd_rawmidi_info *info)
582+
static int __snd_rawmidi_info_select(struct snd_card *card,
583+
struct snd_rawmidi_info *info)
583584
{
584585
struct snd_rawmidi *rmidi;
585586
struct snd_rawmidi_str *pstr;
586587
struct snd_rawmidi_substream *substream;
587588

588-
mutex_lock(&register_mutex);
589589
rmidi = snd_rawmidi_search(card, info->device);
590-
mutex_unlock(&register_mutex);
591590
if (!rmidi)
592591
return -ENXIO;
593592
if (info->stream < 0 || info->stream > 1)
@@ -603,6 +602,16 @@ int snd_rawmidi_info_select(struct snd_card *card, struct snd_rawmidi_info *info
603602
}
604603
return -ENXIO;
605604
}
605+
606+
int snd_rawmidi_info_select(struct snd_card *card, struct snd_rawmidi_info *info)
607+
{
608+
int ret;
609+
610+
mutex_lock(&register_mutex);
611+
ret = __snd_rawmidi_info_select(card, info);
612+
mutex_unlock(&register_mutex);
613+
return ret;
614+
}
606615
EXPORT_SYMBOL(snd_rawmidi_info_select);
607616

608617
static int snd_rawmidi_info_select_user(struct snd_card *card,

0 commit comments

Comments
 (0)