Skip to content

Commit

Permalink
ASoC: Refactor DAPM suspend handling
Browse files Browse the repository at this point in the history
Instead of using stream events to handle power down during suspend
integrate the handling with the normal widget path checking by
replacing all cases where we report a connected endpoint in a path
with a function snd_soc_dapm_suspend_check() which looks at the ALSA
power state for the card and reports false if we are in a D3 state.

Since the core moves us into D3 prior to initating the suspend all
power checks during suspend will cause the widgets to be powered
down. In order to ensure that widgets are powered up on resume set
the card to D2 at the start of resume handling (ALSA API calls
require D0 so we are still protected against userspace access).

Tested-by: Peter Ujfalusi <peter.ujfalusi@nokia.com>
Acked-by: Liam Girdwood <lrg@slimlogic.co.uk>
Signed-off-by: Mark Brown <broonie@opensource.wolfsonmicro.com>
  • Loading branch information
Mark Brown committed May 10, 2010
1 parent 50ae838 commit 9949788
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 23 deletions.
3 changes: 3 additions & 0 deletions sound/soc/soc-core.c
Original file line number Diff line number Diff line change
Expand Up @@ -963,6 +963,9 @@ static void soc_resume_deferred(struct work_struct *work)

dev_dbg(socdev->dev, "starting resume work\n");

/* Bring us up into D2 so that DAPM starts enabling things */
snd_power_change_state(codec->card, SNDRV_CTL_POWER_D2);

if (card->resume_pre)
card->resume_pre(pdev);

Expand Down
53 changes: 30 additions & 23 deletions sound/soc/soc-dapm.c
Original file line number Diff line number Diff line change
Expand Up @@ -430,6 +430,23 @@ static inline void dapm_clear_walk(struct snd_soc_codec *codec)
p->walked = 0;
}

/* We implement power down on suspend by checking the power state of
* the ALSA card - when we are suspending the ALSA state for the card
* is set to D3.
*/
static int snd_soc_dapm_suspend_check(struct snd_soc_dapm_widget *widget)
{
struct snd_soc_codec *codec = widget->codec;

switch (snd_power_get_state(codec->card)) {
case SNDRV_CTL_POWER_D3hot:
case SNDRV_CTL_POWER_D3cold:
return 0;
default:
return 1;
}
}

/*
* Recursively check for a completed path to an active or physically connected
* output widget. Returns number of complete paths.
Expand All @@ -446,20 +463,20 @@ static int is_connected_output_ep(struct snd_soc_dapm_widget *widget)
case snd_soc_dapm_adc:
case snd_soc_dapm_aif_out:
if (widget->active)
return 1;
return snd_soc_dapm_suspend_check(widget);
default:
break;
}

if (widget->connected) {
/* connected pin ? */
if (widget->id == snd_soc_dapm_output && !widget->ext)
return 1;
return snd_soc_dapm_suspend_check(widget);

/* connected jack or spk ? */
if (widget->id == snd_soc_dapm_hp || widget->id == snd_soc_dapm_spk ||
(widget->id == snd_soc_dapm_line && !list_empty(&widget->sources)))
return 1;
return snd_soc_dapm_suspend_check(widget);
}

list_for_each_entry(path, &widget->sinks, list_source) {
Expand Down Expand Up @@ -492,24 +509,24 @@ static int is_connected_input_ep(struct snd_soc_dapm_widget *widget)
case snd_soc_dapm_dac:
case snd_soc_dapm_aif_in:
if (widget->active)
return 1;
return snd_soc_dapm_suspend_check(widget);
default:
break;
}

if (widget->connected) {
/* connected pin ? */
if (widget->id == snd_soc_dapm_input && !widget->ext)
return 1;
return snd_soc_dapm_suspend_check(widget);

/* connected VMID/Bias for lower pops */
if (widget->id == snd_soc_dapm_vmid)
return 1;
return snd_soc_dapm_suspend_check(widget);

/* connected jack ? */
if (widget->id == snd_soc_dapm_mic ||
(widget->id == snd_soc_dapm_line && !list_empty(&widget->sinks)))
return 1;
return snd_soc_dapm_suspend_check(widget);
}

list_for_each_entry(path, &widget->sources, list_sink) {
Expand Down Expand Up @@ -897,22 +914,12 @@ static int dapm_power_widgets(struct snd_soc_codec *codec, int event)
if (!w->power_check)
continue;

/* If we're suspending then pull down all the
* power. */
switch (event) {
case SND_SOC_DAPM_STREAM_SUSPEND:
power = 0;
break;

default:
if (!w->force)
power = w->power_check(w);
else
power = 1;
if (power)
sys_power = 1;
break;
}
if (!w->force)
power = w->power_check(w);
else
power = 1;
if (power)
sys_power = 1;

if (w->power == power)
continue;
Expand Down

0 comments on commit 9949788

Please sign in to comment.