Commit e40c7e3c authored by Jerome Brunet's avatar Jerome Brunet

clk: meson: clk-pll: add enable bit

Add the enable the bit of the pll clocks.
These pll clocks may be disabled but we can't model this as an external
gate since the pll needs to lock when enabled.

Adding this bit allows to drop the poke of the first register of PLL.
This will be useful to model the different components of the pll using
generic clocks elements
Acked-by: default avatarNeil Armstrong <narmstrong@baylibre.com>
Reviewed-by: default avatarMartin Blumenstingl <martin.blumenstingl@googlemail.com>
Tested-by: default avatarMartin Blumenstingl <martin.blumenstingl@googlemail.com>
Signed-off-by: default avatarJerome Brunet <jbrunet@baylibre.com>
parent 5b394b2d
...@@ -24,6 +24,11 @@ static DEFINE_SPINLOCK(meson_clk_lock); ...@@ -24,6 +24,11 @@ static DEFINE_SPINLOCK(meson_clk_lock);
static struct clk_regmap axg_fixed_pll = { static struct clk_regmap axg_fixed_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_MPLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_MPLL_CNTL, .reg_off = HHI_MPLL_CNTL,
.shift = 0, .shift = 0,
...@@ -65,6 +70,11 @@ static struct clk_regmap axg_fixed_pll = { ...@@ -65,6 +70,11 @@ static struct clk_regmap axg_fixed_pll = {
static struct clk_regmap axg_sys_pll = { static struct clk_regmap axg_sys_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_SYS_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_SYS_PLL_CNTL, .reg_off = HHI_SYS_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -197,11 +207,15 @@ static const struct reg_sequence axg_gp0_init_regs[] = { ...@@ -197,11 +207,15 @@ static const struct reg_sequence axg_gp0_init_regs[] = {
{ .reg = HHI_GP0_PLL_CNTL3, .def = 0x0a59a288 }, { .reg = HHI_GP0_PLL_CNTL3, .def = 0x0a59a288 },
{ .reg = HHI_GP0_PLL_CNTL4, .def = 0xc000004d }, { .reg = HHI_GP0_PLL_CNTL4, .def = 0xc000004d },
{ .reg = HHI_GP0_PLL_CNTL5, .def = 0x00078000 }, { .reg = HHI_GP0_PLL_CNTL5, .def = 0x00078000 },
{ .reg = HHI_GP0_PLL_CNTL, .def = 0x40010250 },
}; };
static struct clk_regmap axg_gp0_pll = { static struct clk_regmap axg_gp0_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_GP0_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_GP0_PLL_CNTL, .reg_off = HHI_GP0_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -250,11 +264,15 @@ static const struct reg_sequence axg_hifi_init_regs[] = { ...@@ -250,11 +264,15 @@ static const struct reg_sequence axg_hifi_init_regs[] = {
{ .reg = HHI_HIFI_PLL_CNTL3, .def = 0x0a6a3a88 }, { .reg = HHI_HIFI_PLL_CNTL3, .def = 0x0a6a3a88 },
{ .reg = HHI_HIFI_PLL_CNTL4, .def = 0xc000004d }, { .reg = HHI_HIFI_PLL_CNTL4, .def = 0xc000004d },
{ .reg = HHI_HIFI_PLL_CNTL5, .def = 0x00058000 }, { .reg = HHI_HIFI_PLL_CNTL5, .def = 0x00058000 },
{ .reg = HHI_HIFI_PLL_CNTL, .def = 0x40010250 },
}; };
static struct clk_regmap axg_hifi_pll = { static struct clk_regmap axg_hifi_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_HIFI_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_HIFI_PLL_CNTL, .reg_off = HHI_HIFI_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -637,7 +655,6 @@ static const struct pll_rate_table axg_pcie_pll_rate_table[] = { ...@@ -637,7 +655,6 @@ static const struct pll_rate_table axg_pcie_pll_rate_table[] = {
}; };
static const struct reg_sequence axg_pcie_init_regs[] = { static const struct reg_sequence axg_pcie_init_regs[] = {
{ .reg = HHI_PCIE_PLL_CNTL, .def = 0x400106c8 },
{ .reg = HHI_PCIE_PLL_CNTL1, .def = 0x0084a2aa }, { .reg = HHI_PCIE_PLL_CNTL1, .def = 0x0084a2aa },
{ .reg = HHI_PCIE_PLL_CNTL2, .def = 0xb75020be }, { .reg = HHI_PCIE_PLL_CNTL2, .def = 0xb75020be },
{ .reg = HHI_PCIE_PLL_CNTL3, .def = 0x0a47488e }, { .reg = HHI_PCIE_PLL_CNTL3, .def = 0x0a47488e },
...@@ -648,6 +665,11 @@ static const struct reg_sequence axg_pcie_init_regs[] = { ...@@ -648,6 +665,11 @@ static const struct reg_sequence axg_pcie_init_regs[] = {
static struct clk_regmap axg_pcie_pll = { static struct clk_regmap axg_pcie_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_PCIE_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_PCIE_PLL_CNTL, .reg_off = HHI_PCIE_PLL_CNTL,
.shift = 0, .shift = 0,
......
...@@ -185,12 +185,45 @@ static void meson_clk_pll_init(struct clk_hw *hw) ...@@ -185,12 +185,45 @@ static void meson_clk_pll_init(struct clk_hw *hw)
} }
} }
static int meson_clk_pll_enable(struct clk_hw *hw)
{
struct clk_regmap *clk = to_clk_regmap(hw);
struct meson_clk_pll_data *pll = meson_clk_pll_data(clk);
/* Make sure the pll is in reset */
meson_parm_write(clk->map, &pll->rst, 1);
/* Enable the pll */
meson_parm_write(clk->map, &pll->en, 1);
/* Take the pll out reset */
meson_parm_write(clk->map, &pll->rst, 0);
if (meson_clk_pll_wait_lock(hw))
return -EIO;
return 0;
}
static void meson_clk_pll_disable(struct clk_hw *hw)
{
struct clk_regmap *clk = to_clk_regmap(hw);
struct meson_clk_pll_data *pll = meson_clk_pll_data(clk);
/* Put the pll is in reset */
meson_parm_write(clk->map, &pll->rst, 1);
/* Disable the pll */
meson_parm_write(clk->map, &pll->en, 0);
}
static int meson_clk_pll_set_rate(struct clk_hw *hw, unsigned long rate, static int meson_clk_pll_set_rate(struct clk_hw *hw, unsigned long rate,
unsigned long parent_rate) unsigned long parent_rate)
{ {
struct clk_regmap *clk = to_clk_regmap(hw); struct clk_regmap *clk = to_clk_regmap(hw);
struct meson_clk_pll_data *pll = meson_clk_pll_data(clk); struct meson_clk_pll_data *pll = meson_clk_pll_data(clk);
const struct pll_rate_table *pllt; const struct pll_rate_table *pllt;
unsigned int enabled;
unsigned long old_rate; unsigned long old_rate;
u16 frac = 0; u16 frac = 0;
...@@ -203,8 +236,9 @@ static int meson_clk_pll_set_rate(struct clk_hw *hw, unsigned long rate, ...@@ -203,8 +236,9 @@ static int meson_clk_pll_set_rate(struct clk_hw *hw, unsigned long rate,
if (!pllt) if (!pllt)
return -EINVAL; return -EINVAL;
/* Put the pll in reset to write the params */ enabled = meson_parm_read(clk->map, &pll->en);
meson_parm_write(clk->map, &pll->rst, 1); if (enabled)
meson_clk_pll_disable(hw);
meson_parm_write(clk->map, &pll->n, pllt->n); meson_parm_write(clk->map, &pll->n, pllt->n);
meson_parm_write(clk->map, &pll->m, pllt->m); meson_parm_write(clk->map, &pll->m, pllt->m);
...@@ -221,10 +255,11 @@ static int meson_clk_pll_set_rate(struct clk_hw *hw, unsigned long rate, ...@@ -221,10 +255,11 @@ static int meson_clk_pll_set_rate(struct clk_hw *hw, unsigned long rate,
meson_parm_write(clk->map, &pll->frac, frac); meson_parm_write(clk->map, &pll->frac, frac);
} }
/* make sure the reset is cleared at this point */ /* If the pll is stopped, bail out now */
meson_parm_write(clk->map, &pll->rst, 0); if (!enabled)
return 0;
if (meson_clk_pll_wait_lock(hw)) { if (meson_clk_pll_enable(hw)) {
pr_warn("%s: pll did not lock, trying to restore old rate %lu\n", pr_warn("%s: pll did not lock, trying to restore old rate %lu\n",
__func__, old_rate); __func__, old_rate);
/* /*
...@@ -244,6 +279,8 @@ const struct clk_ops meson_clk_pll_ops = { ...@@ -244,6 +279,8 @@ const struct clk_ops meson_clk_pll_ops = {
.recalc_rate = meson_clk_pll_recalc_rate, .recalc_rate = meson_clk_pll_recalc_rate,
.round_rate = meson_clk_pll_round_rate, .round_rate = meson_clk_pll_round_rate,
.set_rate = meson_clk_pll_set_rate, .set_rate = meson_clk_pll_set_rate,
.enable = meson_clk_pll_enable,
.disable = meson_clk_pll_disable
}; };
const struct clk_ops meson_clk_pll_ro_ops = { const struct clk_ops meson_clk_pll_ro_ops = {
......
...@@ -63,6 +63,7 @@ struct pll_rate_table { ...@@ -63,6 +63,7 @@ struct pll_rate_table {
#define CLK_MESON_PLL_ROUND_CLOSEST BIT(0) #define CLK_MESON_PLL_ROUND_CLOSEST BIT(0)
struct meson_clk_pll_data { struct meson_clk_pll_data {
struct parm en;
struct parm m; struct parm m;
struct parm n; struct parm n;
struct parm frac; struct parm frac;
......
...@@ -177,6 +177,11 @@ static const struct pll_rate_table gxl_gp0_pll_rate_table[] = { ...@@ -177,6 +177,11 @@ static const struct pll_rate_table gxl_gp0_pll_rate_table[] = {
static struct clk_regmap gxbb_fixed_pll = { static struct clk_regmap gxbb_fixed_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_MPLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_MPLL_CNTL, .reg_off = HHI_MPLL_CNTL,
.shift = 0, .shift = 0,
...@@ -230,6 +235,11 @@ static struct clk_fixed_factor gxbb_hdmi_pll_pre_mult = { ...@@ -230,6 +235,11 @@ static struct clk_fixed_factor gxbb_hdmi_pll_pre_mult = {
static struct clk_regmap gxbb_hdmi_pll = { static struct clk_regmap gxbb_hdmi_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_HDMI_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_HDMI_PLL_CNTL, .reg_off = HHI_HDMI_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -282,6 +292,11 @@ static struct clk_regmap gxbb_hdmi_pll = { ...@@ -282,6 +292,11 @@ static struct clk_regmap gxbb_hdmi_pll = {
static struct clk_regmap gxl_hdmi_pll = { static struct clk_regmap gxl_hdmi_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_HDMI_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_HDMI_PLL_CNTL, .reg_off = HHI_HDMI_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -340,6 +355,11 @@ static struct clk_regmap gxl_hdmi_pll = { ...@@ -340,6 +355,11 @@ static struct clk_regmap gxl_hdmi_pll = {
static struct clk_regmap gxbb_sys_pll = { static struct clk_regmap gxbb_sys_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_SYS_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_SYS_PLL_CNTL, .reg_off = HHI_SYS_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -379,11 +399,15 @@ static const struct reg_sequence gxbb_gp0_init_regs[] = { ...@@ -379,11 +399,15 @@ static const struct reg_sequence gxbb_gp0_init_regs[] = {
{ .reg = HHI_GP0_PLL_CNTL2, .def = 0x69c80000 }, { .reg = HHI_GP0_PLL_CNTL2, .def = 0x69c80000 },
{ .reg = HHI_GP0_PLL_CNTL3, .def = 0x0a5590c4 }, { .reg = HHI_GP0_PLL_CNTL3, .def = 0x0a5590c4 },
{ .reg = HHI_GP0_PLL_CNTL4, .def = 0x0000500d }, { .reg = HHI_GP0_PLL_CNTL4, .def = 0x0000500d },
{ .reg = HHI_GP0_PLL_CNTL, .def = 0x4a000228 },
}; };
static struct clk_regmap gxbb_gp0_pll = { static struct clk_regmap gxbb_gp0_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_GP0_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_GP0_PLL_CNTL, .reg_off = HHI_GP0_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -428,11 +452,15 @@ static const struct reg_sequence gxl_gp0_init_regs[] = { ...@@ -428,11 +452,15 @@ static const struct reg_sequence gxl_gp0_init_regs[] = {
{ .reg = HHI_GP0_PLL_CNTL3, .def = 0x0a59a288 }, { .reg = HHI_GP0_PLL_CNTL3, .def = 0x0a59a288 },
{ .reg = HHI_GP0_PLL_CNTL4, .def = 0xc000004d }, { .reg = HHI_GP0_PLL_CNTL4, .def = 0xc000004d },
{ .reg = HHI_GP0_PLL_CNTL5, .def = 0x00078000 }, { .reg = HHI_GP0_PLL_CNTL5, .def = 0x00078000 },
{ .reg = HHI_GP0_PLL_CNTL, .def = 0x40010250 },
}; };
static struct clk_regmap gxl_gp0_pll = { static struct clk_regmap gxl_gp0_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_GP0_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_GP0_PLL_CNTL, .reg_off = HHI_GP0_PLL_CNTL,
.shift = 0, .shift = 0,
......
...@@ -96,6 +96,11 @@ static struct clk_fixed_rate meson8b_xtal = { ...@@ -96,6 +96,11 @@ static struct clk_fixed_rate meson8b_xtal = {
static struct clk_regmap meson8b_fixed_pll = { static struct clk_regmap meson8b_fixed_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_MPLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_MPLL_CNTL, .reg_off = HHI_MPLL_CNTL,
.shift = 0, .shift = 0,
...@@ -138,6 +143,11 @@ static struct clk_regmap meson8b_fixed_pll = { ...@@ -138,6 +143,11 @@ static struct clk_regmap meson8b_fixed_pll = {
static struct clk_regmap meson8b_vid_pll = { static struct clk_regmap meson8b_vid_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_VID_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_VID_PLL_CNTL, .reg_off = HHI_VID_PLL_CNTL,
.shift = 0, .shift = 0,
...@@ -175,6 +185,11 @@ static struct clk_regmap meson8b_vid_pll = { ...@@ -175,6 +185,11 @@ static struct clk_regmap meson8b_vid_pll = {
static struct clk_regmap meson8b_sys_pll = { static struct clk_regmap meson8b_sys_pll = {
.data = &(struct meson_clk_pll_data){ .data = &(struct meson_clk_pll_data){
.en = {
.reg_off = HHI_SYS_PLL_CNTL,
.shift = 30,
.width = 1,
},
.m = { .m = {
.reg_off = HHI_SYS_PLL_CNTL, .reg_off = HHI_SYS_PLL_CNTL,
.shift = 0, .shift = 0,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment