|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Intel Corporation
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* Adjustable fractional divider clock implementation.
|
|
|
|
* Output rate = (m / n) * parent_rate.
|
|
|
|
* Uses rational best approximation algorithm.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/clk-provider.h>
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/device.h>
|
|
|
|
#include <linux/slab.h>
|
|
|
|
#include <linux/rational.h>
|
|
|
|
|
|
|
|
static unsigned long clk_fd_recalc_rate(struct clk_hw *hw,
|
|
|
|
unsigned long parent_rate)
|
|
|
|
{
|
|
|
|
struct clk_fractional_divider *fd = to_clk_fd(hw);
|
|
|
|
unsigned long flags = 0;
|
|
|
|
unsigned long m, n;
|
|
|
|
u32 val;
|
|
|
|
u64 ret;
|
|
|
|
|
|
|
|
if (fd->lock)
|
|
|
|
spin_lock_irqsave(fd->lock, flags);
|
|
|
|
else
|
|
|
|
__acquire(fd->lock);
|
|
|
|
|
|
|
|
val = clk_readl(fd->reg);
|
|
|
|
|
|
|
|
if (fd->lock)
|
|
|
|
spin_unlock_irqrestore(fd->lock, flags);
|
|
|
|
else
|
|
|
|
__release(fd->lock);
|
|
|
|
|
|
|
|
m = (val & fd->mmask) >> fd->mshift;
|
|
|
|
n = (val & fd->nmask) >> fd->nshift;
|
|
|
|
|
|
|
|
if (!n || !m)
|
|
|
|
return parent_rate;
|
|
|
|
|
|
|
|
ret = (u64)parent_rate * m;
|
|
|
|
do_div(ret, n);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void clk_fd_general_approximation(struct clk_hw *hw, unsigned long rate,
|
|
|
|
unsigned long *parent_rate,
|
|
|
|
unsigned long *m, unsigned long *n)
|
|
|
|
{
|
|
|
|
struct clk_fractional_divider *fd = to_clk_fd(hw);
|
|
|
|
unsigned long scale;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Get rate closer to *parent_rate to guarantee there is no overflow
|
|
|
|
* for m and n. In the result it will be the nearest rate left shifted
|
|
|
|
* by (scale - fd->nwidth) bits.
|
|
|
|
*/
|
|
|
|
scale = fls_long(*parent_rate / rate - 1);
|
|
|
|
if (scale > fd->nwidth)
|
|
|
|
rate <<= scale - fd->nwidth;
|
|
|
|
|
|
|
|
rational_best_approximation(rate, *parent_rate,
|
|
|
|
GENMASK(fd->mwidth - 1, 0), GENMASK(fd->nwidth - 1, 0),
|
|
|
|
m, n);
|
|
|
|
}
|
|
|
|
|
|
|
|
static long clk_fd_round_rate(struct clk_hw *hw, unsigned long rate,
|
|
|
|
unsigned long *parent_rate)
|
|
|
|
{
|
|
|
|
struct clk_fractional_divider *fd = to_clk_fd(hw);
|
|
|
|
unsigned long m, n;
|
|
|
|
u64 ret;
|
|
|
|
|
clk: fractional-divider: check parent rate only if flag is set
[ Upstream commit d13501a2bedfbea0983cc868d3f1dc692627f60d ]
Custom approximation of fractional-divider may not need parent clock
rate checking. For example Rockchip SoCs work fine using grand parent
clock rate even if target rate is greater than parent.
This patch checks parent clock rate only if CLK_SET_RATE_PARENT flag
is set.
For detailed example, clock tree of Rockchip I2S audio hardware.
- Clock rate of CPLL is 1.2GHz, GPLL is 491.52MHz.
- i2s1_div is integer divider can divide N (N is 1~128).
Input clock is CPLL or GPLL. Initial divider value is N = 1.
Ex) PLL = CPLL, N = 10, i2s1_div output rate is
CPLL / 10 = 1.2GHz / 10 = 120MHz
- i2s1_frac is fractional divider can divide input to x/y, x and
y are 16bit integer.
CPLL --> | selector | ---> i2s1_div -+--> | selector | --> I2S1 MCLK
GPLL --> | | ,--------------' | |
`--> i2s1_frac ---> | |
Clock mux system try to choose suitable one from i2s1_div and
i2s1_frac for master clock (MCLK) of I2S1.
Bad scenario as follows:
- Try to set MCLK to 8.192MHz (32kHz audio replay)
Candidate setting is
- i2s1_div: GPLL / 60 = 8.192MHz
i2s1_div candidate is exactly same as target clock rate, so mux
choose this clock source. i2s1_div output rate is changed
491.52MHz -> 8.192MHz
- After that try to set to 11.2896MHz (44.1kHz audio replay)
Candidate settings are
- i2s1_div : CPLL / 107 = 11.214945MHz
- i2s1_frac: i2s1_div = 8.192MHz
This is because clk_fd_round_rate() thinks target rate
(11.2896MHz) is higher than parent rate (i2s1_div = 8.192MHz)
and returns parent clock rate.
Above is current upstreamed behavior. Clock mux system choose
i2s1_div, but this clock rate is not acceptable for I2S driver, so
users cannot replay audio.
Expected behavior is:
- Try to set master clock to 11.2896MHz (44.1kHz audio replay)
Candidate settings are
- i2s1_div : CPLL / 107 = 11.214945MHz
- i2s1_frac: i2s1_div * 147/6400 = 11.2896MHz
Change i2s1_div to GPLL / 1 = 491.52MHz at same
time.
If apply this commit, clk_fd_round_rate() calls custom approximate
function of Rockchip even if target rate is higher than parent.
Custom function changes both grand parent (i2s1_div) and parent
(i2s_frac) settings at same time. Clock mux system can choose
i2s1_frac and audio works fine.
Signed-off-by: Katsuhiro Suzuki <katsuhiro@katsuster.net>
Reviewed-by: Heiko Stuebner <heiko@sntech.de>
[sboyd@kernel.org: Make function into a macro instead]
Signed-off-by: Stephen Boyd <sboyd@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
6 years ago
|
|
|
if (!rate || (!clk_hw_can_set_rate_parent(hw) && rate >= *parent_rate))
|
|
|
|
return *parent_rate;
|
|
|
|
|
|
|
|
if (fd->approximation)
|
|
|
|
fd->approximation(hw, rate, parent_rate, &m, &n);
|
|
|
|
else
|
|
|
|
clk_fd_general_approximation(hw, rate, parent_rate, &m, &n);
|
|
|
|
|
|
|
|
ret = (u64)*parent_rate * m;
|
|
|
|
do_div(ret, n);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int clk_fd_set_rate(struct clk_hw *hw, unsigned long rate,
|
|
|
|
unsigned long parent_rate)
|
|
|
|
{
|
|
|
|
struct clk_fractional_divider *fd = to_clk_fd(hw);
|
|
|
|
unsigned long flags = 0;
|
|
|
|
unsigned long m, n;
|
|
|
|
u32 val;
|
|
|
|
|
|
|
|
rational_best_approximation(rate, parent_rate,
|
|
|
|
GENMASK(fd->mwidth - 1, 0), GENMASK(fd->nwidth - 1, 0),
|
|
|
|
&m, &n);
|
|
|
|
|
|
|
|
if (fd->lock)
|
|
|
|
spin_lock_irqsave(fd->lock, flags);
|
|
|
|
else
|
|
|
|
__acquire(fd->lock);
|
|
|
|
|
|
|
|
val = clk_readl(fd->reg);
|
|
|
|
val &= ~(fd->mmask | fd->nmask);
|
|
|
|
val |= (m << fd->mshift) | (n << fd->nshift);
|
|
|
|
clk_writel(val, fd->reg);
|
|
|
|
|
|
|
|
if (fd->lock)
|
|
|
|
spin_unlock_irqrestore(fd->lock, flags);
|
|
|
|
else
|
|
|
|
__release(fd->lock);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct clk_ops clk_fractional_divider_ops = {
|
|
|
|
.recalc_rate = clk_fd_recalc_rate,
|
|
|
|
.round_rate = clk_fd_round_rate,
|
|
|
|
.set_rate = clk_fd_set_rate,
|
|
|
|
};
|
|
|
|
EXPORT_SYMBOL_GPL(clk_fractional_divider_ops);
|
|
|
|
|
|
|
|
struct clk_hw *clk_hw_register_fractional_divider(struct device *dev,
|
|
|
|
const char *name, const char *parent_name, unsigned long flags,
|
|
|
|
void __iomem *reg, u8 mshift, u8 mwidth, u8 nshift, u8 nwidth,
|
|
|
|
u8 clk_divider_flags, spinlock_t *lock)
|
|
|
|
{
|
|
|
|
struct clk_fractional_divider *fd;
|
|
|
|
struct clk_init_data init = {};
|
|
|
|
struct clk_hw *hw;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
fd = kzalloc(sizeof(*fd), GFP_KERNEL);
|
|
|
|
if (!fd)
|
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
|
|
|
init.name = name;
|
|
|
|
init.ops = &clk_fractional_divider_ops;
|
|
|
|
init.flags = flags | CLK_IS_BASIC;
|
|
|
|
init.parent_names = parent_name ? &parent_name : NULL;
|
|
|
|
init.num_parents = parent_name ? 1 : 0;
|
|
|
|
|
|
|
|
fd->reg = reg;
|
|
|
|
fd->mshift = mshift;
|
|
|
|
fd->mwidth = mwidth;
|
|
|
|
fd->mmask = GENMASK(mwidth - 1, 0) << mshift;
|
|
|
|
fd->nshift = nshift;
|
|
|
|
fd->nwidth = nwidth;
|
|
|
|
fd->nmask = GENMASK(nwidth - 1, 0) << nshift;
|
|
|
|
fd->flags = clk_divider_flags;
|
|
|
|
fd->lock = lock;
|
|
|
|
fd->hw.init = &init;
|
|
|
|
|
|
|
|
hw = &fd->hw;
|
|
|
|
ret = clk_hw_register(dev, hw);
|
|
|
|
if (ret) {
|
|
|
|
kfree(fd);
|
|
|
|
hw = ERR_PTR(ret);
|
|
|
|
}
|
|
|
|
|
|
|
|
return hw;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(clk_hw_register_fractional_divider);
|
|
|
|
|
|
|
|
struct clk *clk_register_fractional_divider(struct device *dev,
|
|
|
|
const char *name, const char *parent_name, unsigned long flags,
|
|
|
|
void __iomem *reg, u8 mshift, u8 mwidth, u8 nshift, u8 nwidth,
|
|
|
|
u8 clk_divider_flags, spinlock_t *lock)
|
|
|
|
{
|
|
|
|
struct clk_hw *hw;
|
|
|
|
|
|
|
|
hw = clk_hw_register_fractional_divider(dev, name, parent_name, flags,
|
|
|
|
reg, mshift, mwidth, nshift, nwidth, clk_divider_flags,
|
|
|
|
lock);
|
|
|
|
if (IS_ERR(hw))
|
|
|
|
return ERR_CAST(hw);
|
|
|
|
return hw->clk;
|
|
|
|
}
|
|
|
|
EXPORT_SYMBOL_GPL(clk_register_fractional_divider);
|
|
|
|
|
|
|
|
void clk_hw_unregister_fractional_divider(struct clk_hw *hw)
|
|
|
|
{
|
|
|
|
struct clk_fractional_divider *fd;
|
|
|
|
|
|
|
|
fd = to_clk_fd(hw);
|
|
|
|
|
|
|
|
clk_hw_unregister(hw);
|
|
|
|
kfree(fd);
|
|
|
|
}
|