aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--include/sound/core.h4
-rw-r--r--sound/core/pcm.c24
-rw-r--r--sound/core/sound.c56
3 files changed, 72 insertions, 12 deletions
diff --git a/include/sound/core.h b/include/sound/core.h
index 1359c532b68e..b056ea925ecf 100644
--- a/include/sound/core.h
+++ b/include/sound/core.h
@@ -26,6 +26,7 @@
26#include <linux/mutex.h> /* struct mutex */ 26#include <linux/mutex.h> /* struct mutex */
27#include <linux/rwsem.h> /* struct rw_semaphore */ 27#include <linux/rwsem.h> /* struct rw_semaphore */
28#include <linux/pm.h> /* pm_message_t */ 28#include <linux/pm.h> /* pm_message_t */
29#include <linux/device.h>
29 30
30/* forward declarations */ 31/* forward declarations */
31#ifdef CONFIG_PCI 32#ifdef CONFIG_PCI
@@ -186,6 +187,7 @@ struct snd_minor {
186 int device; /* device number */ 187 int device; /* device number */
187 const struct file_operations *f_ops; /* file operations */ 188 const struct file_operations *f_ops; /* file operations */
188 void *private_data; /* private data for f_ops->open */ 189 void *private_data; /* private data for f_ops->open */
190 struct class_device *class_dev; /* class device for sysfs */
189}; 191};
190 192
191/* sound.c */ 193/* sound.c */
@@ -200,6 +202,8 @@ int snd_register_device(int type, struct snd_card *card, int dev,
200 const char *name); 202 const char *name);
201int snd_unregister_device(int type, struct snd_card *card, int dev); 203int snd_unregister_device(int type, struct snd_card *card, int dev);
202void *snd_lookup_minor_data(unsigned int minor, int type); 204void *snd_lookup_minor_data(unsigned int minor, int type);
205int snd_add_device_sysfs_file(int type, struct snd_card *card, int dev,
206 const struct class_device_attribute *attr);
203 207
204#ifdef CONFIG_SND_OSSEMUL 208#ifdef CONFIG_SND_OSSEMUL
205int snd_register_oss_device(int type, struct snd_card *card, int dev, 209int snd_register_oss_device(int type, struct snd_card *card, int dev,
diff --git a/sound/core/pcm.c b/sound/core/pcm.c
index ed3b09469560..bf8f412988b8 100644
--- a/sound/core/pcm.c
+++ b/sound/core/pcm.c
@@ -907,6 +907,28 @@ void snd_pcm_detach_substream(struct snd_pcm_substream *substream)
907 substream->pstr->substream_opened--; 907 substream->pstr->substream_opened--;
908} 908}
909 909
910static ssize_t show_pcm_class(struct class_device *class_device, char *buf)
911{
912 struct snd_pcm *pcm;
913 const char *str;
914 static const char *strs[SNDRV_PCM_CLASS_LAST + 1] = {
915 [SNDRV_PCM_CLASS_GENERIC] = "generic",
916 [SNDRV_PCM_CLASS_MULTI] = "multi",
917 [SNDRV_PCM_CLASS_MODEM] = "modem",
918 [SNDRV_PCM_CLASS_DIGITIZER] = "digitizer",
919 };
920
921 if (! (pcm = class_get_devdata(class_device)) ||
922 pcm->dev_class > SNDRV_PCM_CLASS_LAST)
923 str = "none";
924 else
925 str = strs[pcm->dev_class];
926 return snprintf(buf, PAGE_SIZE, "%s\n", str);
927}
928
929static struct class_device_attribute pcm_attrs =
930 __ATTR(pcm_class, S_IRUGO, show_pcm_class, NULL);
931
910static int snd_pcm_dev_register(struct snd_device *device) 932static int snd_pcm_dev_register(struct snd_device *device)
911{ 933{
912 int cidx, err; 934 int cidx, err;
@@ -945,6 +967,8 @@ static int snd_pcm_dev_register(struct snd_device *device)
945 mutex_unlock(&register_mutex); 967 mutex_unlock(&register_mutex);
946 return err; 968 return err;
947 } 969 }
970 snd_add_device_sysfs_file(devtype, pcm->card, pcm->device,
971 &pcm_attrs);
948 for (substream = pcm->streams[cidx].substream; substream; substream = substream->next) 972 for (substream = pcm->streams[cidx].substream; substream; substream = substream->next)
949 snd_pcm_timer_init(substream); 973 snd_pcm_timer_init(substream);
950 } 974 }
diff --git a/sound/core/sound.c b/sound/core/sound.c
index b4430db3fa4c..efa476c5210a 100644
--- a/sound/core/sound.c
+++ b/sound/core/sound.c
@@ -268,7 +268,11 @@ int snd_register_device(int type, struct snd_card *card, int dev,
268 snd_minors[minor] = preg; 268 snd_minors[minor] = preg;
269 if (card) 269 if (card)
270 device = card->dev; 270 device = card->dev;
271 class_device_create(sound_class, NULL, MKDEV(major, minor), device, "%s", name); 271 preg->class_dev = class_device_create(sound_class, NULL,
272 MKDEV(major, minor),
273 device, "%s", name);
274 if (preg->class_dev)
275 class_set_devdata(preg->class_dev, private_data);
272 276
273 mutex_unlock(&sound_mutex); 277 mutex_unlock(&sound_mutex);
274 return 0; 278 return 0;
@@ -276,6 +280,24 @@ int snd_register_device(int type, struct snd_card *card, int dev,
276 280
277EXPORT_SYMBOL(snd_register_device); 281EXPORT_SYMBOL(snd_register_device);
278 282
283/* find the matching minor record
284 * return the index of snd_minor, or -1 if not found
285 */
286static int find_snd_minor(int type, struct snd_card *card, int dev)
287{
288 int cardnum, minor;
289 struct snd_minor *mptr;
290
291 cardnum = card ? card->number : -1;
292 for (minor = 0; minor < ARRAY_SIZE(snd_minors); ++minor)
293 if ((mptr = snd_minors[minor]) != NULL &&
294 mptr->type == type &&
295 mptr->card == cardnum &&
296 mptr->device == dev)
297 return minor;
298 return -1;
299}
300
279/** 301/**
280 * snd_unregister_device - unregister the device on the given card 302 * snd_unregister_device - unregister the device on the given card
281 * @type: the device type, SNDRV_DEVICE_TYPE_XXX 303 * @type: the device type, SNDRV_DEVICE_TYPE_XXX
@@ -289,32 +311,42 @@ EXPORT_SYMBOL(snd_register_device);
289 */ 311 */
290int snd_unregister_device(int type, struct snd_card *card, int dev) 312int snd_unregister_device(int type, struct snd_card *card, int dev)
291{ 313{
292 int cardnum, minor; 314 int minor;
293 struct snd_minor *mptr;
294 315
295 cardnum = card ? card->number : -1;
296 mutex_lock(&sound_mutex); 316 mutex_lock(&sound_mutex);
297 for (minor = 0; minor < ARRAY_SIZE(snd_minors); ++minor) 317 minor = find_snd_minor(type, card, dev);
298 if ((mptr = snd_minors[minor]) != NULL && 318 if (minor < 0) {
299 mptr->type == type &&
300 mptr->card == cardnum &&
301 mptr->device == dev)
302 break;
303 if (minor == ARRAY_SIZE(snd_minors)) {
304 mutex_unlock(&sound_mutex); 319 mutex_unlock(&sound_mutex);
305 return -EINVAL; 320 return -EINVAL;
306 } 321 }
307 322
308 class_device_destroy(sound_class, MKDEV(major, minor)); 323 class_device_destroy(sound_class, MKDEV(major, minor));
309 324
325 kfree(snd_minors[minor]);
310 snd_minors[minor] = NULL; 326 snd_minors[minor] = NULL;
311 mutex_unlock(&sound_mutex); 327 mutex_unlock(&sound_mutex);
312 kfree(mptr);
313 return 0; 328 return 0;
314} 329}
315 330
316EXPORT_SYMBOL(snd_unregister_device); 331EXPORT_SYMBOL(snd_unregister_device);
317 332
333int snd_add_device_sysfs_file(int type, struct snd_card *card, int dev,
334 const struct class_device_attribute *attr)
335{
336 int minor, ret = -EINVAL;
337 struct class_device *cdev;
338
339 mutex_lock(&sound_mutex);
340 minor = find_snd_minor(type, card, dev);
341 if (minor >= 0 && (cdev = snd_minors[minor]->class_dev) != NULL)
342 ret = class_device_create_file(cdev, attr);
343 mutex_unlock(&sound_mutex);
344 return ret;
345
346}
347
348EXPORT_SYMBOL(snd_add_device_sysfs_file);
349
318#ifdef CONFIG_PROC_FS 350#ifdef CONFIG_PROC_FS
319/* 351/*
320 * INFO PART 352 * INFO PART