2019-06-04 10:11:33 +02:00
// SPDX-License-Identifier: GPL-2.0-only
2011-02-25 15:39:28 -07:00
/*
* OMAP2 / 3 interface clock control
*
* Copyright ( C ) 2011 Nokia Corporation
* Paul Walmsley
*/
# undef DEBUG
# include <linux/kernel.h>
2012-11-10 16:58:41 -07:00
# include <linux/clk-provider.h>
2011-02-25 15:39:28 -07:00
# include <linux/io.h>
2015-03-02 14:33:54 +02:00
# include <linux/clk/ti.h>
2011-02-25 15:39:28 -07:00
# include "clock.h"
2014-07-02 11:47:47 +03:00
/* Register offsets */
2015-03-03 16:08:42 +02:00
# define OMAP24XX_CM_FCLKEN2 0x04
2014-07-02 11:47:47 +03:00
# define CM_AUTOIDLE 0x30
# define CM_ICLKEN 0x10
2015-03-03 16:08:42 +02:00
# define CM_IDLEST 0x20
# define OMAP24XX_CM_IDLEST_VAL 0
2011-02-25 15:39:28 -07:00
/* Private functions */
/* XXX */
2012-04-27 15:53:48 +05:30
void omap2_clkt_iclk_allow_idle ( struct clk_hw_omap * clk )
2011-02-25 15:39:28 -07:00
{
2013-10-22 11:49:58 +03:00
u32 v ;
2017-02-09 11:24:37 +02:00
struct clk_omap_reg r ;
2011-02-25 15:39:28 -07:00
2017-02-09 11:24:37 +02:00
memcpy ( & r , & clk - > enable_reg , sizeof ( r ) ) ;
r . offset ^ = ( CM_AUTOIDLE ^ CM_ICLKEN ) ;
2011-02-25 15:39:28 -07:00
2017-02-09 11:24:37 +02:00
v = ti_clk_ll_ops - > clk_readl ( & r ) ;
2011-02-25 15:39:28 -07:00
v | = ( 1 < < clk - > enable_bit ) ;
2017-02-09 11:24:37 +02:00
ti_clk_ll_ops - > clk_writel ( v , & r ) ;
2011-02-25 15:39:28 -07:00
}
/* XXX */
2012-04-27 15:53:48 +05:30
void omap2_clkt_iclk_deny_idle ( struct clk_hw_omap * clk )
2011-02-25 15:39:28 -07:00
{
2013-10-22 11:49:58 +03:00
u32 v ;
2017-02-09 11:24:37 +02:00
struct clk_omap_reg r ;
2011-02-25 15:39:28 -07:00
2017-02-09 11:24:37 +02:00
memcpy ( & r , & clk - > enable_reg , sizeof ( r ) ) ;
2011-02-25 15:39:28 -07:00
2017-02-09 11:24:37 +02:00
r . offset ^ = ( CM_AUTOIDLE ^ CM_ICLKEN ) ;
v = ti_clk_ll_ops - > clk_readl ( & r ) ;
2011-02-25 15:39:28 -07:00
v & = ~ ( 1 < < clk - > enable_bit ) ;
2017-02-09 11:24:37 +02:00
ti_clk_ll_ops - > clk_writel ( v , & r ) ;
2011-02-25 15:39:28 -07:00
}
2015-03-03 16:08:42 +02:00
/**
* omap2430_clk_i2chs_find_idlest - return CM_IDLEST info for 2430 I2CHS
* @ clk : struct clk * being enabled
* @ idlest_reg : void __iomem * * to store CM_IDLEST reg address into
* @ idlest_bit : pointer to a u8 to store the CM_IDLEST bit shift into
* @ idlest_val : pointer to a u8 to store the CM_IDLEST indicator
*
* OMAP2430 I2CHS CM_IDLEST bits are in CM_IDLEST1_CORE , but the
* CM_ * CLKEN bits are in CM_ { I , F } CLKEN2_CORE . This custom function
* passes back the correct CM_IDLEST register address for I2CHS
* modules . No return value .
*/
static void omap2430_clk_i2chs_find_idlest ( struct clk_hw_omap * clk ,
2017-02-09 11:24:37 +02:00
struct clk_omap_reg * idlest_reg ,
2015-03-03 16:08:42 +02:00
u8 * idlest_bit ,
u8 * idlest_val )
{
2017-02-09 11:24:37 +02:00
memcpy ( idlest_reg , & clk - > enable_reg , sizeof ( * idlest_reg ) ) ;
idlest_reg - > offset ^ = ( OMAP24XX_CM_FCLKEN2 ^ CM_IDLEST ) ;
2015-03-03 16:08:42 +02:00
* idlest_bit = clk - > enable_bit ;
* idlest_val = OMAP24XX_CM_IDLEST_VAL ;
}
2011-02-25 15:39:28 -07:00
/* Public data */
2012-04-27 15:53:48 +05:30
const struct clk_hw_omap_ops clkhwops_iclk = {
. allow_idle = omap2_clkt_iclk_allow_idle ,
. deny_idle = omap2_clkt_iclk_deny_idle ,
} ;
2012-11-10 16:58:41 -07:00
const struct clk_hw_omap_ops clkhwops_iclk_wait = {
. allow_idle = omap2_clkt_iclk_allow_idle ,
. deny_idle = omap2_clkt_iclk_deny_idle ,
. find_idlest = omap2_clk_dflt_find_idlest ,
. find_companion = omap2_clk_dflt_find_companion ,
} ;
2015-03-03 16:08:42 +02:00
/* 2430 I2CHS has non-standard IDLEST register */
const struct clk_hw_omap_ops clkhwops_omap2430_i2chs_wait = {
. find_idlest = omap2430_clk_i2chs_find_idlest ,
. find_companion = omap2_clk_dflt_find_companion ,
} ;