Commit 7c37ae5c authored by Takashi Iwai's avatar Takashi Iwai

ALSA: seq: Rewrite sequencer device binding with standard bus

We've used the old house-made code for binding the sequencer device
and driver.  This can be far better implemented with the standard
bus nowadays.

This patch refactors the whole sequencer binding code with the bus
/sys/bus/snd_seq.  The devices appear as id-card-device on this bus
and are bound with the drivers corresponding to the given id like the
former implementation.  The module autoload is also kept like before.

There is no change in API functions by this patch, and almost all
transitions are kept inside seq_device.c.  The proc file output will
change slightly but kept compatible as much as possible.

Further integration works will follow in later patches.
Signed-off-by: default avatarTakashi Iwai <tiwai@suse.de>
parent 72496edc
...@@ -43,8 +43,11 @@ struct snd_seq_device { ...@@ -43,8 +43,11 @@ struct snd_seq_device {
void *private_data; /* private data for the caller */ void *private_data; /* private data for the caller */
void (*private_free)(struct snd_seq_device *device); void (*private_free)(struct snd_seq_device *device);
struct list_head list; /* link to next device */ struct list_head list; /* link to next device */
struct device dev;
}; };
#define to_seq_dev(_dev) \
container_of(_dev, struct snd_seq_device, dev)
/* driver operators /* driver operators
* init_device: * init_device:
......
...@@ -36,6 +36,7 @@ ...@@ -36,6 +36,7 @@
* *
*/ */
#include <linux/device.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/module.h> #include <linux/module.h>
#include <sound/core.h> #include <sound/core.h>
...@@ -51,74 +52,54 @@ MODULE_AUTHOR("Takashi Iwai <tiwai@suse.de>"); ...@@ -51,74 +52,54 @@ MODULE_AUTHOR("Takashi Iwai <tiwai@suse.de>");
MODULE_DESCRIPTION("ALSA sequencer device management"); MODULE_DESCRIPTION("ALSA sequencer device management");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
/* driver state */ struct snd_seq_driver {
#define DRIVER_EMPTY 0 struct device_driver driver;
#define DRIVER_LOADED (1<<0) char id[ID_LEN];
#define DRIVER_REQUESTED (1<<1) int argsize;
#define DRIVER_LOCKED (1<<2) struct snd_seq_dev_ops ops;
#define DRIVER_REQUESTING (1<<3) };
struct ops_list { #define to_seq_drv(_drv) \
char id[ID_LEN]; /* driver id */ container_of(_drv, struct snd_seq_driver, driver)
int driver; /* driver state */
int used; /* reference counter */
int argsize; /* argument size */
/* operators */ /*
struct snd_seq_dev_ops ops; * bus definition
*/
static int snd_seq_bus_match(struct device *dev, struct device_driver *drv)
{
struct snd_seq_device *sdev = to_seq_dev(dev);
struct snd_seq_driver *sdrv = to_seq_drv(drv);
/* registered devices */ return strcmp(sdrv->id, sdev->id) == 0 &&
struct list_head dev_list; /* list of devices */ sdrv->argsize == sdev->argsize;
int num_devices; /* number of associated devices */ }
int num_init_devices; /* number of initialized devices */
struct mutex reg_mutex;
struct list_head list; /* next driver */ static struct bus_type snd_seq_bus_type = {
.name = "snd_seq",
.match = snd_seq_bus_match,
}; };
/*
static LIST_HEAD(opslist); * proc interface -- just for compatibility
static int num_ops; */
static DEFINE_MUTEX(ops_mutex);
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
static struct snd_info_entry *info_entry; static struct snd_info_entry *info_entry;
#endif
/* static int print_dev_info(struct device *dev, void *data)
* prototypes {
*/ struct snd_seq_device *sdev = to_seq_dev(dev);
static int snd_seq_device_free(struct snd_seq_device *dev); struct snd_info_buffer *buffer = data;
static int snd_seq_device_dev_free(struct snd_device *device);
static int snd_seq_device_dev_register(struct snd_device *device);
static int snd_seq_device_dev_disconnect(struct snd_device *device);
static int init_device(struct snd_seq_device *dev, struct ops_list *ops);
static int free_device(struct snd_seq_device *dev, struct ops_list *ops);
static struct ops_list *find_driver(char *id, int create_if_empty);
static struct ops_list *create_driver(char *id);
static void unlock_driver(struct ops_list *ops);
static void remove_drivers(void);
/* snd_iprintf(buffer, "snd-%s,%s,%d\n", sdev->id,
* show all drivers and their status dev->driver ? "loaded" : "empty",
*/ dev->driver ? 1 : 0);
return 0;
}
#ifdef CONFIG_PROC_FS
static void snd_seq_device_info(struct snd_info_entry *entry, static void snd_seq_device_info(struct snd_info_entry *entry,
struct snd_info_buffer *buffer) struct snd_info_buffer *buffer)
{ {
struct ops_list *ops; bus_for_each_dev(&snd_seq_bus_type, NULL, buffer, print_dev_info);
mutex_lock(&ops_mutex);
list_for_each_entry(ops, &opslist, list) {
snd_iprintf(buffer, "snd-%s%s%s%s,%d\n",
ops->id,
ops->driver & DRIVER_LOADED ? ",loaded" : (ops->driver == DRIVER_EMPTY ? ",empty" : ""),
ops->driver & DRIVER_REQUESTED ? ",requested" : "",
ops->driver & DRIVER_LOCKED ? ",locked" : "",
ops->num_devices);
}
mutex_unlock(&ops_mutex);
} }
#endif #endif
...@@ -141,52 +122,29 @@ void snd_seq_autoload_unlock(void) ...@@ -141,52 +122,29 @@ void snd_seq_autoload_unlock(void)
} }
EXPORT_SYMBOL(snd_seq_autoload_unlock); EXPORT_SYMBOL(snd_seq_autoload_unlock);
static void autoload_drivers(void) static int request_seq_drv(struct device *dev, void *data)
{ {
/* avoid reentrance */ struct snd_seq_device *sdev = to_seq_dev(dev);
if (atomic_inc_return(&snd_seq_in_init) == 1) {
struct ops_list *ops;
mutex_lock(&ops_mutex);
list_for_each_entry(ops, &opslist, list) {
if ((ops->driver & DRIVER_REQUESTING) &&
!(ops->driver & DRIVER_REQUESTED)) {
ops->used++;
mutex_unlock(&ops_mutex);
ops->driver |= DRIVER_REQUESTED;
request_module("snd-%s", ops->id);
mutex_lock(&ops_mutex);
ops->used--;
}
}
mutex_unlock(&ops_mutex);
}
atomic_dec(&snd_seq_in_init);
}
static void call_autoload(struct work_struct *work) if (!dev->driver)
{ request_module("snd-%s", sdev->id);
autoload_drivers(); return 0;
} }
static DECLARE_WORK(autoload_work, call_autoload); static void autoload_drivers(struct work_struct *work)
static void try_autoload(struct ops_list *ops)
{ {
if (!ops->driver) { /* avoid reentrance */
ops->driver |= DRIVER_REQUESTING; if (atomic_inc_return(&snd_seq_in_init) == 1)
schedule_work(&autoload_work); bus_for_each_dev(&snd_seq_bus_type, NULL, NULL,
} request_seq_drv);
atomic_dec(&snd_seq_in_init);
} }
static DECLARE_WORK(autoload_work, autoload_drivers);
static void queue_autoload_drivers(void) static void queue_autoload_drivers(void)
{ {
struct ops_list *ops; schedule_work(&autoload_work);
mutex_lock(&ops_mutex);
list_for_each_entry(ops, &opslist, list)
try_autoload(ops);
mutex_unlock(&ops_mutex);
} }
void snd_seq_autoload_init(void) void snd_seq_autoload_init(void)
...@@ -206,9 +164,50 @@ void snd_seq_device_load_drivers(void) ...@@ -206,9 +164,50 @@ void snd_seq_device_load_drivers(void)
} }
EXPORT_SYMBOL(snd_seq_device_load_drivers); EXPORT_SYMBOL(snd_seq_device_load_drivers);
#else #else
#define try_autoload(ops) /* NOP */ #define queue_autoload_drivers() /* NOP */
#endif #endif
/*
* device management
*/
static int snd_seq_device_dev_free(struct snd_device *device)
{
struct snd_seq_device *dev = device->device_data;
put_device(&dev->dev);
return 0;
}
static int snd_seq_device_dev_register(struct snd_device *device)
{
struct snd_seq_device *dev = device->device_data;
int err;
err = device_add(&dev->dev);
if (err < 0)
return err;
if (!dev->dev.driver)
queue_autoload_drivers();
return 0;
}
static int snd_seq_device_dev_disconnect(struct snd_device *device)
{
struct snd_seq_device *dev = device->device_data;
device_del(&dev->dev);
return 0;
}
static void snd_seq_dev_release(struct device *dev)
{
struct snd_seq_device *sdev = to_seq_dev(dev);
if (sdev->private_free)
sdev->private_free(sdev);
kfree(sdev);
}
/* /*
* register a sequencer device * register a sequencer device
* card = card info * card = card info
...@@ -220,7 +219,6 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize, ...@@ -220,7 +219,6 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize,
struct snd_seq_device **result) struct snd_seq_device **result)
{ {
struct snd_seq_device *dev; struct snd_seq_device *dev;
struct ops_list *ops;
int err; int err;
static struct snd_device_ops dops = { static struct snd_device_ops dops = {
.dev_free = snd_seq_device_dev_free, .dev_free = snd_seq_device_dev_free,
...@@ -234,16 +232,10 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize, ...@@ -234,16 +232,10 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize,
if (snd_BUG_ON(!id)) if (snd_BUG_ON(!id))
return -EINVAL; return -EINVAL;
ops = find_driver(id, 1); dev = kzalloc(sizeof(*dev) + argsize, GFP_KERNEL);
if (ops == NULL) if (!dev)
return -ENOMEM; return -ENOMEM;
dev = kzalloc(sizeof(*dev)*2 + argsize, GFP_KERNEL);
if (dev == NULL) {
unlock_driver(ops);
return -ENOMEM;
}
/* set up device info */ /* set up device info */
dev->card = card; dev->card = card;
dev->device = device; dev->device = device;
...@@ -251,20 +243,19 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize, ...@@ -251,20 +243,19 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize,
dev->argsize = argsize; dev->argsize = argsize;
dev->status = SNDRV_SEQ_DEVICE_FREE; dev->status = SNDRV_SEQ_DEVICE_FREE;
/* add this device to the list */ device_initialize(&dev->dev);
mutex_lock(&ops->reg_mutex); dev->dev.parent = &card->card_dev;
list_add_tail(&dev->list, &ops->dev_list); dev->dev.bus = &snd_seq_bus_type;
ops->num_devices++; dev->dev.release = snd_seq_dev_release;
mutex_unlock(&ops->reg_mutex); dev_set_name(&dev->dev, "%s-%d-%d", dev->id, card->number, device);
if ((err = snd_device_new(card, SNDRV_DEV_SEQUENCER, dev, &dops)) < 0) { /* add this device to the list */
snd_seq_device_free(dev); err = snd_device_new(card, SNDRV_DEV_SEQUENCER, dev, &dops);
if (err < 0) {
put_device(&dev->dev);
return err; return err;
} }
try_autoload(ops);
unlock_driver(ops);
if (result) if (result)
*result = dev; *result = dev;
...@@ -273,82 +264,33 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize, ...@@ -273,82 +264,33 @@ int snd_seq_device_new(struct snd_card *card, int device, char *id, int argsize,
EXPORT_SYMBOL(snd_seq_device_new); EXPORT_SYMBOL(snd_seq_device_new);
/* /*
* free the existing device * driver binding - just pass to each driver callback
*/ */
static int snd_seq_device_free(struct snd_seq_device *dev) static int snd_seq_drv_probe(struct device *dev)
{
struct ops_list *ops;
if (snd_BUG_ON(!dev))
return -EINVAL;
ops = find_driver(dev->id, 0);
if (ops == NULL)
return -ENXIO;
/* remove the device from the list */
mutex_lock(&ops->reg_mutex);
list_del(&dev->list);
ops->num_devices--;
mutex_unlock(&ops->reg_mutex);
free_device(dev, ops);
if (dev->private_free)
dev->private_free(dev);
kfree(dev);
unlock_driver(ops);
return 0;
}
static int snd_seq_device_dev_free(struct snd_device *device)
{ {
struct snd_seq_device *dev = device->device_data; struct snd_seq_driver *sdrv = to_seq_drv(dev->driver);
return snd_seq_device_free(dev); struct snd_seq_device *sdev = to_seq_dev(dev);
} int err;
/*
* register the device
*/
static int snd_seq_device_dev_register(struct snd_device *device)
{
struct snd_seq_device *dev = device->device_data;
struct ops_list *ops;
ops = find_driver(dev->id, 0);
if (ops == NULL)
return -ENOENT;
/* initialize this device if the corresponding driver was
* already loaded
*/
if (ops->driver & DRIVER_LOADED)
init_device(dev, ops);
unlock_driver(ops); err = sdrv->ops.init_device(sdev);
if (err < 0)
return err;
sdev->status = SNDRV_SEQ_DEVICE_REGISTERED;
return 0; return 0;
} }
EXPORT_SYMBOL(snd_seq_device_register_driver);
/* static int snd_seq_drv_remove(struct device *dev)
* disconnect the device
*/
static int snd_seq_device_dev_disconnect(struct snd_device *device)
{ {
struct snd_seq_device *dev = device->device_data; struct snd_seq_driver *sdrv = to_seq_drv(dev->driver);
struct ops_list *ops; struct snd_seq_device *sdev = to_seq_dev(dev);
int err;
ops = find_driver(dev->id, 0);
if (ops == NULL)
return -ENOENT;
free_device(dev, ops);
unlock_driver(ops); err = sdrv->ops.free_device(sdev);
if (err < 0)
return err;
sdev->status = SNDRV_SEQ_DEVICE_FREE;
return 0; return 0;
} }
EXPORT_SYMBOL(snd_seq_device_unregister_driver);
/* /*
* register device driver * register device driver
...@@ -358,226 +300,66 @@ EXPORT_SYMBOL(snd_seq_device_unregister_driver); ...@@ -358,226 +300,66 @@ EXPORT_SYMBOL(snd_seq_device_unregister_driver);
int snd_seq_device_register_driver(char *id, struct snd_seq_dev_ops *entry, int snd_seq_device_register_driver(char *id, struct snd_seq_dev_ops *entry,
int argsize) int argsize)
{ {
struct ops_list *ops; struct snd_seq_driver *sdrv;
struct snd_seq_device *dev; int err;
if (id == NULL || entry == NULL || if (id == NULL || entry == NULL ||
entry->init_device == NULL || entry->free_device == NULL) entry->init_device == NULL || entry->free_device == NULL)
return -EINVAL; return -EINVAL;
ops = find_driver(id, 1); sdrv = kzalloc(sizeof(*sdrv), GFP_KERNEL);
if (ops == NULL) if (!sdrv)
return -ENOMEM; return -ENOMEM;
if (ops->driver & DRIVER_LOADED) {
pr_warn("ALSA: seq: driver_register: driver '%s' already exists\n", id);
unlock_driver(ops);
return -EBUSY;
}
mutex_lock(&ops->reg_mutex);
/* copy driver operators */
ops->ops = *entry;
ops->driver |= DRIVER_LOADED;
ops->argsize = argsize;
/* initialize existing devices if necessary */
list_for_each_entry(dev, &ops->dev_list, list) {
init_device(dev, ops);
}
mutex_unlock(&ops->reg_mutex);
unlock_driver(ops); sdrv->driver.name = id;
sdrv->driver.bus = &snd_seq_bus_type;
return 0; sdrv->driver.probe = snd_seq_drv_probe;
sdrv->driver.remove = snd_seq_drv_remove;
strlcpy(sdrv->id, id, sizeof(sdrv->id));
sdrv->argsize = argsize;
sdrv->ops = *entry;
err = driver_register(&sdrv->driver);
if (err < 0)
kfree(sdrv);
return err;
} }
EXPORT_SYMBOL(snd_seq_device_register_driver);
/* callback to find a specific driver; data is a pointer to the id string ptr.
/* * when the id matches, store the driver pointer in return and break the loop.
* create driver record
*/ */
static struct ops_list * create_driver(char *id) static int find_drv(struct device_driver *drv, void *data)
{ {
struct ops_list *ops; struct snd_seq_driver *sdrv = to_seq_drv(drv);
void **ptr = (void **)data;
ops = kzalloc(sizeof(*ops), GFP_KERNEL);
if (ops == NULL)
return ops;
/* set up driver entry */
strlcpy(ops->id, id, sizeof(ops->id));
mutex_init(&ops->reg_mutex);
/*
* The ->reg_mutex locking rules are per-driver, so we create
* separate per-driver lock classes:
*/
lockdep_set_class(&ops->reg_mutex, (struct lock_class_key *)id);
ops->driver = DRIVER_EMPTY;
INIT_LIST_HEAD(&ops->dev_list);
/* lock this instance */
ops->used = 1;
/* register driver entry */
mutex_lock(&ops_mutex);
list_add_tail(&ops->list, &opslist);
num_ops++;
mutex_unlock(&ops_mutex);
return ops; if (strcmp(sdrv->id, *ptr))
return 0; /* id don't match, continue the loop */
*ptr = sdrv;
return 1; /* break the loop */
} }
/* /*
* unregister the specified driver * unregister the specified driver
*/ */
int snd_seq_device_unregister_driver(char *id) int snd_seq_device_unregister_driver(char *id)
{ {
struct ops_list *ops; struct snd_seq_driver *sdrv = (struct snd_seq_driver *)id;
struct snd_seq_device *dev;
ops = find_driver(id, 0); if (!bus_for_each_drv(&snd_seq_bus_type, NULL, &sdrv, find_drv))
if (ops == NULL)
return -ENXIO; return -ENXIO;
if (! (ops->driver & DRIVER_LOADED) || driver_unregister(&sdrv->driver);
(ops->driver & DRIVER_LOCKED)) { kfree(sdrv);
pr_err("ALSA: seq: driver_unregister: cannot unload driver '%s': status=%x\n",
id, ops->driver);
unlock_driver(ops);
return -EBUSY;
}
/* close and release all devices associated with this driver */
mutex_lock(&ops->reg_mutex);
ops->driver |= DRIVER_LOCKED; /* do not remove this driver recursively */
list_for_each_entry(dev, &ops->dev_list, list) {
free_device(dev, ops);
}
ops->driver = 0;
if (ops->num_init_devices > 0)
pr_err("ALSA: seq: free_driver: init_devices > 0!! (%d)\n",
ops->num_init_devices);
mutex_unlock(&ops->reg_mutex);
unlock_driver(ops);
/* remove empty driver entries */
remove_drivers();
return 0;
}
/*
* remove empty driver entries
*/
static void remove_drivers(void)
{
struct list_head *head;
mutex_lock(&ops_mutex);
head = opslist.next;
while (head != &opslist) {
struct ops_list *ops = list_entry(head, struct ops_list, list);
if (! (ops->driver & DRIVER_LOADED) &&
ops->used == 0 && ops->num_devices == 0) {
head = head->next;
list_del(&ops->list);
kfree(ops);
num_ops--;
} else
head = head->next;
}
mutex_unlock(&ops_mutex);
}
/*
* initialize the device - call init_device operator
*/
static int init_device(struct snd_seq_device *dev, struct ops_list *ops)
{
if (! (ops->driver & DRIVER_LOADED))
return 0; /* driver is not loaded yet */
if (dev->status != SNDRV_SEQ_DEVICE_FREE)
return 0; /* already initialized */
if (ops->argsize != dev->argsize) {
pr_err("ALSA: seq: incompatible device '%s' for plug-in '%s' (%d %d)\n",
dev->name, ops->id, ops->argsize, dev->argsize);
return -EINVAL;
}
if (ops->ops.init_device(dev) >= 0) {
dev->status = SNDRV_SEQ_DEVICE_REGISTERED;
ops->num_init_devices++;
} else {
pr_err("ALSA: seq: init_device failed: %s: %s\n",
dev->name, dev->id);
}
return 0; return 0;
} }
EXPORT_SYMBOL(snd_seq_device_unregister_driver);
/*
* release the device - call free_device operator
*/
static int free_device(struct snd_seq_device *dev, struct ops_list *ops)
{
int result;
if (! (ops->driver & DRIVER_LOADED))
return 0; /* driver is not loaded yet */
if (dev->status != SNDRV_SEQ_DEVICE_REGISTERED)
return 0; /* not registered */
if (ops->argsize != dev->argsize) {
pr_err("ALSA: seq: incompatible device '%s' for plug-in '%s' (%d %d)\n",
dev->name, ops->id, ops->argsize, dev->argsize);
return -EINVAL;
}
if ((result = ops->ops.free_device(dev)) >= 0 || result == -ENXIO) {
dev->status = SNDRV_SEQ_DEVICE_FREE;
dev->driver_data = NULL;
ops->num_init_devices--;
} else {
pr_err("ALSA: seq: free_device failed: %s: %s\n",
dev->name, dev->id);
}
return 0;
}
/*
* find the matching driver with given id
*/
static struct ops_list * find_driver(char *id, int create_if_empty)
{
struct ops_list *ops;
mutex_lock(&ops_mutex);
list_for_each_entry(ops, &opslist, list) {
if (strcmp(ops->id, id) == 0) {
ops->used++;
mutex_unlock(&ops_mutex);
return ops;
}
}
mutex_unlock(&ops_mutex);
if (create_if_empty)
return create_driver(id);
return NULL;
}
static void unlock_driver(struct ops_list *ops)
{
mutex_lock(&ops_mutex);
ops->used--;
mutex_unlock(&ops_mutex);
}
/* /*
* module part * module part
*/ */
static int __init alsa_seq_device_init(void) static int __init seq_dev_proc_init(void)
{ {
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
info_entry = snd_info_create_module_entry(THIS_MODULE, "drivers", info_entry = snd_info_create_module_entry(THIS_MODULE, "drivers",
...@@ -594,17 +376,28 @@ static int __init alsa_seq_device_init(void) ...@@ -594,17 +376,28 @@ static int __init alsa_seq_device_init(void)
return 0; return 0;
} }
static int __init alsa_seq_device_init(void)
{
int err;
err = bus_register(&snd_seq_bus_type);
if (err < 0)
return err;
err = seq_dev_proc_init();
if (err < 0)
bus_unregister(&snd_seq_bus_type);
return err;
}
static void __exit alsa_seq_device_exit(void) static void __exit alsa_seq_device_exit(void)
{ {
#ifdef CONFIG_MODULES #ifdef CONFIG_MODULES
cancel_work_sync(&autoload_work); cancel_work_sync(&autoload_work);
#endif #endif
remove_drivers();
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
snd_info_free_entry(info_entry); snd_info_free_entry(info_entry);
#endif #endif
if (num_ops) bus_unregister(&snd_seq_bus_type);
pr_err("ALSA: seq: drivers not released (%d)\n", num_ops);
} }
module_init(alsa_seq_device_init) module_init(alsa_seq_device_init)
......
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