Commit dfef01e1 authored by Takashi Iwai's avatar Takashi Iwai

ALSA: memalloc: Don't exceed over the requested size

snd_dma_alloc_pages_fallback() tries to allocate pages again when the
allocation fails with reduced size.  But the first try actually
*increases* the size to power-of-two, which may give back a larger
chunk than the requested size.  This confuses the callers, e.g. sgbuf
assumes that the size is equal or less, and it may result in a bad
loop due to the underflow and eventually lead to Oops.

The code of this function seems incorrectly assuming the usage of
get_order().  We need to decrease at first, then align to
power-of-two.
Reported-and-tested-by: default avatarhe, bo <bo.he@intel.com>
Reported-by: default avatarzhang jun <jun.zhang@intel.com>
Cc: <stable@vger.kernel.org>
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent f3d737b6
...@@ -242,16 +242,12 @@ int snd_dma_alloc_pages_fallback(int type, struct device *device, size_t size, ...@@ -242,16 +242,12 @@ int snd_dma_alloc_pages_fallback(int type, struct device *device, size_t size,
int err; int err;
while ((err = snd_dma_alloc_pages(type, device, size, dmab)) < 0) { while ((err = snd_dma_alloc_pages(type, device, size, dmab)) < 0) {
size_t aligned_size;
if (err != -ENOMEM) if (err != -ENOMEM)
return err; return err;
if (size <= PAGE_SIZE) if (size <= PAGE_SIZE)
return -ENOMEM; return -ENOMEM;
aligned_size = PAGE_SIZE << get_order(size);
if (size != aligned_size)
size = aligned_size;
else
size >>= 1; size >>= 1;
size = PAGE_SIZE << get_order(size);
} }
if (! dmab->area) if (! dmab->area)
return -ENOMEM; return -ENOMEM;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment