]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
ALSA: seq: Fix uninitialised heap leak in snd_seq_event_dup()
authorHyeongJun An <sammiee5311@gmail.com>
Tue, 23 Jun 2026 23:38:40 +0000 (08:38 +0900)
committerTakashi Iwai <tiwai@suse.de>
Wed, 24 Jun 2026 09:03:25 +0000 (11:03 +0200)
snd_seq_event_dup() copies an incoming event into a pool cell and, in
the UMP-enabled build, clears the trailing cell->ump.raw.extra word that
the memcpy() did not cover.  The guard deciding whether to clear it
compares the copied size against sizeof(cell->event):

memcpy(&cell->ump, event, size);
if (size < sizeof(cell->event))
cell->ump.raw.extra = 0;

For a legacy (non-UMP) event, size == sizeof(struct snd_seq_event) ==
sizeof(cell->event), so the condition is false and the extra word keeps
stale data.  The cell pool is allocated with kvmalloc() (not zeroed) and
cells are reused via a free list, so that word holds uninitialised heap
or leftover event data.

When such a cell is delivered to a UMP client (client->midi_version > 0)
that set SNDRV_SEQ_FILTER_NO_CONVERT -- so the legacy event reaches it
unconverted -- snd_seq_read() reads it out as the larger struct
snd_seq_ump_event and copies the stale word to user space, a 4-byte
kernel heap infoleak to an unprivileged /dev/snd/seq client.

Compare against sizeof(cell->ump) instead, so the trailing word is zeroed
for every event shorter than the UMP cell.

Fixes: 46397622a3fa ("ALSA: seq: Add UMP support")
Cc: stable@vger.kernel.org
Assisted-by: Claude:claude-opus-4-8
Signed-off-by: HyeongJun An <sammiee5311@gmail.com>
Link: https://patch.msgid.link/20260623233841.853326-1-sammiee5311@gmail.com
Signed-off-by: Takashi Iwai <tiwai@suse.de>
sound/core/seq/seq_memory.c

index ca9f6db0022c82a3175dfa7ca0eb76415d4cbc54..209b08c2a94088e6c61710a9a714f5fbe3e330ac 100644 (file)
@@ -364,7 +364,7 @@ int snd_seq_event_dup(struct snd_seq_pool *pool, struct snd_seq_event *event,
        size = snd_seq_event_packet_size(event);
        memcpy(&cell->ump, event, size);
 #if IS_ENABLED(CONFIG_SND_SEQ_UMP)
-       if (size < sizeof(cell->event))
+       if (size < sizeof(cell->ump))
                cell->ump.raw.extra = 0;
 #endif