Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit a38ced17 authored by Ed Blake's avatar Ed Blake Committed by Mark Brown
Browse files

ASoC: img-i2s-out: Add control of sys clock to runtime PM



Disable sys clock as well as ref clock when runtime suspended.

Signed-off-by: default avatarEd Blake <ed.blake@sondrel.com>
Signed-off-by: default avatarMark Brown <broonie@kernel.org>
parent 9b4acd33
Loading
Loading
Loading
Loading
+31 −29
Original line number Original line Diff line number Diff line
@@ -71,8 +71,8 @@ static int img_i2s_out_runtime_suspend(struct device *dev)
{
{
	struct img_i2s_out *i2s = dev_get_drvdata(dev);
	struct img_i2s_out *i2s = dev_get_drvdata(dev);


	if (!i2s->force_clk_active)
	clk_disable_unprepare(i2s->clk_ref);
	clk_disable_unprepare(i2s->clk_ref);
	clk_disable_unprepare(i2s->clk_sys);


	return 0;
	return 0;
}
}
@@ -82,12 +82,17 @@ static int img_i2s_out_runtime_resume(struct device *dev)
	struct img_i2s_out *i2s = dev_get_drvdata(dev);
	struct img_i2s_out *i2s = dev_get_drvdata(dev);
	int ret;
	int ret;


	if (!i2s->force_clk_active) {
	ret = clk_prepare_enable(i2s->clk_sys);
		ret = clk_prepare_enable(i2s->clk_ref);
	if (ret) {
	if (ret) {
		dev_err(dev, "clk_enable failed: %d\n", ret);
		dev_err(dev, "clk_enable failed: %d\n", ret);
		return ret;
		return ret;
	}
	}

	ret = clk_prepare_enable(i2s->clk_ref);
	if (ret) {
		dev_err(dev, "clk_enable failed: %d\n", ret);
		clk_disable_unprepare(i2s->clk_sys);
		return ret;
	}
	}


	return 0;
	return 0;
@@ -289,7 +294,7 @@ static int img_i2s_out_hw_params(struct snd_pcm_substream *substream,
static int img_i2s_out_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
static int img_i2s_out_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
{
{
	struct img_i2s_out *i2s = snd_soc_dai_get_drvdata(dai);
	struct img_i2s_out *i2s = snd_soc_dai_get_drvdata(dai);
	int i;
	int i, ret;
	bool force_clk_active;
	bool force_clk_active;
	u32 chan_control_mask, control_mask, chan_control_set = 0;
	u32 chan_control_mask, control_mask, chan_control_set = 0;
	u32 reg, control_set = 0;
	u32 reg, control_set = 0;
@@ -344,6 +349,10 @@ static int img_i2s_out_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)


	chan_control_mask = IMG_I2S_OUT_CHAN_CTL_CLKT_MASK;
	chan_control_mask = IMG_I2S_OUT_CHAN_CTL_CLKT_MASK;


	ret = pm_runtime_get_sync(i2s->dev);
	if (ret < 0)
		return ret;

	img_i2s_out_disable(i2s);
	img_i2s_out_disable(i2s);


	reg = img_i2s_out_readl(i2s, IMG_I2S_OUT_CTL);
	reg = img_i2s_out_readl(i2s, IMG_I2S_OUT_CTL);
@@ -363,6 +372,7 @@ static int img_i2s_out_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
		img_i2s_out_ch_enable(i2s, i);
		img_i2s_out_ch_enable(i2s, i);


	img_i2s_out_enable(i2s);
	img_i2s_out_enable(i2s);
	pm_runtime_put(i2s->dev);


	i2s->force_clk_active = force_clk_active;
	i2s->force_clk_active = force_clk_active;


@@ -469,16 +479,20 @@ static int img_i2s_out_probe(struct platform_device *pdev)
		return PTR_ERR(i2s->clk_ref);
		return PTR_ERR(i2s->clk_ref);
	}
	}


	ret = clk_prepare_enable(i2s->clk_sys);
	if (ret)
		return ret;

	i2s->suspend_ch_ctl = devm_kzalloc(dev,
	i2s->suspend_ch_ctl = devm_kzalloc(dev,
		sizeof(*i2s->suspend_ch_ctl) * i2s->max_i2s_chan, GFP_KERNEL);
		sizeof(*i2s->suspend_ch_ctl) * i2s->max_i2s_chan, GFP_KERNEL);
	if (!i2s->suspend_ch_ctl) {
	if (!i2s->suspend_ch_ctl)
		ret = -ENOMEM;
		return -ENOMEM;
		goto err_clk_disable;

	pm_runtime_enable(&pdev->dev);
	if (!pm_runtime_enabled(&pdev->dev)) {
		ret = img_i2s_out_runtime_resume(&pdev->dev);
		if (ret)
			goto err_pm_disable;
	}
	}
	ret = pm_runtime_get_sync(&pdev->dev);
	if (ret < 0)
		goto err_suspend;


	reg = IMG_I2S_OUT_CTL_FRM_SIZE_MASK;
	reg = IMG_I2S_OUT_CTL_FRM_SIZE_MASK;
	img_i2s_out_writel(i2s, reg, IMG_I2S_OUT_CTL);
	img_i2s_out_writel(i2s, reg, IMG_I2S_OUT_CTL);
@@ -492,13 +506,7 @@ static int img_i2s_out_probe(struct platform_device *pdev)
		img_i2s_out_ch_writel(i2s, i, reg, IMG_I2S_OUT_CH_CTL);
		img_i2s_out_ch_writel(i2s, i, reg, IMG_I2S_OUT_CH_CTL);


	img_i2s_out_reset(i2s);
	img_i2s_out_reset(i2s);

	pm_runtime_put(&pdev->dev);
	pm_runtime_enable(&pdev->dev);
	if (!pm_runtime_enabled(&pdev->dev)) {
		ret = img_i2s_out_runtime_resume(&pdev->dev);
		if (ret)
			goto err_pm_disable;
	}


	i2s->active_channels = 1;
	i2s->active_channels = 1;
	i2s->dma_data.addr = res->start + IMG_I2S_OUT_TX_FIFO;
	i2s->dma_data.addr = res->start + IMG_I2S_OUT_TX_FIFO;
@@ -529,22 +537,16 @@ static int img_i2s_out_probe(struct platform_device *pdev)
		img_i2s_out_runtime_suspend(&pdev->dev);
		img_i2s_out_runtime_suspend(&pdev->dev);
err_pm_disable:
err_pm_disable:
	pm_runtime_disable(&pdev->dev);
	pm_runtime_disable(&pdev->dev);
err_clk_disable:
	clk_disable_unprepare(i2s->clk_sys);


	return ret;
	return ret;
}
}


static int img_i2s_out_dev_remove(struct platform_device *pdev)
static int img_i2s_out_dev_remove(struct platform_device *pdev)
{
{
	struct img_i2s_out *i2s = platform_get_drvdata(pdev);

	pm_runtime_disable(&pdev->dev);
	pm_runtime_disable(&pdev->dev);
	if (!pm_runtime_status_suspended(&pdev->dev))
	if (!pm_runtime_status_suspended(&pdev->dev))
		img_i2s_out_runtime_suspend(&pdev->dev);
		img_i2s_out_runtime_suspend(&pdev->dev);


	clk_disable_unprepare(i2s->clk_sys);

	return 0;
	return 0;
}
}