aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorCharles Keepax <ckeepax@opensource.wolfsonmicro.com>2015-05-01 07:37:26 -0400
committerMark Brown <broonie@kernel.org>2015-05-06 12:13:01 -0400
commit561ed680b764b288feeb74a24e1d9fb3da98ec7b (patch)
treedf2de90bd178c0873e526b4ea985ee97a17a29da
parent773da9b358bfbef1b7a862425fea0d9d9d3443f8 (diff)
ASoC: dapm: Add support for autodisable mux controls
Commit 57295073b6ac ("ASoC: dapm: Implement mixer input auto-disable") added support for autodisable controls, controls whose values are only written to the hardware when their respective widgets are powered up. But it only added support for controls based on the mixer abstraction. This patch add support for mux controls (DAPM controls based on the enum abstraction) to be auto-disabled as well. As each mux can only have a single control, there is no need to tie the autodisable widget to the inputs (as is done for the mixer controls) it can be tided directly to the mux widget itself. Note that it is assumed that the first entry in a autodisable mux control will always represent the off state for the mux and is what the mux will be set to whilst it is disabled. Signed-off-by: Charles Keepax <ckeepax@opensource.wolfsonmicro.com> Reviewed-by: Lars-Peter Clausen <lars@metafoo.de> Tested-by: Lars-Peter Clausen <lars@metafoo.de> Signed-off-by: Mark Brown <broonie@kernel.org>
-rw-r--r--include/sound/soc.h10
-rw-r--r--sound/soc/soc-dapm.c78
2 files changed, 69 insertions, 19 deletions
diff --git a/include/sound/soc.h b/include/sound/soc.h
index b257a09a98d1..2f2e59e1513e 100644
--- a/include/sound/soc.h
+++ b/include/sound/soc.h
@@ -192,6 +192,10 @@
192 .mask = xmask, .items = xitems, .texts = xtexts, .values = xvalues} 192 .mask = xmask, .items = xitems, .texts = xtexts, .values = xvalues}
193#define SOC_VALUE_ENUM_SINGLE(xreg, xshift, xmask, xitems, xtexts, xvalues) \ 193#define SOC_VALUE_ENUM_SINGLE(xreg, xshift, xmask, xitems, xtexts, xvalues) \
194 SOC_VALUE_ENUM_DOUBLE(xreg, xshift, xshift, xmask, xitems, xtexts, xvalues) 194 SOC_VALUE_ENUM_DOUBLE(xreg, xshift, xshift, xmask, xitems, xtexts, xvalues)
195#define SOC_VALUE_ENUM_SINGLE_AUTODISABLE(xreg, xshift, xmask, xitems, xtexts, xvalues) \
196{ .reg = xreg, .shift_l = xshift, .shift_r = xshift, \
197 .mask = xmask, .items = xitems, .texts = xtexts, \
198 .values = xvalues, .autodisable = 1}
195#define SOC_ENUM_SINGLE_VIRT(xitems, xtexts) \ 199#define SOC_ENUM_SINGLE_VIRT(xitems, xtexts) \
196 SOC_ENUM_SINGLE(SND_SOC_NOPM, 0, xitems, xtexts) 200 SOC_ENUM_SINGLE(SND_SOC_NOPM, 0, xitems, xtexts)
197#define SOC_ENUM(xname, xenum) \ 201#define SOC_ENUM(xname, xenum) \
@@ -312,6 +316,11 @@
312 ARRAY_SIZE(xtexts), xtexts, xvalues) 316 ARRAY_SIZE(xtexts), xtexts, xvalues)
313#define SOC_VALUE_ENUM_SINGLE_DECL(name, xreg, xshift, xmask, xtexts, xvalues) \ 317#define SOC_VALUE_ENUM_SINGLE_DECL(name, xreg, xshift, xmask, xtexts, xvalues) \
314 SOC_VALUE_ENUM_DOUBLE_DECL(name, xreg, xshift, xshift, xmask, xtexts, xvalues) 318 SOC_VALUE_ENUM_DOUBLE_DECL(name, xreg, xshift, xshift, xmask, xtexts, xvalues)
319
320#define SOC_VALUE_ENUM_SINGLE_AUTODISABLE_DECL(name, xreg, xshift, xmask, xtexts, xvalues) \
321 const struct soc_enum name = SOC_VALUE_ENUM_SINGLE_AUTODISABLE(xreg, \
322 xshift, xmask, ARRAY_SIZE(xtexts), xtexts, xvalues)
323
315#define SOC_ENUM_SINGLE_VIRT_DECL(name, xtexts) \ 324#define SOC_ENUM_SINGLE_VIRT_DECL(name, xtexts) \
316 const struct soc_enum name = SOC_ENUM_SINGLE_VIRT(ARRAY_SIZE(xtexts), xtexts) 325 const struct soc_enum name = SOC_ENUM_SINGLE_VIRT(ARRAY_SIZE(xtexts), xtexts)
317 326
@@ -1188,6 +1197,7 @@ struct soc_enum {
1188 unsigned int mask; 1197 unsigned int mask;
1189 const char * const *texts; 1198 const char * const *texts;
1190 const unsigned int *values; 1199 const unsigned int *values;
1200 unsigned int autodisable:1;
1191}; 1201};
1192 1202
1193/** 1203/**
diff --git a/sound/soc/soc-dapm.c b/sound/soc/soc-dapm.c
index a0d97f89eb75..79e6cf4b7de1 100644
--- a/sound/soc/soc-dapm.c
+++ b/sound/soc/soc-dapm.c
@@ -308,6 +308,7 @@ static int dapm_kcontrol_data_alloc(struct snd_soc_dapm_widget *widget,
308{ 308{
309 struct dapm_kcontrol_data *data; 309 struct dapm_kcontrol_data *data;
310 struct soc_mixer_control *mc; 310 struct soc_mixer_control *mc;
311 struct soc_enum *e;
311 const char *name; 312 const char *name;
312 int ret; 313 int ret;
313 314
@@ -355,6 +356,41 @@ static int dapm_kcontrol_data_alloc(struct snd_soc_dapm_widget *widget,
355 } 356 }
356 } 357 }
357 break; 358 break;
359 case snd_soc_dapm_mux:
360 e = (struct soc_enum *)kcontrol->private_value;
361
362 if (e->autodisable) {
363 struct snd_soc_dapm_widget template;
364
365 name = kasprintf(GFP_KERNEL, "%s %s", kcontrol->id.name,
366 "Autodisable");
367 if (!name) {
368 ret = -ENOMEM;
369 goto err_data;
370 }
371
372 memset(&template, 0, sizeof(template));
373 template.reg = e->reg;
374 template.mask = e->mask << e->shift_l;
375 template.shift = e->shift_l;
376 template.off_val = snd_soc_enum_item_to_val(e, 0);
377 template.on_val = template.off_val;
378 template.id = snd_soc_dapm_kcontrol;
379 template.name = name;
380
381 data->value = template.on_val;
382
383 data->widget = snd_soc_dapm_new_control(widget->dapm,
384 &template);
385 if (!data->widget) {
386 ret = -ENOMEM;
387 goto err_name;
388 }
389
390 snd_soc_dapm_add_path(widget->dapm, data->widget,
391 widget, NULL, NULL);
392 }
393 break;
358 default: 394 default:
359 break; 395 break;
360 } 396 }
@@ -418,11 +454,6 @@ static void dapm_kcontrol_add_path(const struct snd_kcontrol *kcontrol,
418 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol); 454 struct dapm_kcontrol_data *data = snd_kcontrol_chip(kcontrol);
419 455
420 list_add_tail(&path->list_kcontrol, &data->paths); 456 list_add_tail(&path->list_kcontrol, &data->paths);
421
422 if (data->widget) {
423 snd_soc_dapm_add_path(data->widget->dapm, data->widget,
424 path->source, NULL, NULL);
425 }
426} 457}
427 458
428static bool dapm_kcontrol_is_powered(const struct snd_kcontrol *kcontrol) 459static bool dapm_kcontrol_is_powered(const struct snd_kcontrol *kcontrol)
@@ -820,6 +851,7 @@ static int dapm_new_mixer(struct snd_soc_dapm_widget *w)
820{ 851{
821 int i, ret; 852 int i, ret;
822 struct snd_soc_dapm_path *path; 853 struct snd_soc_dapm_path *path;
854 struct dapm_kcontrol_data *data;
823 855
824 /* add kcontrol */ 856 /* add kcontrol */
825 for (i = 0; i < w->num_kcontrols; i++) { 857 for (i = 0; i < w->num_kcontrols; i++) {
@@ -829,16 +861,20 @@ static int dapm_new_mixer(struct snd_soc_dapm_widget *w)
829 if (path->name != (char *)w->kcontrol_news[i].name) 861 if (path->name != (char *)w->kcontrol_news[i].name)
830 continue; 862 continue;
831 863
832 if (w->kcontrols[i]) { 864 if (!w->kcontrols[i]) {
833 dapm_kcontrol_add_path(w->kcontrols[i], path); 865 ret = dapm_create_or_share_mixmux_kcontrol(w, i);
834 continue; 866 if (ret < 0)
867 return ret;
835 } 868 }
836 869
837 ret = dapm_create_or_share_mixmux_kcontrol(w, i);
838 if (ret < 0)
839 return ret;
840
841 dapm_kcontrol_add_path(w->kcontrols[i], path); 870 dapm_kcontrol_add_path(w->kcontrols[i], path);
871
872 data = snd_kcontrol_chip(w->kcontrols[i]);
873 if (data->widget)
874 snd_soc_dapm_add_path(data->widget->dapm,
875 data->widget,
876 path->source,
877 NULL, NULL);
842 } 878 }
843 } 879 }
844 880
@@ -2945,16 +2981,19 @@ int snd_soc_dapm_get_enum_double(struct snd_kcontrol *kcontrol,
2945 struct snd_ctl_elem_value *ucontrol) 2981 struct snd_ctl_elem_value *ucontrol)
2946{ 2982{
2947 struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol); 2983 struct snd_soc_dapm_context *dapm = snd_soc_dapm_kcontrol_dapm(kcontrol);
2984 struct snd_soc_card *card = dapm->card;
2948 struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; 2985 struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
2949 unsigned int reg_val, val; 2986 unsigned int reg_val, val;
2950 2987
2951 if (e->reg != SND_SOC_NOPM) { 2988 mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
2989 if (e->reg != SND_SOC_NOPM && dapm_kcontrol_is_powered(kcontrol)) {
2952 int ret = soc_dapm_read(dapm, e->reg, &reg_val); 2990 int ret = soc_dapm_read(dapm, e->reg, &reg_val);
2953 if (ret) 2991 if (ret)
2954 return ret; 2992 return ret;
2955 } else { 2993 } else {
2956 reg_val = dapm_kcontrol_get_value(kcontrol); 2994 reg_val = dapm_kcontrol_get_value(kcontrol);
2957 } 2995 }
2996 mutex_unlock(&card->dapm_mutex);
2958 2997
2959 val = (reg_val >> e->shift_l) & e->mask; 2998 val = (reg_val >> e->shift_l) & e->mask;
2960 ucontrol->value.enumerated.item[0] = snd_soc_enum_val_to_item(e, val); 2999 ucontrol->value.enumerated.item[0] = snd_soc_enum_val_to_item(e, val);
@@ -2984,7 +3023,7 @@ int snd_soc_dapm_put_enum_double(struct snd_kcontrol *kcontrol,
2984 struct snd_soc_card *card = dapm->card; 3023 struct snd_soc_card *card = dapm->card;
2985 struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; 3024 struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
2986 unsigned int *item = ucontrol->value.enumerated.item; 3025 unsigned int *item = ucontrol->value.enumerated.item;
2987 unsigned int val, change; 3026 unsigned int val, change, reg_change = 0;
2988 unsigned int mask; 3027 unsigned int mask;
2989 struct snd_soc_dapm_update update; 3028 struct snd_soc_dapm_update update;
2990 int ret = 0; 3029 int ret = 0;
@@ -3003,19 +3042,20 @@ int snd_soc_dapm_put_enum_double(struct snd_kcontrol *kcontrol,
3003 3042
3004 mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME); 3043 mutex_lock_nested(&card->dapm_mutex, SND_SOC_DAPM_CLASS_RUNTIME);
3005 3044
3045 change = dapm_kcontrol_set_value(kcontrol, val);
3046
3006 if (e->reg != SND_SOC_NOPM) 3047 if (e->reg != SND_SOC_NOPM)
3007 change = soc_dapm_test_bits(dapm, e->reg, mask, val); 3048 reg_change = soc_dapm_test_bits(dapm, e->reg, mask, val);
3008 else
3009 change = dapm_kcontrol_set_value(kcontrol, val);
3010 3049
3011 if (change) { 3050 if (change || reg_change) {
3012 if (e->reg != SND_SOC_NOPM) { 3051 if (reg_change) {
3013 update.kcontrol = kcontrol; 3052 update.kcontrol = kcontrol;
3014 update.reg = e->reg; 3053 update.reg = e->reg;
3015 update.mask = mask; 3054 update.mask = mask;
3016 update.val = val; 3055 update.val = val;
3017 card->update = &update; 3056 card->update = &update;
3018 } 3057 }
3058 change |= reg_change;
3019 3059
3020 ret = soc_dapm_mux_update_power(card, kcontrol, item[0], e); 3060 ret = soc_dapm_mux_update_power(card, kcontrol, item[0], e);
3021 3061