提交 d8a64d6a 编写于 作者: C Charles Keepax 提交者: Mark Brown

ASoC: wm_adsp: Factor out ADSP2 boot proceedure

Move the ADSP2 boot proceedure into a work structure in preparation for
running it asynchronously with the reset of the audio path bring up.
Signed-off-by: NCharles Keepax <ckeepax@opensource.wolfsonmicro.com>
Signed-off-by: NMark Brown <broonie@linaro.org>
上级 5095f55d
...@@ -1492,107 +1492,131 @@ static int wm_adsp2_ena(struct wm_adsp *dsp) ...@@ -1492,107 +1492,131 @@ static int wm_adsp2_ena(struct wm_adsp *dsp)
return 0; return 0;
} }
int wm_adsp2_event(struct snd_soc_dapm_widget *w, void wm_adsp2_boot_work(struct work_struct *work)
struct snd_kcontrol *kcontrol, int event)
{ {
struct snd_soc_codec *codec = w->codec; struct wm_adsp *dsp = container_of(work,
struct wm_adsp *dsps = snd_soc_codec_get_drvdata(codec); struct wm_adsp,
struct wm_adsp *dsp = &dsps[w->shift]; boot_work);
struct wm_adsp_alg_region *alg_region;
struct wm_coeff_ctl *ctl;
unsigned int val;
int ret; int ret;
unsigned int val;
dsp->card = codec->card; /*
* For simplicity set the DSP clock rate to be the
* SYSCLK rate rather than making it configurable.
*/
ret = regmap_read(dsp->regmap, ARIZONA_SYSTEM_CLOCK_1, &val);
if (ret != 0) {
adsp_err(dsp, "Failed to read SYSCLK state: %d\n", ret);
return;
}
val = (val & ARIZONA_SYSCLK_FREQ_MASK)
>> ARIZONA_SYSCLK_FREQ_SHIFT;
switch (event) { ret = regmap_update_bits_async(dsp->regmap,
case SND_SOC_DAPM_POST_PMU: dsp->base + ADSP2_CLOCKING,
/* ADSP2_CLK_SEL_MASK, val);
* For simplicity set the DSP clock rate to be the if (ret != 0) {
* SYSCLK rate rather than making it configurable. adsp_err(dsp, "Failed to set clock rate: %d\n", ret);
*/ return;
ret = regmap_read(dsp->regmap, ARIZONA_SYSTEM_CLOCK_1, &val); }
if (ret != 0) {
adsp_err(dsp, "Failed to read SYSCLK state: %d\n",
ret);
return ret;
}
val = (val & ARIZONA_SYSCLK_FREQ_MASK)
>> ARIZONA_SYSCLK_FREQ_SHIFT;
ret = regmap_update_bits_async(dsp->regmap, if (dsp->dvfs) {
dsp->base + ADSP2_CLOCKING, ret = regmap_read(dsp->regmap,
ADSP2_CLK_SEL_MASK, val); dsp->base + ADSP2_CLOCKING, &val);
if (ret != 0) { if (ret != 0) {
adsp_err(dsp, "Failed to set clock rate: %d\n", dev_err(dsp->dev, "Failed to read clocking: %d\n", ret);
ret); return;
return ret;
} }
if (dsp->dvfs) { if ((val & ADSP2_CLK_SEL_MASK) >= 3) {
ret = regmap_read(dsp->regmap, ret = regulator_enable(dsp->dvfs);
dsp->base + ADSP2_CLOCKING, &val);
if (ret != 0) { if (ret != 0) {
dev_err(dsp->dev, dev_err(dsp->dev,
"Failed to read clocking: %d\n", ret); "Failed to enable supply: %d\n",
return ret; ret);
return;
} }
if ((val & ADSP2_CLK_SEL_MASK) >= 3) { ret = regulator_set_voltage(dsp->dvfs,
ret = regulator_enable(dsp->dvfs); 1800000,
if (ret != 0) { 1800000);
dev_err(dsp->dev, if (ret != 0) {
"Failed to enable supply: %d\n", dev_err(dsp->dev,
ret); "Failed to raise supply: %d\n",
return ret; ret);
} return;
ret = regulator_set_voltage(dsp->dvfs,
1800000,
1800000);
if (ret != 0) {
dev_err(dsp->dev,
"Failed to raise supply: %d\n",
ret);
return ret;
}
} }
} }
}
ret = wm_adsp2_ena(dsp); ret = wm_adsp2_ena(dsp);
if (ret != 0) if (ret != 0)
return ret; return;
ret = wm_adsp_load(dsp); ret = wm_adsp_load(dsp);
if (ret != 0) if (ret != 0)
goto err; goto err;
ret = wm_adsp_setup_algs(dsp); ret = wm_adsp_setup_algs(dsp);
if (ret != 0) if (ret != 0)
goto err; goto err;
ret = wm_adsp_load_coeff(dsp); ret = wm_adsp_load_coeff(dsp);
if (ret != 0) if (ret != 0)
goto err; goto err;
/* Initialize caches for enabled and unset controls */ /* Initialize caches for enabled and unset controls */
ret = wm_coeff_init_control_caches(dsp); ret = wm_coeff_init_control_caches(dsp);
if (ret != 0) if (ret != 0)
goto err; goto err;
/* Sync set controls */ /* Sync set controls */
ret = wm_coeff_sync_controls(dsp); ret = wm_coeff_sync_controls(dsp);
if (ret != 0) if (ret != 0)
goto err; goto err;
ret = regmap_update_bits_async(dsp->regmap,
dsp->base + ADSP2_CONTROL,
ADSP2_CORE_ENA,
ADSP2_CORE_ENA);
if (ret != 0)
goto err;
dsp->running = true;
return;
ret = regmap_update_bits_async(dsp->regmap, err:
dsp->base + ADSP2_CONTROL, regmap_update_bits(dsp->regmap, dsp->base + ADSP2_CONTROL,
ADSP2_CORE_ENA | ADSP2_START, ADSP2_SYS_ENA | ADSP2_CORE_ENA | ADSP2_START, 0);
ADSP2_CORE_ENA | ADSP2_START); }
int wm_adsp2_event(struct snd_soc_dapm_widget *w,
struct snd_kcontrol *kcontrol, int event)
{
struct snd_soc_codec *codec = w->codec;
struct wm_adsp *dsps = snd_soc_codec_get_drvdata(codec);
struct wm_adsp *dsp = &dsps[w->shift];
struct wm_adsp_alg_region *alg_region;
struct wm_coeff_ctl *ctl;
int ret;
dsp->card = codec->card;
switch (event) {
case SND_SOC_DAPM_POST_PMU:
queue_work(system_unbound_wq, &dsp->boot_work);
flush_work(&dsp->boot_work);
if (!dsp->running)
return -EIO;
ret = regmap_update_bits(dsp->regmap,
dsp->base + ADSP2_CONTROL,
ADSP2_START,
ADSP2_START);
if (ret != 0) if (ret != 0)
goto err; goto err;
dsp->running = true;
break; break;
case SND_SOC_DAPM_PRE_PMD: case SND_SOC_DAPM_PRE_PMD:
...@@ -1663,6 +1687,7 @@ int wm_adsp2_init(struct wm_adsp *adsp, bool dvfs) ...@@ -1663,6 +1687,7 @@ int wm_adsp2_init(struct wm_adsp *adsp, bool dvfs)
INIT_LIST_HEAD(&adsp->alg_regions); INIT_LIST_HEAD(&adsp->alg_regions);
INIT_LIST_HEAD(&adsp->ctl_list); INIT_LIST_HEAD(&adsp->ctl_list);
INIT_WORK(&adsp->boot_work, wm_adsp2_boot_work);
if (dvfs) { if (dvfs) {
adsp->dvfs = devm_regulator_get(adsp->dev, "DCVDD"); adsp->dvfs = devm_regulator_get(adsp->dev, "DCVDD");
......
...@@ -59,6 +59,8 @@ struct wm_adsp { ...@@ -59,6 +59,8 @@ struct wm_adsp {
struct regulator *dvfs; struct regulator *dvfs;
struct list_head ctl_list; struct list_head ctl_list;
struct work_struct boot_work;
}; };
#define WM_ADSP1(wname, num) \ #define WM_ADSP1(wname, num) \
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册