]> git.ipfire.org Git - people/teissler/ipfire-2.x.git/blobdiff - src/patches/suse-2.6.27.31/patches.drivers/alsa-hda-beep-dig-switch
Move xen patchset to new version's subdir.
[people/teissler/ipfire-2.x.git] / src / patches / suse-2.6.27.31 / patches.drivers / alsa-hda-beep-dig-switch
diff --git a/src/patches/suse-2.6.27.31/patches.drivers/alsa-hda-beep-dig-switch b/src/patches/suse-2.6.27.31/patches.drivers/alsa-hda-beep-dig-switch
new file mode 100644 (file)
index 0000000..e4c0b40
--- /dev/null
@@ -0,0 +1,168 @@
+From: Takashi Iwai <tiwai@suse.de>
+Subject: ALSA: hda - Add digital beep playback switch for STAC/IDT codecs
+Patch-mainline: 
+References: #444572
+
+The digital beep widget may have no mute control, and always enabling
+the beep is ofen pretty annoying, especially on laptops.
+
+This patch adds a mixer control "PC Beep Playback Switch" when there
+is no mixer amp mute is found, and controls it on software.
+
+Reference: Novell bnc#444572
+       https://bugzilla.novell.com/show_bug.cgi?id=444572
+
+Signed-off-by: Takashi Iwai <tiwai@suse.de>
+
+---
+ sound/pci/hda/hda_beep.c       |    4 ++
+ sound/pci/hda/hda_beep.h       |    1 
+ sound/pci/hda/patch_sigmatel.c |   69 +++++++++++++++++++++++++++++++++++++----
+ 3 files changed, 68 insertions(+), 6 deletions(-)
+
+--- a/sound/pci/hda/hda_beep.c
++++ b/sound/pci/hda/hda_beep.c
+@@ -37,6 +37,9 @@ static void snd_hda_generate_beep(struct
+               container_of(work, struct hda_beep, beep_work);
+       struct hda_codec *codec = beep->codec;
++      if (!beep->enabled)
++              return;
++
+       /* generate tone */
+       snd_hda_codec_write_cache(codec, beep->nid, 0,
+                       AC_VERB_SET_BEEP_CONTROL, beep->tone);
+@@ -115,6 +118,7 @@ int snd_hda_attach_beep_device(struct hd
+       beep->nid = nid;
+       beep->dev = input_dev;
+       beep->codec = codec;
++      beep->enabled = 1;
+       codec->beep = beep;
+       INIT_WORK(&beep->beep_work, &snd_hda_generate_beep);
+--- a/sound/pci/hda/hda_beep.h
++++ b/sound/pci/hda/hda_beep.h
+@@ -31,6 +31,7 @@ struct hda_beep {
+       char phys[32];
+       int tone;
+       int nid;
++      int enabled;
+       struct work_struct beep_work; /* scheduled task for beep event */
+ };
+--- a/sound/pci/hda/patch_sigmatel.c
++++ b/sound/pci/hda/patch_sigmatel.c
+@@ -2592,8 +2592,10 @@ static struct snd_kcontrol_new stac92xx_
+ };
+ /* add dynamic controls */
+-static int stac92xx_add_control_idx(struct sigmatel_spec *spec, int type,
+-              int idx, const char *name, unsigned long val)
++static int stac92xx_add_control_temp(struct sigmatel_spec *spec,
++                                   struct snd_kcontrol_new *ktemp,
++                                   int idx, const char *name,
++                                   unsigned long val)
+ {
+       struct snd_kcontrol_new *knew;
+@@ -2612,20 +2614,29 @@ static int stac92xx_add_control_idx(stru
+       }
+       knew = &spec->kctl_alloc[spec->num_kctl_used];
+-      *knew = stac92xx_control_templates[type];
++      *knew = *ktemp;
+       knew->index = idx;
+       knew->name = kstrdup(name, GFP_KERNEL);
+-      if (! knew->name)
++      if (!knew->name)
+               return -ENOMEM;
+       knew->private_value = val;
+       spec->num_kctl_used++;
+       return 0;
+ }
++static inline int stac92xx_add_control_idx(struct sigmatel_spec *spec,
++                                         int type, int idx, const char *name,
++                                         unsigned long val)
++{
++      return stac92xx_add_control_temp(spec,
++                                       &stac92xx_control_templates[type],
++                                       idx, name, val);
++}
++
+ /* add dynamic controls */
+-static int stac92xx_add_control(struct sigmatel_spec *spec, int type,
+-              const char *name, unsigned long val)
++static inline int stac92xx_add_control(struct sigmatel_spec *spec, int type,
++                                     const char *name, unsigned long val)
+ {
+       return stac92xx_add_control_idx(spec, type, 0, name, val);
+ }
+@@ -3067,6 +3078,43 @@ static int stac92xx_auto_create_beep_ctl
+       return 0;
+ }
++#ifdef CONFIG_SND_HDA_INPUT_BEEP
++#define stac92xx_dig_beep_switch_info snd_ctl_boolean_mono_info
++
++static int stac92xx_dig_beep_switch_get(struct snd_kcontrol *kcontrol,
++                                      struct snd_ctl_elem_value *ucontrol)
++{
++      struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
++      ucontrol->value.integer.value[0] = codec->beep->enabled;
++      return 0;
++}
++
++static int stac92xx_dig_beep_switch_put(struct snd_kcontrol *kcontrol,
++                                      struct snd_ctl_elem_value *ucontrol)
++{
++      struct hda_codec *codec = snd_kcontrol_chip(kcontrol);
++      int enabled = !!ucontrol->value.integer.value[0];
++      if (codec->beep->enabled != enabled) {
++              codec->beep->enabled = enabled;
++              return 1;
++      }
++      return 0;
++}
++
++static struct snd_kcontrol_new stac92xx_dig_beep_ctrl = {
++      .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
++      .info = stac92xx_dig_beep_switch_info,
++      .get = stac92xx_dig_beep_switch_get,
++      .put = stac92xx_dig_beep_switch_put,
++};
++
++static int stac92xx_beep_switch_ctl(struct hda_codec *codec)
++{
++      return stac92xx_add_control_temp(codec->spec, &stac92xx_dig_beep_ctrl,
++                                       0, "PC Beep Playback Switch", 0);
++}
++#endif
++
+ static int stac92xx_auto_create_mux_input_ctls(struct hda_codec *codec)
+ {
+       struct sigmatel_spec *spec = codec->spec;
+@@ -3373,6 +3421,7 @@ static int stac92xx_parse_auto_config(st
+ #ifdef CONFIG_SND_HDA_INPUT_BEEP
+       if (spec->digbeep_nid > 0) {
+               hda_nid_t nid = spec->digbeep_nid;
++              unsigned int caps;
+               err = stac92xx_auto_create_beep_ctls(codec, nid);
+               if (err < 0)
+@@ -3380,6 +3429,14 @@ static int stac92xx_parse_auto_config(st
+               err = snd_hda_attach_beep_device(codec, nid);
+               if (err < 0)
+                       return err;
++              /* if no beep switch is available, make its own one */
++              caps = query_amp_caps(codec, nid, HDA_OUTPUT);
++              if (codec->beep &&
++                  !((caps & AC_AMPCAP_MUTE) >> AC_AMPCAP_MUTE_SHIFT)) {
++                      err = stac92xx_beep_switch_ctl(codec);
++                      if (err < 0)
++                              return err;
++              }
+       }
+ #endif