|
@@ -71,8 +71,8 @@ static int img_i2s_out_runtime_suspend(struct device *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;
|
|
|
}
|
|
@@ -82,12 +82,17 @@ static int img_i2s_out_runtime_resume(struct device *dev)
|
|
|
struct img_i2s_out *i2s = dev_get_drvdata(dev);
|
|
|
int ret;
|
|
|
|
|
|
- if (!i2s->force_clk_active) {
|
|
|
- ret = clk_prepare_enable(i2s->clk_ref);
|
|
|
- if (ret) {
|
|
|
- dev_err(dev, "clk_enable failed: %d\n", ret);
|
|
|
- return ret;
|
|
|
- }
|
|
|
+ ret = clk_prepare_enable(i2s->clk_sys);
|
|
|
+ if (ret) {
|
|
|
+ dev_err(dev, "clk_enable failed: %d\n", 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;
|
|
@@ -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)
|
|
|
{
|
|
|
struct img_i2s_out *i2s = snd_soc_dai_get_drvdata(dai);
|
|
|
- int i;
|
|
|
+ int i, ret;
|
|
|
bool force_clk_active;
|
|
|
u32 chan_control_mask, control_mask, chan_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;
|
|
|
|
|
|
+ ret = pm_runtime_get_sync(i2s->dev);
|
|
|
+ if (ret < 0)
|
|
|
+ return ret;
|
|
|
+
|
|
|
img_i2s_out_disable(i2s);
|
|
|
|
|
|
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_enable(i2s);
|
|
|
+ pm_runtime_put(i2s->dev);
|
|
|
|
|
|
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);
|
|
|
}
|
|
|
|
|
|
- ret = clk_prepare_enable(i2s->clk_sys);
|
|
|
- if (ret)
|
|
|
- return ret;
|
|
|
-
|
|
|
i2s->suspend_ch_ctl = devm_kzalloc(dev,
|
|
|
sizeof(*i2s->suspend_ch_ctl) * i2s->max_i2s_chan, GFP_KERNEL);
|
|
|
- if (!i2s->suspend_ch_ctl) {
|
|
|
- ret = -ENOMEM;
|
|
|
- goto err_clk_disable;
|
|
|
+ if (!i2s->suspend_ch_ctl)
|
|
|
+ return -ENOMEM;
|
|
|
+
|
|
|
+ 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;
|
|
|
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_reset(i2s);
|
|
|
-
|
|
|
- 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;
|
|
|
- }
|
|
|
+ pm_runtime_put(&pdev->dev);
|
|
|
|
|
|
i2s->active_channels = 1;
|
|
|
i2s->dma_data.addr = res->start + IMG_I2S_OUT_TX_FIFO;
|
|
@@ -529,22 +537,16 @@ err_suspend:
|
|
|
img_i2s_out_runtime_suspend(&pdev->dev);
|
|
|
err_pm_disable:
|
|
|
pm_runtime_disable(&pdev->dev);
|
|
|
-err_clk_disable:
|
|
|
- clk_disable_unprepare(i2s->clk_sys);
|
|
|
|
|
|
return ret;
|
|
|
}
|
|
|
|
|
|
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);
|
|
|
if (!pm_runtime_status_suspended(&pdev->dev))
|
|
|
img_i2s_out_runtime_suspend(&pdev->dev);
|
|
|
|
|
|
- clk_disable_unprepare(i2s->clk_sys);
|
|
|
-
|
|
|
return 0;
|
|
|
}
|
|
|
|