ALSA: hda - Avoid unnecessary power-up at mixer amp changes
authorTakashi Iwai <tiwai@suse.de>
Fri, 20 Feb 2015 09:26:25 +0000 (10:26 +0100)
committerTakashi Iwai <tiwai@suse.de>
Fri, 20 Feb 2015 09:37:48 +0000 (10:37 +0100)
When the mixer amp is touched by control elements, we don't have to
power up always; if the codec was suspended at the time, we can just
update the amp cache and it's reflected to the hardware upon resume.

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

index 0734cb35887d7a5d62aff6f576f362284ea7b26c..5e755eb6f19a0b79e1969af24eb62dd3f34902a8 100644 (file)
@@ -2148,11 +2148,10 @@ EXPORT_SYMBOL_GPL(snd_hda_codec_amp_read);
 
 static int codec_amp_update(struct hda_codec *codec, hda_nid_t nid, int ch,
                            int direction, int idx, int mask, int val,
-                           bool init_only)
+                           bool init_only, bool cache_only)
 {
        struct hda_amp_info *info;
        unsigned int caps;
-       unsigned int cache_only;
 
        if (snd_BUG_ON(mask & ~0xff))
                mask &= 0xff;
@@ -2170,7 +2169,7 @@ static int codec_amp_update(struct hda_codec *codec, hda_nid_t nid, int ch,
                return 0;
        }
        info->vol[ch] = val;
-       cache_only = info->head.dirty = codec->cached_write;
+       info->head.dirty |= cache_only;
        caps = info->amp_caps;
        mutex_unlock(&codec->hash_mutex);
        if (!cache_only)
@@ -2194,7 +2193,8 @@ static int codec_amp_update(struct hda_codec *codec, hda_nid_t nid, int ch,
 int snd_hda_codec_amp_update(struct hda_codec *codec, hda_nid_t nid, int ch,
                             int direction, int idx, int mask, int val)
 {
-       return codec_amp_update(codec, nid, ch, direction, idx, mask, val, false);
+       return codec_amp_update(codec, nid, ch, direction, idx, mask, val,
+                               false, codec->cached_write);
 }
 EXPORT_SYMBOL_GPL(snd_hda_codec_amp_update);
 
@@ -2241,7 +2241,8 @@ EXPORT_SYMBOL_GPL(snd_hda_codec_amp_stereo);
 int snd_hda_codec_amp_init(struct hda_codec *codec, hda_nid_t nid, int ch,
                           int dir, int idx, int mask, int val)
 {
-       return codec_amp_update(codec, nid, ch, dir, idx, mask, val, true);
+       return codec_amp_update(codec, nid, ch, dir, idx, mask, val, true,
+                               codec->cached_write);
 }
 EXPORT_SYMBOL_GPL(snd_hda_codec_amp_init);
 
@@ -2383,8 +2384,8 @@ update_amp_value(struct hda_codec *codec, hda_nid_t nid,
        maxval = get_amp_max_value(codec, nid, dir, 0);
        if (val > maxval)
                val = maxval;
-       return snd_hda_codec_amp_update(codec, nid, ch, dir, idx,
-                                       HDA_AMP_VOLMASK, val);
+       return codec_amp_update(codec, nid, ch, dir, idx, HDA_AMP_VOLMASK, val,
+                               false, !hda_codec_is_power_on(codec));
 }
 
 /**
@@ -2434,14 +2435,12 @@ int snd_hda_mixer_amp_volume_put(struct snd_kcontrol *kcontrol,
        long *valp = ucontrol->value.integer.value;
        int change = 0;
 
-       snd_hda_power_up(codec);
        if (chs & 1) {
                change = update_amp_value(codec, nid, 0, dir, idx, ofs, *valp);
                valp++;
        }
        if (chs & 2)
                change |= update_amp_value(codec, nid, 1, dir, idx, ofs, *valp);
-       snd_hda_power_down(codec);
        return change;
 }
 EXPORT_SYMBOL_GPL(snd_hda_mixer_amp_volume_put);
@@ -3109,19 +3108,19 @@ int snd_hda_mixer_amp_switch_put(struct snd_kcontrol *kcontrol,
        long *valp = ucontrol->value.integer.value;
        int change = 0;
 
-       snd_hda_power_up(codec);
        if (chs & 1) {
-               change = snd_hda_codec_amp_update(codec, nid, 0, dir, idx,
-                                                 HDA_AMP_MUTE,
-                                                 *valp ? 0 : HDA_AMP_MUTE);
+               change = codec_amp_update(codec, nid, 0, dir, idx,
+                                         HDA_AMP_MUTE,
+                                         *valp ? 0 : HDA_AMP_MUTE, false,
+                                         !hda_codec_is_power_on(codec));
                valp++;
        }
        if (chs & 2)
-               change |= snd_hda_codec_amp_update(codec, nid, 1, dir, idx,
-                                                  HDA_AMP_MUTE,
-                                                  *valp ? 0 : HDA_AMP_MUTE);
+               change |= codec_amp_update(codec, nid, 1, dir, idx,
+                                          HDA_AMP_MUTE,
+                                          *valp ? 0 : HDA_AMP_MUTE, false,
+                                          !hda_codec_is_power_on(codec));
        hda_call_check_power_status(codec, nid);
-       snd_hda_power_down(codec);
        return change;
 }
 EXPORT_SYMBOL_GPL(snd_hda_mixer_amp_switch_put);