mirror of
https://source.denx.de/u-boot/u-boot.git
synced 2025-08-18 21:21:26 +02:00
When bringing in the series 'arm: dts: am62-beagleplay: Fix Beagleplay Ethernet"' I failed to notice that b4 noticed it was based on next and so took that as the base commit and merged that part of next to master. This reverts commit c8ffd1356d42223cbb8c86280a083cc3c93e6426, reversing changes made to 2ee6f3a5f7550de3599faef9704e166e5dcace35. Reported-by: Jonas Karlman <jonas@kwiboo.se> Signed-off-by: Tom Rini <trini@konsulko.com>
383 lines
9.4 KiB
C
383 lines
9.4 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/*
|
|
* Copyright 2021 NXP
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <asm/io.h>
|
|
#include <malloc.h>
|
|
#include <clk-uclass.h>
|
|
#include <dm/device.h>
|
|
#include <dm/devres.h>
|
|
#include <linux/bitfield.h>
|
|
#include <linux/bitops.h>
|
|
#include <linux/clk-provider.h>
|
|
#include <linux/delay.h>
|
|
#include <linux/err.h>
|
|
#include <linux/iopoll.h>
|
|
#include <clk.h>
|
|
#include <div64.h>
|
|
|
|
#include "clk.h"
|
|
|
|
#define UBOOT_DM_CLK_IMX_FRACN_GPPLL "imx_clk_fracn_gppll"
|
|
|
|
#define PLL_CTRL 0x0
|
|
#define HW_CTRL_SEL BIT(16)
|
|
#define CLKMUX_BYPASS BIT(2)
|
|
#define CLKMUX_EN BIT(1)
|
|
#define POWERUP_MASK BIT(0)
|
|
|
|
#define PLL_ANA_PRG 0x10
|
|
#define PLL_SPREAD_SPECTRUM 0x30
|
|
|
|
#define PLL_NUMERATOR 0x40
|
|
#define PLL_MFN_MASK GENMASK(31, 2)
|
|
|
|
#define PLL_DENOMINATOR 0x50
|
|
#define PLL_MFD_MASK GENMASK(29, 0)
|
|
|
|
#define PLL_DIV 0x60
|
|
#define PLL_MFI_MASK GENMASK(24, 16)
|
|
#define PLL_RDIV_MASK GENMASK(15, 13)
|
|
#define PLL_ODIV_MASK GENMASK(7, 0)
|
|
|
|
#define PLL_DFS_CTRL(x) (0x70 + (x) * 0x10)
|
|
|
|
#define PLL_STATUS 0xF0
|
|
#define LOCK_STATUS BIT(0)
|
|
|
|
#define DFS_STATUS 0xF4
|
|
|
|
#define LOCK_TIMEOUT_US 200
|
|
|
|
#define PLL_FRACN_GP(_rate, _mfi, _mfn, _mfd, _rdiv, _odiv) \
|
|
{ \
|
|
.rate = (_rate), \
|
|
.mfi = (_mfi), \
|
|
.mfn = (_mfn), \
|
|
.mfd = (_mfd), \
|
|
.rdiv = (_rdiv), \
|
|
.odiv = (_odiv), \
|
|
}
|
|
|
|
#define PLL_FRACN_GP_INTEGER(_rate, _mfi, _rdiv, _odiv) \
|
|
{ \
|
|
.rate = (_rate), \
|
|
.mfi = (_mfi), \
|
|
.mfn = 0, \
|
|
.mfd = 0, \
|
|
.rdiv = (_rdiv), \
|
|
.odiv = (_odiv), \
|
|
}
|
|
|
|
struct clk_fracn_gppll {
|
|
struct clk clk;
|
|
void __iomem *base;
|
|
const struct imx_fracn_gppll_rate_table *rate_table;
|
|
int rate_count;
|
|
u32 flags;
|
|
};
|
|
|
|
/*
|
|
* Fvco = (Fref / rdiv) * (MFI + MFN / MFD)
|
|
* Fout = Fvco / odiv
|
|
* The (Fref / rdiv) should be in range 20MHz to 40MHz
|
|
* The Fvco should be in range 2.5Ghz to 5Ghz
|
|
*/
|
|
static const struct imx_fracn_gppll_rate_table fracn_tbl[] = {
|
|
PLL_FRACN_GP(650000000U, 162, 50, 100, 0, 6),
|
|
PLL_FRACN_GP(594000000U, 198, 0, 1, 0, 8),
|
|
PLL_FRACN_GP(560000000U, 140, 0, 1, 0, 6),
|
|
PLL_FRACN_GP(498000000U, 166, 0, 1, 0, 8),
|
|
PLL_FRACN_GP(484000000U, 121, 0, 1, 0, 6),
|
|
PLL_FRACN_GP(445333333U, 167, 0, 1, 0, 9),
|
|
PLL_FRACN_GP(400000000U, 200, 0, 1, 0, 12),
|
|
PLL_FRACN_GP(393216000U, 163, 84, 100, 0, 10),
|
|
PLL_FRACN_GP(300000000U, 150, 0, 1, 0, 12)
|
|
};
|
|
|
|
struct imx_fracn_gppll_clk imx_fracn_gppll = {
|
|
.rate_table = fracn_tbl,
|
|
.rate_count = ARRAY_SIZE(fracn_tbl),
|
|
};
|
|
|
|
/*
|
|
* Fvco = (Fref / rdiv) * MFI
|
|
* Fout = Fvco / odiv
|
|
* The (Fref / rdiv) should be in range 20MHz to 40MHz
|
|
* The Fvco should be in range 2.5Ghz to 5Ghz
|
|
*/
|
|
static const struct imx_fracn_gppll_rate_table int_tbl[] = {
|
|
PLL_FRACN_GP_INTEGER(1700000000U, 141, 1, 2),
|
|
PLL_FRACN_GP_INTEGER(1400000000U, 175, 1, 3),
|
|
PLL_FRACN_GP_INTEGER(900000000U, 150, 1, 4),
|
|
};
|
|
|
|
struct imx_fracn_gppll_clk imx_fracn_gppll_integer = {
|
|
.rate_table = int_tbl,
|
|
.rate_count = ARRAY_SIZE(int_tbl),
|
|
};
|
|
|
|
#define to_clk_fracn_gppll(_clk) container_of(_clk, struct clk_fracn_gppll, clk)
|
|
|
|
static const struct imx_fracn_gppll_rate_table *
|
|
imx_get_pll_settings(struct clk_fracn_gppll *pll, unsigned long rate)
|
|
{
|
|
const struct imx_fracn_gppll_rate_table *rate_table = pll->rate_table;
|
|
int i;
|
|
|
|
for (i = 0; i < pll->rate_count; i++)
|
|
if (rate == rate_table[i].rate)
|
|
return &rate_table[i];
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static unsigned long clk_fracn_gppll_round_rate(struct clk *clk, unsigned long rate)
|
|
{
|
|
struct clk_fracn_gppll *pll = to_clk_fracn_gppll(clk);
|
|
const struct imx_fracn_gppll_rate_table *rate_table = pll->rate_table;
|
|
int i;
|
|
|
|
/* Assuming rate_table is in descending order */
|
|
for (i = 0; i < pll->rate_count; i++)
|
|
if (rate >= rate_table[i].rate)
|
|
return rate_table[i].rate;
|
|
|
|
/* return minimum supported value */
|
|
return rate_table[pll->rate_count - 1].rate;
|
|
}
|
|
|
|
static unsigned long clk_fracn_gppll_recalc_rate(struct clk *clk)
|
|
{
|
|
struct clk_fracn_gppll *pll = to_clk_fracn_gppll(clk);
|
|
const struct imx_fracn_gppll_rate_table *rate_table = pll->rate_table;
|
|
u32 pll_numerator, pll_denominator, pll_div;
|
|
u32 mfi, mfn, mfd, rdiv, odiv;
|
|
u64 fvco = clk_get_parent_rate(clk);
|
|
long rate = 0;
|
|
int i;
|
|
|
|
pll_numerator = readl_relaxed(pll->base + PLL_NUMERATOR);
|
|
mfn = FIELD_GET(PLL_MFN_MASK, pll_numerator);
|
|
|
|
pll_denominator = readl_relaxed(pll->base + PLL_DENOMINATOR);
|
|
mfd = FIELD_GET(PLL_MFD_MASK, pll_denominator);
|
|
|
|
pll_div = readl_relaxed(pll->base + PLL_DIV);
|
|
mfi = FIELD_GET(PLL_MFI_MASK, pll_div);
|
|
|
|
rdiv = FIELD_GET(PLL_RDIV_MASK, pll_div);
|
|
odiv = FIELD_GET(PLL_ODIV_MASK, pll_div);
|
|
|
|
/*
|
|
* Sometimes, the recalculated rate has deviation due to
|
|
* the frac part. So find the accurate pll rate from the table
|
|
* first, if no match rate in the table, use the rate calculated
|
|
* from the equation below.
|
|
*/
|
|
for (i = 0; i < pll->rate_count; i++) {
|
|
if (rate_table[i].mfn == mfn && rate_table[i].mfi == mfi &&
|
|
rate_table[i].mfd == mfd && rate_table[i].rdiv == rdiv &&
|
|
rate_table[i].odiv == odiv)
|
|
rate = rate_table[i].rate;
|
|
}
|
|
|
|
if (rate)
|
|
return (unsigned long)rate;
|
|
|
|
if (!rdiv)
|
|
rdiv = rdiv + 1;
|
|
|
|
switch (odiv) {
|
|
case 0:
|
|
odiv = 2;
|
|
break;
|
|
case 1:
|
|
odiv = 3;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (pll->flags & CLK_FRACN_GPPLL_INTEGER) {
|
|
/* Fvco = (Fref / rdiv) * MFI */
|
|
fvco = fvco * mfi;
|
|
do_div(fvco, rdiv * odiv);
|
|
} else {
|
|
/* Fvco = (Fref / rdiv) * (MFI + MFN / MFD) */
|
|
fvco = fvco * mfi * mfd + fvco * mfn;
|
|
do_div(fvco, mfd * rdiv * odiv);
|
|
}
|
|
|
|
return (unsigned long)fvco;
|
|
}
|
|
|
|
static int clk_fracn_gppll_wait_lock(struct clk_fracn_gppll *pll)
|
|
{
|
|
u32 val;
|
|
|
|
return readl_poll_timeout(pll->base + PLL_STATUS, val,
|
|
val & LOCK_STATUS, LOCK_TIMEOUT_US);
|
|
}
|
|
|
|
static ulong clk_fracn_gppll_set_rate(struct clk *clk, unsigned long drate)
|
|
{
|
|
struct clk_fracn_gppll *pll = to_clk_fracn_gppll(clk);
|
|
const struct imx_fracn_gppll_rate_table *rate;
|
|
u32 tmp, pll_div, ana_mfn;
|
|
int ret;
|
|
|
|
rate = imx_get_pll_settings(pll, drate);
|
|
|
|
/* Hardware control select disable. PLL is control by register */
|
|
tmp = readl_relaxed(pll->base + PLL_CTRL);
|
|
tmp &= ~HW_CTRL_SEL;
|
|
writel_relaxed(tmp, pll->base + PLL_CTRL);
|
|
|
|
/* Disable output */
|
|
tmp = readl_relaxed(pll->base + PLL_CTRL);
|
|
tmp &= ~CLKMUX_EN;
|
|
writel_relaxed(tmp, pll->base + PLL_CTRL);
|
|
|
|
/* Power Down */
|
|
tmp &= ~POWERUP_MASK;
|
|
writel_relaxed(tmp, pll->base + PLL_CTRL);
|
|
|
|
/* Disable BYPASS */
|
|
tmp &= ~CLKMUX_BYPASS;
|
|
writel_relaxed(tmp, pll->base + PLL_CTRL);
|
|
|
|
pll_div = FIELD_PREP(PLL_RDIV_MASK, rate->rdiv) | rate->odiv |
|
|
FIELD_PREP(PLL_MFI_MASK, rate->mfi);
|
|
writel_relaxed(pll_div, pll->base + PLL_DIV);
|
|
if (pll->flags & CLK_FRACN_GPPLL_FRACN) {
|
|
writel_relaxed(rate->mfd, pll->base + PLL_DENOMINATOR);
|
|
writel_relaxed(FIELD_PREP(PLL_MFN_MASK, rate->mfn), pll->base + PLL_NUMERATOR);
|
|
}
|
|
|
|
/* Wait for 5us according to fracn mode pll doc */
|
|
udelay(5);
|
|
|
|
/* Enable Powerup */
|
|
tmp |= POWERUP_MASK;
|
|
writel_relaxed(tmp, pll->base + PLL_CTRL);
|
|
|
|
/* Wait Lock */
|
|
ret = clk_fracn_gppll_wait_lock(pll);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* Enable output */
|
|
tmp |= CLKMUX_EN;
|
|
writel_relaxed(tmp, pll->base + PLL_CTRL);
|
|
|
|
ana_mfn = readl_relaxed(pll->base + PLL_STATUS);
|
|
ana_mfn = FIELD_GET(PLL_MFN_MASK, ana_mfn);
|
|
|
|
WARN(ana_mfn != rate->mfn, "ana_mfn != rate->mfn\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int clk_fracn_gppll_prepare(struct clk *clk)
|
|
{
|
|
struct clk_fracn_gppll *pll = to_clk_fracn_gppll(clk);
|
|
u32 val;
|
|
int ret;
|
|
|
|
val = readl_relaxed(pll->base + PLL_CTRL);
|
|
if (val & POWERUP_MASK)
|
|
return 0;
|
|
|
|
val |= CLKMUX_BYPASS;
|
|
writel_relaxed(val, pll->base + PLL_CTRL);
|
|
|
|
val |= POWERUP_MASK;
|
|
writel_relaxed(val, pll->base + PLL_CTRL);
|
|
|
|
val |= CLKMUX_EN;
|
|
writel_relaxed(val, pll->base + PLL_CTRL);
|
|
|
|
ret = clk_fracn_gppll_wait_lock(pll);
|
|
if (ret)
|
|
return ret;
|
|
|
|
val &= ~CLKMUX_BYPASS;
|
|
writel_relaxed(val, pll->base + PLL_CTRL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int clk_fracn_gppll_unprepare(struct clk *clk)
|
|
{
|
|
struct clk_fracn_gppll *pll = to_clk_fracn_gppll(dev_get_clk_ptr(clk->dev));
|
|
u32 val;
|
|
|
|
val = readl_relaxed(pll->base + PLL_CTRL);
|
|
val &= ~POWERUP_MASK;
|
|
writel_relaxed(val, pll->base + PLL_CTRL);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct clk_ops clk_fracn_gppll_ops = {
|
|
.enable = clk_fracn_gppll_prepare,
|
|
.disable = clk_fracn_gppll_unprepare,
|
|
.get_rate = clk_fracn_gppll_recalc_rate,
|
|
.set_rate = clk_fracn_gppll_set_rate,
|
|
.round_rate = clk_fracn_gppll_round_rate,
|
|
};
|
|
|
|
static struct clk *_imx_clk_fracn_gppll(const char *name, const char *parent_name,
|
|
void __iomem *base,
|
|
const struct imx_fracn_gppll_clk *pll_clk,
|
|
u32 pll_flags)
|
|
{
|
|
struct clk_fracn_gppll *pll;
|
|
struct clk *clk;
|
|
int ret;
|
|
|
|
pll = kzalloc(sizeof(*pll), GFP_KERNEL);
|
|
if (!pll)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
pll->base = base;
|
|
pll->rate_table = pll_clk->rate_table;
|
|
pll->rate_count = pll_clk->rate_count;
|
|
pll->flags = pll_flags;
|
|
|
|
clk = &pll->clk;
|
|
|
|
ret = clk_register(clk, UBOOT_DM_CLK_IMX_FRACN_GPPLL,
|
|
name, parent_name);
|
|
if (ret) {
|
|
pr_err("%s: failed to register pll %s %d\n", __func__, name, ret);
|
|
kfree(pll);
|
|
return ERR_PTR(ret);
|
|
}
|
|
|
|
return clk;
|
|
}
|
|
|
|
struct clk *imx_clk_fracn_gppll(const char *name, const char *parent_name, void __iomem *base,
|
|
const struct imx_fracn_gppll_clk *pll_clk)
|
|
{
|
|
return _imx_clk_fracn_gppll(name, parent_name, base, pll_clk, CLK_FRACN_GPPLL_FRACN);
|
|
}
|
|
|
|
struct clk *imx_clk_fracn_gppll_integer(const char *name, const char *parent_name,
|
|
void __iomem *base,
|
|
const struct imx_fracn_gppll_clk *pll_clk)
|
|
{
|
|
return _imx_clk_fracn_gppll(name, parent_name, base, pll_clk, CLK_FRACN_GPPLL_INTEGER);
|
|
}
|
|
|
|
U_BOOT_DRIVER(clk_fracn_gppll) = {
|
|
.name = UBOOT_DM_CLK_IMX_FRACN_GPPLL,
|
|
.id = UCLASS_CLK,
|
|
.ops = &clk_fracn_gppll_ops,
|
|
.flags = DM_FLAG_PRE_RELOC,
|
|
};
|