diff options
author | Takashi Iwai <tiwai@suse.de> | 2016-07-04 08:02:15 -0400 |
---|---|---|
committer | Takashi Iwai <tiwai@suse.de> | 2016-07-04 08:02:15 -0400 |
commit | 3fa6993fef634e05d200d141a85df0b044572364 (patch) | |
tree | d42f578d6f7f0ef27f4f556a36afe643636e5ad1 | |
parent | 046e9ceefab985a137fbd15b946497a88e5a6513 (diff) |
ALSA: timer: Fix negative queue usage by racy accesses
The user timer tu->qused counter may go to a negative value when
multiple concurrent reads are performed since both the check and the
decrement of tu->qused are done in two individual locked contexts.
This results in bogus read outs, and the endless loop in the
user-space side.
The fix is to move the decrement of the tu->qused counter into the
same spinlock context as the zero-check of the counter.
Cc: <stable@vger.kernel.org>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
-rw-r--r-- | sound/core/timer.c | 2 |
1 files changed, 1 insertions, 1 deletions
diff --git a/sound/core/timer.c b/sound/core/timer.c index e722022d325d..9a6157ea6881 100644 --- a/sound/core/timer.c +++ b/sound/core/timer.c | |||
@@ -1955,6 +1955,7 @@ static ssize_t snd_timer_user_read(struct file *file, char __user *buffer, | |||
1955 | 1955 | ||
1956 | qhead = tu->qhead++; | 1956 | qhead = tu->qhead++; |
1957 | tu->qhead %= tu->queue_size; | 1957 | tu->qhead %= tu->queue_size; |
1958 | tu->qused--; | ||
1958 | spin_unlock_irq(&tu->qlock); | 1959 | spin_unlock_irq(&tu->qlock); |
1959 | 1960 | ||
1960 | if (tu->tread) { | 1961 | if (tu->tread) { |
@@ -1968,7 +1969,6 @@ static ssize_t snd_timer_user_read(struct file *file, char __user *buffer, | |||
1968 | } | 1969 | } |
1969 | 1970 | ||
1970 | spin_lock_irq(&tu->qlock); | 1971 | spin_lock_irq(&tu->qlock); |
1971 | tu->qused--; | ||
1972 | if (err < 0) | 1972 | if (err < 0) |
1973 | goto _error; | 1973 | goto _error; |
1974 | result += unit; | 1974 | result += unit; |