sound: push BKL into open functions

This moves the lock_kernel() call from soundcore_open
to the individual OSS device drivers, where we can deal
with it one driver at a time if needed, or just kill
off the drivers.

All core components in ALSA already provide
adequate locking in their open()-functions
and do not require the big kernel lock, so
there is no need to add the BKL there.

Signed-off-by: Arnd Bergmann <arnd@arndb.de>
Signed-off-by: Takashi Iwai <tiwai@suse.de>
This commit is contained in:
Arnd Bergmann 2010-07-11 12:16:36 +02:00 committed by Takashi Iwai
parent 395c61d196
commit 90dc763fef
9 changed files with 95 additions and 35 deletions

View file

@ -8,6 +8,7 @@
#include "linux/slab.h" #include "linux/slab.h"
#include "linux/sound.h" #include "linux/sound.h"
#include "linux/soundcard.h" #include "linux/soundcard.h"
#include "linux/smp_lock.h"
#include "asm/uaccess.h" #include "asm/uaccess.h"
#include "init.h" #include "init.h"
#include "os.h" #include "os.h"
@ -198,7 +199,10 @@ static int hostaudio_open(struct inode *inode, struct file *file)
if (file->f_mode & FMODE_WRITE) if (file->f_mode & FMODE_WRITE)
w = 1; w = 1;
lock_kernel();
ret = os_open_file(dsp, of_set_rw(OPENFLAGS(), r, w), 0); ret = os_open_file(dsp, of_set_rw(OPENFLAGS(), r, w), 0);
unlock_kernel();
if (ret < 0) { if (ret < 0) {
kfree(state); kfree(state);
return ret; return ret;
@ -254,7 +258,9 @@ static int hostmixer_open_mixdev(struct inode *inode, struct file *file)
if (file->f_mode & FMODE_WRITE) if (file->f_mode & FMODE_WRITE)
w = 1; w = 1;
lock_kernel();
ret = os_open_file(mixer, of_set_rw(OPENFLAGS(), r, w), 0); ret = os_open_file(mixer, of_set_rw(OPENFLAGS(), r, w), 0);
unlock_kernel();
if (ret < 0) { if (ret < 0) {
printk(KERN_ERR "hostaudio_open_mixdev failed to open '%s', " printk(KERN_ERR "hostaudio_open_mixdev failed to open '%s', "

View file

@ -43,6 +43,7 @@
#include <linux/sound.h> #include <linux/sound.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/soundcard.h> #include <linux/soundcard.h>
#include <linux/smp_lock.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/kernel.h> #include <linux/kernel.h>
@ -807,7 +808,9 @@ au1550_llseek(struct file *file, loff_t offset, int origin)
static int static int
au1550_open_mixdev(struct inode *inode, struct file *file) au1550_open_mixdev(struct inode *inode, struct file *file)
{ {
lock_kernel();
file->private_data = &au1550_state; file->private_data = &au1550_state;
unlock_kernel();
return 0; return 0;
} }
@ -1797,21 +1800,22 @@ au1550_open(struct inode *inode, struct file *file)
#endif #endif
file->private_data = s; file->private_data = s;
lock_kernel();
/* wait for device to become free */ /* wait for device to become free */
mutex_lock(&s->open_mutex); mutex_lock(&s->open_mutex);
while (s->open_mode & file->f_mode) { while (s->open_mode & file->f_mode) {
if (file->f_flags & O_NONBLOCK) { ret = -EBUSY;
mutex_unlock(&s->open_mutex); if (file->f_flags & O_NONBLOCK)
return -EBUSY; goto out;
}
add_wait_queue(&s->open_wait, &wait); add_wait_queue(&s->open_wait, &wait);
__set_current_state(TASK_INTERRUPTIBLE); __set_current_state(TASK_INTERRUPTIBLE);
mutex_unlock(&s->open_mutex); mutex_unlock(&s->open_mutex);
schedule(); schedule();
remove_wait_queue(&s->open_wait, &wait); remove_wait_queue(&s->open_wait, &wait);
set_current_state(TASK_RUNNING); set_current_state(TASK_RUNNING);
ret = -ERESTARTSYS;
if (signal_pending(current)) if (signal_pending(current))
return -ERESTARTSYS; goto out2;
mutex_lock(&s->open_mutex); mutex_lock(&s->open_mutex);
} }
@ -1840,17 +1844,21 @@ au1550_open(struct inode *inode, struct file *file)
if (file->f_mode & FMODE_READ) { if (file->f_mode & FMODE_READ) {
if ((ret = prog_dmabuf_adc(s))) if ((ret = prog_dmabuf_adc(s)))
return ret; goto out;
} }
if (file->f_mode & FMODE_WRITE) { if (file->f_mode & FMODE_WRITE) {
if ((ret = prog_dmabuf_dac(s))) if ((ret = prog_dmabuf_dac(s)))
return ret; goto out;
} }
s->open_mode |= file->f_mode & (FMODE_READ | FMODE_WRITE); s->open_mode |= file->f_mode & (FMODE_READ | FMODE_WRITE);
mutex_unlock(&s->open_mutex);
mutex_init(&s->sem); mutex_init(&s->sem);
return 0; ret = 0;
out:
mutex_unlock(&s->open_mutex);
out2:
unlock_kernel();
return ret;
} }
static int static int

View file

@ -323,9 +323,13 @@ static struct {
static int mixer_open(struct inode *inode, struct file *file) static int mixer_open(struct inode *inode, struct file *file)
{ {
if (!try_module_get(dmasound.mach.owner)) lock_kernel();
if (!try_module_get(dmasound.mach.owner)) {
unlock_kernel();
return -ENODEV; return -ENODEV;
}
mixer.busy = 1; mixer.busy = 1;
unlock_kernel();
return 0; return 0;
} }
@ -737,8 +741,11 @@ static int sq_open(struct inode *inode, struct file *file)
{ {
int rc; int rc;
if (!try_module_get(dmasound.mach.owner)) lock_kernel();
if (!try_module_get(dmasound.mach.owner)) {
unlock_kernel();
return -ENODEV; return -ENODEV;
}
rc = write_sq_open(file); /* checks the f_mode */ rc = write_sq_open(file); /* checks the f_mode */
if (rc) if (rc)
@ -781,10 +788,11 @@ static int sq_open(struct inode *inode, struct file *file)
sound_set_format(AFMT_MU_LAW); sound_set_format(AFMT_MU_LAW);
} }
#endif #endif
unlock_kernel();
return 0; return 0;
out: out:
module_put(dmasound.mach.owner); module_put(dmasound.mach.owner);
unlock_kernel();
return rc; return rc;
} }
@ -1226,12 +1234,17 @@ static int state_open(struct inode *inode, struct file *file)
{ {
char *buffer = state.buf; char *buffer = state.buf;
int len = 0; int len = 0;
int ret;
lock_kernel();
ret = -EBUSY;
if (state.busy) if (state.busy)
return -EBUSY; goto out;
ret = -ENODEV;
if (!try_module_get(dmasound.mach.owner)) if (!try_module_get(dmasound.mach.owner))
return -ENODEV; goto out;
state.ptr = 0; state.ptr = 0;
state.busy = 1; state.busy = 1;
@ -1293,7 +1306,10 @@ printk("dmasound: stat buffer used %d bytes\n", len) ;
printk(KERN_ERR "dmasound_core: stat buffer overflowed!\n"); printk(KERN_ERR "dmasound_core: stat buffer overflowed!\n");
state.len = len; state.len = len;
return 0; ret = 0;
out:
unlock_kernel();
return ret;
} }
static int state_release(struct inode *inode, struct file *file) static int state_release(struct inode *inode, struct file *file)

View file

@ -756,12 +756,15 @@ static int dev_open(struct inode *inode, struct file *file)
int minor = iminor(inode); int minor = iminor(inode);
int err = 0; int err = 0;
lock_kernel();
if (minor == dev.dsp_minor) { if (minor == dev.dsp_minor) {
if ((file->f_mode & FMODE_WRITE && if ((file->f_mode & FMODE_WRITE &&
test_bit(F_AUDIO_WRITE_INUSE, &dev.flags)) || test_bit(F_AUDIO_WRITE_INUSE, &dev.flags)) ||
(file->f_mode & FMODE_READ && (file->f_mode & FMODE_READ &&
test_bit(F_AUDIO_READ_INUSE, &dev.flags))) test_bit(F_AUDIO_READ_INUSE, &dev.flags))) {
return -EBUSY; err = -EBUSY;
goto out;
}
if ((err = dsp_open(file)) >= 0) { if ((err = dsp_open(file)) >= 0) {
dev.nresets = 0; dev.nresets = 0;
@ -782,7 +785,8 @@ static int dev_open(struct inode *inode, struct file *file)
/* nothing */ /* nothing */
} else } else
err = -EINVAL; err = -EINVAL;
out:
unlock_kernel();
return err; return err;
} }

View file

@ -16,6 +16,7 @@
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/fs.h> #include <linux/fs.h>
#include <linux/sound.h> #include <linux/sound.h>
#include <linux/smp_lock.h>
#include <linux/soundcard.h> #include <linux/soundcard.h>
#include <linux/interrupt.h> #include <linux/interrupt.h>
#include <linux/hrtimer.h> #include <linux/hrtimer.h>
@ -216,13 +217,17 @@ static int dac_audio_open(struct inode *inode, struct file *file)
{ {
if (file->f_mode & FMODE_READ) if (file->f_mode & FMODE_READ)
return -ENODEV; return -ENODEV;
if (in_use)
lock_kernel();
if (in_use) {
unlock_kernel();
return -EBUSY; return -EBUSY;
}
in_use = 1; in_use = 1;
dac_audio_start(); dac_audio_start();
unlock_kernel();
return 0; return 0;
} }

View file

@ -210,42 +210,44 @@ static int sound_open(struct inode *inode, struct file *file)
printk(KERN_ERR "Invalid minor device %d\n", dev); printk(KERN_ERR "Invalid minor device %d\n", dev);
return -ENXIO; return -ENXIO;
} }
lock_kernel();
switch (dev & 0x0f) { switch (dev & 0x0f) {
case SND_DEV_CTL: case SND_DEV_CTL:
dev >>= 4; dev >>= 4;
if (dev >= 0 && dev < MAX_MIXER_DEV && mixer_devs[dev] == NULL) { if (dev >= 0 && dev < MAX_MIXER_DEV && mixer_devs[dev] == NULL) {
request_module("mixer%d", dev); request_module("mixer%d", dev);
} }
retval = -ENXIO;
if (dev && (dev >= num_mixers || mixer_devs[dev] == NULL)) if (dev && (dev >= num_mixers || mixer_devs[dev] == NULL))
return -ENXIO; break;
if (!try_module_get(mixer_devs[dev]->owner)) if (!try_module_get(mixer_devs[dev]->owner))
return -ENXIO; break;
retval = 0;
break; break;
case SND_DEV_SEQ: case SND_DEV_SEQ:
case SND_DEV_SEQ2: case SND_DEV_SEQ2:
if ((retval = sequencer_open(dev, file)) < 0) retval = sequencer_open(dev, file);
return retval;
break; break;
case SND_DEV_MIDIN: case SND_DEV_MIDIN:
if ((retval = MIDIbuf_open(dev, file)) < 0) retval = MIDIbuf_open(dev, file);
return retval;
break; break;
case SND_DEV_DSP: case SND_DEV_DSP:
case SND_DEV_DSP16: case SND_DEV_DSP16:
case SND_DEV_AUDIO: case SND_DEV_AUDIO:
if ((retval = audio_open(dev, file)) < 0) retval = audio_open(dev, file);
return retval;
break; break;
default: default:
printk(KERN_ERR "Invalid minor device %d\n", dev); printk(KERN_ERR "Invalid minor device %d\n", dev);
return -ENXIO; retval = -ENXIO;
} }
unlock_kernel();
return 0; return 0;
} }

View file

@ -68,6 +68,7 @@
#include <linux/delay.h> #include <linux/delay.h>
#include <linux/sound.h> #include <linux/sound.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/smp_lock.h>
#include <linux/soundcard.h> #include <linux/soundcard.h>
#include <linux/ac97_codec.h> #include <linux/ac97_codec.h>
#include <linux/pci.h> #include <linux/pci.h>
@ -1534,6 +1535,7 @@ static int cs4297a_open_mixdev(struct inode *inode, struct file *file)
CS_DBGOUT(CS_FUNCTION | CS_OPEN, 4, CS_DBGOUT(CS_FUNCTION | CS_OPEN, 4,
printk(KERN_INFO "cs4297a: cs4297a_open_mixdev()+\n")); printk(KERN_INFO "cs4297a: cs4297a_open_mixdev()+\n"));
lock_kernel();
list_for_each(entry, &cs4297a_devs) list_for_each(entry, &cs4297a_devs)
{ {
s = list_entry(entry, struct cs4297a_state, list); s = list_entry(entry, struct cs4297a_state, list);
@ -1544,6 +1546,8 @@ static int cs4297a_open_mixdev(struct inode *inode, struct file *file)
{ {
CS_DBGOUT(CS_FUNCTION | CS_OPEN | CS_ERROR, 2, CS_DBGOUT(CS_FUNCTION | CS_OPEN | CS_ERROR, 2,
printk(KERN_INFO "cs4297a: cs4297a_open_mixdev()- -ENODEV\n")); printk(KERN_INFO "cs4297a: cs4297a_open_mixdev()- -ENODEV\n"));
unlock_kernel();
return -ENODEV; return -ENODEV;
} }
VALIDATE_STATE(s); VALIDATE_STATE(s);
@ -1551,6 +1555,7 @@ static int cs4297a_open_mixdev(struct inode *inode, struct file *file)
CS_DBGOUT(CS_FUNCTION | CS_OPEN, 4, CS_DBGOUT(CS_FUNCTION | CS_OPEN, 4,
printk(KERN_INFO "cs4297a: cs4297a_open_mixdev()- 0\n")); printk(KERN_INFO "cs4297a: cs4297a_open_mixdev()- 0\n"));
unlock_kernel();
return nonseekable_open(inode, file); return nonseekable_open(inode, file);
} }
@ -2369,7 +2374,7 @@ static int cs4297a_release(struct inode *inode, struct file *file)
return 0; return 0;
} }
static int cs4297a_open(struct inode *inode, struct file *file) static int cs4297a_locked_open(struct inode *inode, struct file *file)
{ {
int minor = iminor(inode); int minor = iminor(inode);
struct cs4297a_state *s=NULL; struct cs4297a_state *s=NULL;
@ -2486,6 +2491,16 @@ static int cs4297a_open(struct inode *inode, struct file *file)
return nonseekable_open(inode, file); return nonseekable_open(inode, file);
} }
static int cs4297a_open(struct inode *inode, struct file *file)
{
int ret;
lock_kernel();
ret = cs4297a_open(inode, file);
unlock_kernel();
return ret;
}
// ****************************************************************************************** // ******************************************************************************************
// Wave (audio) file operations struct. // Wave (audio) file operations struct.

View file

@ -2921,6 +2921,7 @@ static int vwsnd_audio_open(struct inode *inode, struct file *file)
DBGE("(inode=0x%p, file=0x%p)\n", inode, file); DBGE("(inode=0x%p, file=0x%p)\n", inode, file);
lock_kernel();
INC_USE_COUNT; INC_USE_COUNT;
for (devc = vwsnd_dev_list; devc; devc = devc->next_dev) for (devc = vwsnd_dev_list; devc; devc = devc->next_dev)
if ((devc->audio_minor & ~0x0F) == (minor & ~0x0F)) if ((devc->audio_minor & ~0x0F) == (minor & ~0x0F))
@ -2928,6 +2929,7 @@ static int vwsnd_audio_open(struct inode *inode, struct file *file)
if (devc == NULL) { if (devc == NULL) {
DEC_USE_COUNT; DEC_USE_COUNT;
unlock_kernel();
return -ENODEV; return -ENODEV;
} }
@ -2936,11 +2938,13 @@ static int vwsnd_audio_open(struct inode *inode, struct file *file)
mutex_unlock(&devc->open_mutex); mutex_unlock(&devc->open_mutex);
if (file->f_flags & O_NONBLOCK) { if (file->f_flags & O_NONBLOCK) {
DEC_USE_COUNT; DEC_USE_COUNT;
unlock_kernel();
return -EBUSY; return -EBUSY;
} }
interruptible_sleep_on(&devc->open_wait); interruptible_sleep_on(&devc->open_wait);
if (signal_pending(current)) { if (signal_pending(current)) {
DEC_USE_COUNT; DEC_USE_COUNT;
unlock_kernel();
return -ERESTARTSYS; return -ERESTARTSYS;
} }
mutex_lock(&devc->open_mutex); mutex_lock(&devc->open_mutex);
@ -2993,6 +2997,7 @@ static int vwsnd_audio_open(struct inode *inode, struct file *file)
file->private_data = devc; file->private_data = devc;
DBGRV(); DBGRV();
unlock_kernel();
return 0; return 0;
} }
@ -3062,15 +3067,18 @@ static int vwsnd_mixer_open(struct inode *inode, struct file *file)
DBGEV("(inode=0x%p, file=0x%p)\n", inode, file); DBGEV("(inode=0x%p, file=0x%p)\n", inode, file);
INC_USE_COUNT; INC_USE_COUNT;
lock_kernel();
for (devc = vwsnd_dev_list; devc; devc = devc->next_dev) for (devc = vwsnd_dev_list; devc; devc = devc->next_dev)
if (devc->mixer_minor == iminor(inode)) if (devc->mixer_minor == iminor(inode))
break; break;
if (devc == NULL) { if (devc == NULL) {
DEC_USE_COUNT; DEC_USE_COUNT;
unlock_kernel();
return -ENODEV; return -ENODEV;
} }
file->private_data = devc; file->private_data = devc;
unlock_kernel();
return 0; return 0;
} }

View file

@ -629,12 +629,8 @@ static int soundcore_open(struct inode *inode, struct file *file)
file->f_op = new_fops; file->f_op = new_fops;
spin_unlock(&sound_loader_lock); spin_unlock(&sound_loader_lock);
if (file->f_op->open) { if (file->f_op->open)
/* TODO: push down BKL into indivial open functions */
lock_kernel();
err = file->f_op->open(inode,file); err = file->f_op->open(inode,file);
unlock_kernel();
}
if (err) { if (err) {
fops_put(file->f_op); fops_put(file->f_op);