diff options
Diffstat (limited to 'drivers/dma/stm32-dmamux.c')
| -rw-r--r-- | drivers/dma/stm32-dmamux.c | 93 | 
1 files changed, 77 insertions, 16 deletions
diff --git a/drivers/dma/stm32-dmamux.c b/drivers/dma/stm32-dmamux.c index 3c89bd39e096..12f7637e13a1 100644 --- a/drivers/dma/stm32-dmamux.c +++ b/drivers/dma/stm32-dmamux.c @@ -35,12 +35,14 @@ struct stm32_dmamux {  struct stm32_dmamux_data {  	struct dma_router dmarouter;  	struct clk *clk; -	struct reset_control *rst;  	void __iomem *iomem;  	u32 dma_requests; /* Number of DMA requests connected to DMAMUX */  	u32 dmamux_requests; /* Number of DMA requests routed toward DMAs */  	spinlock_t lock; /* Protects register access */  	unsigned long *dma_inuse; /* Used DMA channel */ +	u32 ccr[STM32_DMAMUX_MAX_DMA_REQUESTS]; /* Used to backup CCR register +						 * in suspend +						 */  	u32 dma_reqs[]; /* Number of DMA Request per DMA masters.  			 *  [0] holds number of DMA Masters.  			 *  To be kept at very end end of this structure @@ -179,6 +181,7 @@ static int stm32_dmamux_probe(struct platform_device *pdev)  	struct stm32_dmamux_data *stm32_dmamux;  	struct resource *res;  	void __iomem *iomem; +	struct reset_control *rst;  	int i, count, ret;  	u32 dma_req; @@ -251,16 +254,26 @@ static int stm32_dmamux_probe(struct platform_device *pdev)  	stm32_dmamux->clk = devm_clk_get(&pdev->dev, NULL);  	if (IS_ERR(stm32_dmamux->clk)) {  		ret = PTR_ERR(stm32_dmamux->clk); -		if (ret == -EPROBE_DEFER) -			dev_info(&pdev->dev, "Missing controller clock\n"); +		if (ret != -EPROBE_DEFER) +			dev_err(&pdev->dev, "Missing clock controller\n"); +		return ret; +	} + +	ret = clk_prepare_enable(stm32_dmamux->clk); +	if (ret < 0) { +		dev_err(&pdev->dev, "clk_prep_enable error: %d\n", ret);  		return ret;  	} -	stm32_dmamux->rst = devm_reset_control_get(&pdev->dev, NULL); -	if (!IS_ERR(stm32_dmamux->rst)) { -		reset_control_assert(stm32_dmamux->rst); +	rst = devm_reset_control_get(&pdev->dev, NULL); +	if (IS_ERR(rst)) { +		ret = PTR_ERR(rst); +		if (ret == -EPROBE_DEFER) +			goto err_clk; +	} else { +		reset_control_assert(rst);  		udelay(2); -		reset_control_deassert(stm32_dmamux->rst); +		reset_control_deassert(rst);  	}  	stm32_dmamux->iomem = iomem; @@ -271,14 +284,6 @@ static int stm32_dmamux_probe(struct platform_device *pdev)  	pm_runtime_set_active(&pdev->dev);  	pm_runtime_enable(&pdev->dev); -	if (!IS_ERR(stm32_dmamux->clk)) { -		ret = clk_prepare_enable(stm32_dmamux->clk); -		if (ret < 0) { -			dev_err(&pdev->dev, "clk_prep_enable error: %d\n", ret); -			return ret; -		} -	} -  	pm_runtime_get_noresume(&pdev->dev);  	/* Reset the dmamux */ @@ -287,8 +292,17 @@ static int stm32_dmamux_probe(struct platform_device *pdev)  	pm_runtime_put(&pdev->dev); -	return of_dma_router_register(node, stm32_dmamux_route_allocate, +	ret = of_dma_router_register(node, stm32_dmamux_route_allocate,  				     &stm32_dmamux->dmarouter); +	if (ret) +		goto err_clk; + +	return 0; + +err_clk: +	clk_disable_unprepare(stm32_dmamux->clk); + +	return ret;  }  #ifdef CONFIG_PM @@ -318,7 +332,54 @@ static int stm32_dmamux_runtime_resume(struct device *dev)  }  #endif +#ifdef CONFIG_PM_SLEEP +static int stm32_dmamux_suspend(struct device *dev) +{ +	struct platform_device *pdev = to_platform_device(dev); +	struct stm32_dmamux_data *stm32_dmamux = platform_get_drvdata(pdev); +	int i, ret; + +	ret = pm_runtime_get_sync(dev); +	if (ret < 0) +		return ret; + +	for (i = 0; i < stm32_dmamux->dma_requests; i++) +		stm32_dmamux->ccr[i] = stm32_dmamux_read(stm32_dmamux->iomem, +							 STM32_DMAMUX_CCR(i)); + +	pm_runtime_put_sync(dev); + +	pm_runtime_force_suspend(dev); + +	return 0; +} + +static int stm32_dmamux_resume(struct device *dev) +{ +	struct platform_device *pdev = to_platform_device(dev); +	struct stm32_dmamux_data *stm32_dmamux = platform_get_drvdata(pdev); +	int i, ret; + +	ret = pm_runtime_force_resume(dev); +	if (ret < 0) +		return ret; + +	ret = pm_runtime_get_sync(dev); +	if (ret < 0) +		return ret; + +	for (i = 0; i < stm32_dmamux->dma_requests; i++) +		stm32_dmamux_write(stm32_dmamux->iomem, STM32_DMAMUX_CCR(i), +				   stm32_dmamux->ccr[i]); + +	pm_runtime_put_sync(dev); + +	return 0; +} +#endif +  static const struct dev_pm_ops stm32_dmamux_pm_ops = { +	SET_SYSTEM_SLEEP_PM_OPS(stm32_dmamux_suspend, stm32_dmamux_resume)  	SET_RUNTIME_PM_OPS(stm32_dmamux_runtime_suspend,  			   stm32_dmamux_runtime_resume, NULL)  };  |