ALSA: hda - Add hints for reconfig
authorTakashi Iwai <tiwai@suse.de>
Wed, 30 Jul 2008 13:01:46 +0000 (15:01 +0200)
committerTakashi Iwai <tiwai@suse.de>
Mon, 13 Oct 2008 00:43:04 +0000 (02:43 +0200)
This patch adds the "hints" for reconfiguring codecs.  The hints
are simply string arrays and can be freely used/parsed by the codec
patch.  The hints can be input via hwdep sysfs files.

Signed-off-by: Takashi Iwai <tiwai@suse.de>
sound/pci/hda/hda_codec.h
sound/pci/hda/hda_hwdep.c

index 38a9bb6bafb0f8db9c0d284c853a4c8b55d97fd9..a77ba223af4073838c4d50c6a48df259441ee67c 100644 (file)
@@ -754,6 +754,7 @@ struct hda_codec {
 #ifdef CONFIG_SND_HDA_HWDEP
        struct snd_hwdep *hwdep;        /* assigned hwdep device */
        struct snd_array init_verbs;    /* additional init verbs */
+       struct snd_array hints;         /* additional hints */
 #endif
 
        /* misc flags */
index f3400d75eba4b57ffc47e31c15cfb96ce77e6322..653da1d3e4dfb6018f489d90ec1226e5988fecfb 100644 (file)
@@ -23,6 +23,7 @@
 #include <linux/pci.h>
 #include <linux/compat.h>
 #include <linux/mutex.h>
+#include <linux/ctype.h>
 #include <sound/core.h>
 #include "hda_codec.h"
 #include "hda_local.h"
@@ -98,8 +99,16 @@ static int hda_hwdep_open(struct snd_hwdep *hw, struct file *file)
 
 static void clear_hwdep_elements(struct hda_codec *codec)
 {
+       char **head;
+       int i;
+
        /* clear init verbs */
        snd_array_free(&codec->init_verbs);
+       /* clear hints */
+       head = codec->hints.list;
+       for (i = 0; i < codec->hints.used; i++, head++)
+               kfree(*head);
+       snd_array_free(&codec->hints);
 }
 
 static void hwdep_free(struct snd_hwdep *hwdep)
@@ -131,6 +140,7 @@ int __devinit snd_hda_create_hwdep(struct hda_codec *codec)
 #endif
 
        snd_array_init(&codec->init_verbs, sizeof(struct hda_verb), 32);
+       snd_array_init(&codec->hints, sizeof(char *), 32);
 
        return 0;
 }
@@ -280,6 +290,29 @@ static ssize_t init_verbs_store(struct device *dev,
        return count;
 }
 
+static ssize_t hints_store(struct device *dev,
+                          struct device_attribute *attr,
+                          const char *buf, size_t count)
+{
+       struct snd_hwdep *hwdep = dev_get_drvdata(dev);
+       struct hda_codec *codec = hwdep->private_data;
+       char *p;
+       char **hint;
+
+       if (!*buf || isspace(*buf) || *buf == '#' || *buf == '\n')
+               return count;
+       p = kstrndup_noeol(buf, 1024);
+       if (!p)
+               return -ENOMEM;
+       hint = snd_array_new(&codec->hints);
+       if (!hint) {
+               kfree(p);
+               return -ENOMEM;
+       }
+       *hint = p;
+       return count;
+}
+
 #define CODEC_ATTR_RW(type) \
        __ATTR(type, 0644, type##_show, type##_store)
 #define CODEC_ATTR_RO(type) \
@@ -296,6 +329,7 @@ static struct device_attribute codec_attrs[] = {
        CODEC_ATTR_RW(name),
        CODEC_ATTR_RW(modelname),
        CODEC_ATTR_WO(init_verbs),
+       CODEC_ATTR_WO(hints),
        CODEC_ATTR_WO(reconfig),
        CODEC_ATTR_WO(clear),
 };