forked from mirrors/linux
		
	i2c: imx: use clk notifier for rate changes
Instead of repeatedly calling clk_get_rate for each transfer, register a clock notifier to update the cached divider value each time the clock rate actually changes. Signed-off-by: Lucas Stach <l.stach@pengutronix.de> Reviewed-by: Philipp Zabel <p.zabel@pengutronix.de> Signed-off-by: Wolfram Sang <wsa@the-dreams.de>
This commit is contained in:
		
							parent
							
								
									398432edde
								
							
						
					
					
						commit
						90ad2cbe88
					
				
					 1 changed files with 25 additions and 7 deletions
				
			
		|  | @ -194,6 +194,7 @@ struct imx_i2c_dma { | ||||||
| struct imx_i2c_struct { | struct imx_i2c_struct { | ||||||
| 	struct i2c_adapter	adapter; | 	struct i2c_adapter	adapter; | ||||||
| 	struct clk		*clk; | 	struct clk		*clk; | ||||||
|  | 	struct notifier_block	clk_change_nb; | ||||||
| 	void __iomem		*base; | 	void __iomem		*base; | ||||||
| 	wait_queue_head_t	queue; | 	wait_queue_head_t	queue; | ||||||
| 	unsigned long		i2csr; | 	unsigned long		i2csr; | ||||||
|  | @ -467,15 +468,14 @@ static int i2c_imx_acked(struct imx_i2c_struct *i2c_imx) | ||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static void i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx) | static void i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx, | ||||||
|  | 			    unsigned int i2c_clk_rate) | ||||||
| { | { | ||||||
| 	struct imx_i2c_clk_pair *i2c_clk_div = i2c_imx->hwdata->clk_div; | 	struct imx_i2c_clk_pair *i2c_clk_div = i2c_imx->hwdata->clk_div; | ||||||
| 	unsigned int i2c_clk_rate; |  | ||||||
| 	unsigned int div; | 	unsigned int div; | ||||||
| 	int i; | 	int i; | ||||||
| 
 | 
 | ||||||
| 	/* Divider value calculation */ | 	/* Divider value calculation */ | ||||||
| 	i2c_clk_rate = clk_get_rate(i2c_imx->clk); |  | ||||||
| 	if (i2c_imx->cur_clk == i2c_clk_rate) | 	if (i2c_imx->cur_clk == i2c_clk_rate) | ||||||
| 		return; | 		return; | ||||||
| 
 | 
 | ||||||
|  | @ -510,6 +510,20 @@ static void i2c_imx_set_clk(struct imx_i2c_struct *i2c_imx) | ||||||
| #endif | #endif | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | static int i2c_imx_clk_notifier_call(struct notifier_block *nb, | ||||||
|  | 				     unsigned long action, void *data) | ||||||
|  | { | ||||||
|  | 	struct clk_notifier_data *ndata = data; | ||||||
|  | 	struct imx_i2c_struct *i2c_imx = container_of(&ndata->clk, | ||||||
|  | 						      struct imx_i2c_struct, | ||||||
|  | 						      clk); | ||||||
|  | 
 | ||||||
|  | 	if (action & POST_RATE_CHANGE) | ||||||
|  | 		i2c_imx_set_clk(i2c_imx, ndata->new_rate); | ||||||
|  | 
 | ||||||
|  | 	return NOTIFY_OK; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| static int i2c_imx_start(struct imx_i2c_struct *i2c_imx) | static int i2c_imx_start(struct imx_i2c_struct *i2c_imx) | ||||||
| { | { | ||||||
| 	unsigned int temp = 0; | 	unsigned int temp = 0; | ||||||
|  | @ -517,8 +531,6 @@ static int i2c_imx_start(struct imx_i2c_struct *i2c_imx) | ||||||
| 
 | 
 | ||||||
| 	dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); | 	dev_dbg(&i2c_imx->adapter.dev, "<%s>\n", __func__); | ||||||
| 
 | 
 | ||||||
| 	i2c_imx_set_clk(i2c_imx); |  | ||||||
| 
 |  | ||||||
| 	imx_i2c_write_reg(i2c_imx->ifdr, i2c_imx, IMX_I2C_IFDR); | 	imx_i2c_write_reg(i2c_imx->ifdr, i2c_imx, IMX_I2C_IFDR); | ||||||
| 	/* Enable I2C controller */ | 	/* Enable I2C controller */ | ||||||
| 	imx_i2c_write_reg(i2c_imx->hwdata->i2sr_clr_opcode, i2c_imx, IMX_I2C_I2SR); | 	imx_i2c_write_reg(i2c_imx->hwdata->i2sr_clr_opcode, i2c_imx, IMX_I2C_I2SR); | ||||||
|  | @ -1131,6 +1143,9 @@ static int i2c_imx_probe(struct platform_device *pdev) | ||||||
| 				   "clock-frequency", &i2c_imx->bitrate); | 				   "clock-frequency", &i2c_imx->bitrate); | ||||||
| 	if (ret < 0 && pdata && pdata->bitrate) | 	if (ret < 0 && pdata && pdata->bitrate) | ||||||
| 		i2c_imx->bitrate = pdata->bitrate; | 		i2c_imx->bitrate = pdata->bitrate; | ||||||
|  | 	i2c_imx->clk_change_nb.notifier_call = i2c_imx_clk_notifier_call; | ||||||
|  | 	clk_notifier_register(i2c_imx->clk, &i2c_imx->clk_change_nb); | ||||||
|  | 	i2c_imx_set_clk(i2c_imx, clk_get_rate(i2c_imx->clk)); | ||||||
| 
 | 
 | ||||||
| 	/* Set up chip registers to defaults */ | 	/* Set up chip registers to defaults */ | ||||||
| 	imx_i2c_write_reg(i2c_imx->hwdata->i2cr_ien_opcode ^ I2CR_IEN, | 	imx_i2c_write_reg(i2c_imx->hwdata->i2cr_ien_opcode ^ I2CR_IEN, | ||||||
|  | @ -1141,12 +1156,12 @@ static int i2c_imx_probe(struct platform_device *pdev) | ||||||
| 	ret = i2c_imx_init_recovery_info(i2c_imx, pdev); | 	ret = i2c_imx_init_recovery_info(i2c_imx, pdev); | ||||||
| 	/* Give it another chance if pinctrl used is not ready yet */ | 	/* Give it another chance if pinctrl used is not ready yet */ | ||||||
| 	if (ret == -EPROBE_DEFER) | 	if (ret == -EPROBE_DEFER) | ||||||
| 		goto rpm_disable; | 		goto clk_notifier_unregister; | ||||||
| 
 | 
 | ||||||
| 	/* Add I2C adapter */ | 	/* Add I2C adapter */ | ||||||
| 	ret = i2c_add_numbered_adapter(&i2c_imx->adapter); | 	ret = i2c_add_numbered_adapter(&i2c_imx->adapter); | ||||||
| 	if (ret < 0) | 	if (ret < 0) | ||||||
| 		goto rpm_disable; | 		goto clk_notifier_unregister; | ||||||
| 
 | 
 | ||||||
| 	pm_runtime_mark_last_busy(&pdev->dev); | 	pm_runtime_mark_last_busy(&pdev->dev); | ||||||
| 	pm_runtime_put_autosuspend(&pdev->dev); | 	pm_runtime_put_autosuspend(&pdev->dev); | ||||||
|  | @ -1162,6 +1177,8 @@ static int i2c_imx_probe(struct platform_device *pdev) | ||||||
| 
 | 
 | ||||||
| 	return 0;   /* Return OK */ | 	return 0;   /* Return OK */ | ||||||
| 
 | 
 | ||||||
|  | clk_notifier_unregister: | ||||||
|  | 	clk_notifier_unregister(i2c_imx->clk, &i2c_imx->clk_change_nb); | ||||||
| rpm_disable: | rpm_disable: | ||||||
| 	pm_runtime_put_noidle(&pdev->dev); | 	pm_runtime_put_noidle(&pdev->dev); | ||||||
| 	pm_runtime_disable(&pdev->dev); | 	pm_runtime_disable(&pdev->dev); | ||||||
|  | @ -1195,6 +1212,7 @@ static int i2c_imx_remove(struct platform_device *pdev) | ||||||
| 	imx_i2c_write_reg(0, i2c_imx, IMX_I2C_I2CR); | 	imx_i2c_write_reg(0, i2c_imx, IMX_I2C_I2CR); | ||||||
| 	imx_i2c_write_reg(0, i2c_imx, IMX_I2C_I2SR); | 	imx_i2c_write_reg(0, i2c_imx, IMX_I2C_I2SR); | ||||||
| 
 | 
 | ||||||
|  | 	clk_notifier_unregister(i2c_imx->clk, &i2c_imx->clk_change_nb); | ||||||
| 	clk_disable_unprepare(i2c_imx->clk); | 	clk_disable_unprepare(i2c_imx->clk); | ||||||
| 
 | 
 | ||||||
| 	pm_runtime_put_noidle(&pdev->dev); | 	pm_runtime_put_noidle(&pdev->dev); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in a new issue
	
	 Lucas Stach
						Lucas Stach