ALSA: hda - Fix power-saving during playing beep sound
authorTakashi Iwai <tiwai@suse.de>
Mon, 18 Mar 2013 10:29:56 +0000 (11:29 +0100)
committerTakashi Iwai <tiwai@suse.de>
Mon, 18 Mar 2013 11:58:47 +0000 (12:58 +0100)
While playing the digital beep tone, the codec shouldn't be turned
off.  This patch adds proper snd_hda_power_up()/down() calls at each
time when the beep is played or off.

Also, this fixes automatically an unnecessary codec power-up at
detaching the beep device when the beep isn't being played.

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

index 0849aac449f2043fe1df237c48773436b5aefe7d..35bf5316ccc69cb38bf9916656c068d21fd71470 100644 (file)
@@ -39,13 +39,23 @@ static void snd_hda_generate_beep(struct work_struct *work)
        struct hda_beep *beep =
                container_of(work, struct hda_beep, beep_work);
        struct hda_codec *codec = beep->codec;
+       int tone;
 
        if (!beep->enabled)
                return;
 
+       tone = beep->tone;
+       if (tone && !beep->playing) {
+               snd_hda_power_up(codec);
+               beep->playing = 1;
+       }
        /* generate tone */
        snd_hda_codec_write(codec, beep->nid, 0,
-                       AC_VERB_SET_BEEP_CONTROL, beep->tone);
+                           AC_VERB_SET_BEEP_CONTROL, tone);
+       if (!tone && beep->playing) {
+               beep->playing = 0;
+               snd_hda_power_down(codec);
+       }
 }
 
 /* (non-standard) Linear beep tone calculation for IDT/STAC codecs 
@@ -115,14 +125,23 @@ static int snd_hda_beep_event(struct input_dev *dev, unsigned int type,
        return 0;
 }
 
+static void turn_off_beep(struct hda_beep *beep)
+{
+       cancel_work_sync(&beep->beep_work);
+       if (beep->playing) {
+               /* turn off beep */
+               snd_hda_codec_write(beep->codec, beep->nid, 0,
+                                   AC_VERB_SET_BEEP_CONTROL, 0);
+               beep->playing = 0;
+               snd_hda_power_down(beep->codec);
+       }
+}
+
 static void snd_hda_do_detach(struct hda_beep *beep)
 {
        input_unregister_device(beep->dev);
        beep->dev = NULL;
-       cancel_work_sync(&beep->beep_work);
-       /* turn off beep for sure */
-       snd_hda_codec_write(beep->codec, beep->nid, 0,
-                                 AC_VERB_SET_BEEP_CONTROL, 0);
+       turn_off_beep(beep);
 }
 
 static int snd_hda_do_attach(struct hda_beep *beep)
@@ -170,12 +189,8 @@ int snd_hda_enable_beep_device(struct hda_codec *codec, int enable)
        enable = !!enable;
        if (beep->enabled != enable) {
                beep->enabled = enable;
-               if (!enable) {
-                       cancel_work_sync(&beep->beep_work);
-                       /* turn off beep */
-                       snd_hda_codec_write(beep->codec, beep->nid, 0,
-                                                 AC_VERB_SET_BEEP_CONTROL, 0);
-               }
+               if (!enable)
+                       turn_off_beep(beep);
                return 1;
        }
        return 0;
index 4dc6933bc655bf2e1572d35bb958ad7c76f7482f..cb88464676b656e109bd452ffbe1a94114cdcdd4 100644 (file)
@@ -36,6 +36,7 @@ struct hda_beep {
        hda_nid_t nid;
        unsigned int enabled:1;
        unsigned int linear_tone:1;     /* linear tone for IDT/STAC codec */
+       unsigned int playing:1;
        struct work_struct beep_work; /* scheduled task for beep event */
        struct mutex mutex;
 };