mirror of
https://github.com/AsahiLinux/u-boot
synced 2024-11-29 08:01:08 +00:00
clk: rockchip: Add rk322x gamc clock support
Assuming mac_clk is fed by an external clock, set clk_rmii_src clock select control register from IO for rgmii interface. Signed-off-by: David Wu <david.wu@rock-chips.com> Acked-by: Philipp Tomsich <philipp.tomsich@theobroma-systems.com> Reviewed-by: Philipp Tomsich <philipp.tomsich@theobroma-systems.com>
This commit is contained in:
parent
20ee0fd825
commit
5bb616c6e2
1 changed files with 107 additions and 0 deletions
|
@ -239,6 +239,41 @@ static ulong rockchip_mmc_get_clk(struct rk322x_cru *cru, uint clk_general_rate,
|
||||||
return DIV_TO_RATE(src_rate, div) / 2;
|
return DIV_TO_RATE(src_rate, div) / 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static ulong rk322x_mac_set_clk(struct rk322x_cru *cru, uint freq)
|
||||||
|
{
|
||||||
|
ulong ret;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The gmac clock can be derived either from an external clock
|
||||||
|
* or can be generated from internally by a divider from SCLK_MAC.
|
||||||
|
*/
|
||||||
|
if (readl(&cru->cru_clksel_con[5]) & BIT(5)) {
|
||||||
|
/* An external clock will always generate the right rate... */
|
||||||
|
ret = freq;
|
||||||
|
} else {
|
||||||
|
u32 con = readl(&cru->cru_clksel_con[5]);
|
||||||
|
ulong pll_rate;
|
||||||
|
u8 div;
|
||||||
|
|
||||||
|
if ((con >> MAC_PLL_SEL_SHIFT) & MAC_PLL_SEL_MASK)
|
||||||
|
pll_rate = GPLL_HZ;
|
||||||
|
else
|
||||||
|
/* CPLL is not set */
|
||||||
|
return -EPERM;
|
||||||
|
|
||||||
|
div = DIV_ROUND_UP(pll_rate, freq) - 1;
|
||||||
|
if (div <= 0x1f)
|
||||||
|
rk_clrsetreg(&cru->cru_clksel_con[5], CLK_MAC_DIV_MASK,
|
||||||
|
div << CLK_MAC_DIV_SHIFT);
|
||||||
|
else
|
||||||
|
debug("Unsupported div for gmac:%d\n", div);
|
||||||
|
|
||||||
|
return DIV_TO_RATE(pll_rate, div);
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
static ulong rockchip_mmc_set_clk(struct rk322x_cru *cru, uint clk_general_rate,
|
static ulong rockchip_mmc_set_clk(struct rk322x_cru *cru, uint clk_general_rate,
|
||||||
int periph, uint freq)
|
int periph, uint freq)
|
||||||
{
|
{
|
||||||
|
@ -352,6 +387,11 @@ static ulong rk322x_clk_set_rate(struct clk *clk, ulong rate)
|
||||||
case CLK_DDR:
|
case CLK_DDR:
|
||||||
new_rate = rk322x_ddr_set_clk(priv->cru, rate);
|
new_rate = rk322x_ddr_set_clk(priv->cru, rate);
|
||||||
break;
|
break;
|
||||||
|
case SCLK_MAC:
|
||||||
|
new_rate = rk322x_mac_set_clk(priv->cru, rate);
|
||||||
|
break;
|
||||||
|
case PLL_GPLL:
|
||||||
|
return 0;
|
||||||
default:
|
default:
|
||||||
return -ENOENT;
|
return -ENOENT;
|
||||||
}
|
}
|
||||||
|
@ -359,9 +399,76 @@ static ulong rk322x_clk_set_rate(struct clk *clk, ulong rate)
|
||||||
return new_rate;
|
return new_rate;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int rk322x_gmac_set_parent(struct clk *clk, struct clk *parent)
|
||||||
|
{
|
||||||
|
struct rk322x_clk_priv *priv = dev_get_priv(clk->dev);
|
||||||
|
struct rk322x_cru *cru = priv->cru;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the requested parent is in the same clock-controller and the id
|
||||||
|
* is SCLK_MAC_SRC ("sclk_gmac_src"), switch to the internal clock.
|
||||||
|
*/
|
||||||
|
if ((parent->dev == clk->dev) && (parent->id == SCLK_MAC_SRC)) {
|
||||||
|
debug("%s: switching RGMII to SCLK_MAC_SRC\n", __func__);
|
||||||
|
rk_clrsetreg(&cru->cru_clksel_con[5], BIT(5), 0);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* If the requested parent is in the same clock-controller and the id
|
||||||
|
* is SCLK_MAC_EXTCLK (sclk_mac_extclk), switch to the external clock.
|
||||||
|
*/
|
||||||
|
if ((parent->dev == clk->dev) && (parent->id == SCLK_MAC_EXTCLK)) {
|
||||||
|
debug("%s: switching RGMII to SCLK_MAC_EXTCLK\n", __func__);
|
||||||
|
rk_clrsetreg(&cru->cru_clksel_con[5], BIT(5), BIT(5));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int rk322x_gmac_extclk_set_parent(struct clk *clk, struct clk *parent)
|
||||||
|
{
|
||||||
|
struct rk322x_clk_priv *priv = dev_get_priv(clk->dev);
|
||||||
|
const char *clock_output_name;
|
||||||
|
struct rk322x_cru *cru = priv->cru;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = dev_read_string_index(parent->dev, "clock-output-names",
|
||||||
|
parent->id, &clock_output_name);
|
||||||
|
if (ret < 0)
|
||||||
|
return -ENODATA;
|
||||||
|
|
||||||
|
if (!strcmp(clock_output_name, "ext_gmac")) {
|
||||||
|
debug("%s: switching gmac extclk to ext_gmac\n", __func__);
|
||||||
|
rk_clrsetreg(&cru->cru_clksel_con[29], BIT(10), 0);
|
||||||
|
return 0;
|
||||||
|
} else if (!strcmp(clock_output_name, "phy_50m_out")) {
|
||||||
|
debug("%s: switching gmac extclk to phy_50m_out\n", __func__);
|
||||||
|
rk_clrsetreg(&cru->cru_clksel_con[29], BIT(10), BIT(10));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int rk322x_clk_set_parent(struct clk *clk, struct clk *parent)
|
||||||
|
{
|
||||||
|
switch (clk->id) {
|
||||||
|
case SCLK_MAC:
|
||||||
|
return rk322x_gmac_set_parent(clk, parent);
|
||||||
|
case SCLK_MAC_EXTCLK:
|
||||||
|
return rk322x_gmac_extclk_set_parent(clk, parent);
|
||||||
|
}
|
||||||
|
|
||||||
|
debug("%s: unsupported clk %ld\n", __func__, clk->id);
|
||||||
|
return -ENOENT;
|
||||||
|
}
|
||||||
|
|
||||||
static struct clk_ops rk322x_clk_ops = {
|
static struct clk_ops rk322x_clk_ops = {
|
||||||
.get_rate = rk322x_clk_get_rate,
|
.get_rate = rk322x_clk_get_rate,
|
||||||
.set_rate = rk322x_clk_set_rate,
|
.set_rate = rk322x_clk_set_rate,
|
||||||
|
.set_parent = rk322x_clk_set_parent,
|
||||||
};
|
};
|
||||||
|
|
||||||
static int rk322x_clk_ofdata_to_platdata(struct udevice *dev)
|
static int rk322x_clk_ofdata_to_platdata(struct udevice *dev)
|
||||||
|
|
Loading…
Reference in a new issue