// SPDX-License-Identifier: GPL-2.0-only /* * Copyright (c) 2022 MediaTek Inc. * Author: Edward-JW Yang <edward-jw.yang@mediatek.com> */ #include <linux/io.h> #include <linux/iopoll.h> #include "clk-mtk.h" #include "clk-pllfh.h" #include "clk-fhctl.h" #define PERCENT_TO_DDSLMT(dds, percent_m10) \ ((((dds) * (percent_m10)) >> 5) / 100) static const struct fhctl_offset fhctl_offset_v1 = { .offset_hp_en = 0x0, .offset_clk_con = 0x4, .offset_rst_con = 0x8, .offset_slope0 = 0xc, .offset_slope1 = 0x10, .offset_cfg = 0x0, .offset_updnlmt = 0x4, .offset_dds = 0x8, .offset_dvfs = 0xc, .offset_mon = 0x10, }; static const struct fhctl_offset fhctl_offset_v2 = { .offset_hp_en = 0x0, .offset_clk_con = 0x8, .offset_rst_con = 0xc, .offset_slope0 = 0x10, .offset_slope1 = 0x14, .offset_cfg = 0x0, .offset_updnlmt = 0x4, .offset_dds = 0x8, .offset_dvfs = 0xc, .offset_mon = 0x10, }; const struct fhctl_offset *fhctl_get_offset_table(enum fhctl_variant v) { switch (v) { case FHCTL_PLLFH_V1: return &fhctl_offset_v1; case FHCTL_PLLFH_V2: return &fhctl_offset_v2; default: return ERR_PTR(-EINVAL); }; } static void dump_hw(struct mtk_clk_pll *pll, struct fh_pll_regs *regs, const struct fh_pll_data *data) { pr_info("hp_en<%x>,clk_con<%x>,slope0<%x>,slope1<%x>\n", readl(regs->reg_hp_en), readl(regs->reg_clk_con), readl(regs->reg_slope0), readl(regs->reg_slope1)); pr_info("cfg<%x>,lmt<%x>,dds<%x>,dvfs<%x>,mon<%x>\n", readl(regs->reg_cfg), readl(regs->reg_updnlmt), readl(regs->reg_dds), readl(regs->reg_dvfs), readl(regs->reg_mon)); pr_info("pcw<%x>\n", readl(pll->pcw_addr)); } static int fhctl_set_ssc_regs(struct mtk_clk_pll *pll, struct fh_pll_regs *regs, const struct fh_pll_data *data, u32 rate) { u32 updnlmt_val, r; writel((readl(regs->reg_cfg) & ~(data->frddsx_en)), regs->reg_cfg); writel((readl(regs->reg_cfg) & ~(data->sfstrx_en)), regs->reg_cfg); writel((readl(regs->reg_cfg) & ~(data->fhctlx_en)), regs->reg_cfg); if (rate > 0) { /* Set the relative parameter registers (dt/df/upbnd/downbnd) */ r = readl(regs->reg_cfg); r &= ~(data->msk_frddsx_dys); r |= (data->df_val << (ffs(data->msk_frddsx_dys) - 1)); writel(r, regs->reg_cfg); r = readl(regs->reg_cfg); r &= ~(data->msk_frddsx_dts); r |= (data->dt_val << (ffs(data->msk_frddsx_dts) - 1)); writel(r, regs->reg_cfg); writel((readl(pll->pcw_addr) & data->dds_mask) | data->tgl_org, regs->reg_dds); /* Calculate UPDNLMT */ updnlmt_val = PERCENT_TO_DDSLMT((readl(regs->reg_dds) & data->dds_mask), rate) << data->updnlmt_shft; writel(updnlmt_val, regs->reg_updnlmt); writel(readl(regs->reg_hp_en) | BIT(data->fh_id), regs->reg_hp_en); /* Enable SSC */ writel(readl(regs->reg_cfg) | data->frddsx_en, regs->reg_cfg); /* Enable Hopping control */ writel(readl(regs->reg_cfg) | data->fhctlx_en, regs->reg_cfg); } else { /* Switch to APMIXEDSYS control */ writel(readl(regs->reg_hp_en) & ~BIT(data->fh_id), regs->reg_hp_en); /* Wait for DDS to be stable */ udelay(30); } return 0; } static int hopping_hw_flow(struct mtk_clk_pll *pll, struct fh_pll_regs *regs, const struct fh_pll_data *data, struct fh_pll_state *state, unsigned int new_dds) { u32 dds_mask = data->dds_mask; u32 mon_dds = 0; u32 con_pcw_tmp; int ret; if (state->ssc_rate) fhctl_set_ssc_regs(pll, regs, data, 0); writel((readl(pll->pcw_addr) & dds_mask) | data->tgl_org, regs->reg_dds); writel(readl(regs->reg_cfg) | data->sfstrx_en, regs->reg_cfg); writel(readl(regs->reg_cfg) | data->fhctlx_en, regs->reg_cfg); writel(data->slope0_value, regs->reg_slope0); writel(data->slope1_value, regs->reg_slope1); writel(readl(regs->reg_hp_en) | BIT(data->fh_id), regs->reg_hp_en); writel((new_dds) | (data->dvfs_tri), regs->reg_dvfs); /* Wait 1000 us until DDS stable */ ret = readl_poll_timeout_atomic(regs->reg_mon, mon_dds, (mon_dds & dds_mask) == new_dds, 10, 1000); if (ret) { pr_warn("%s: FHCTL hopping timeout\n", pll->data->name); dump_hw(pll, regs, data); } con_pcw_tmp = readl(pll->pcw_addr) & (~dds_mask); con_pcw_tmp = (con_pcw_tmp | (readl(regs->reg_mon) & dds_mask) | data->pcwchg); writel(con_pcw_tmp, pll->pcw_addr); writel(readl(regs->reg_hp_en) & ~BIT(data->fh_id), regs->reg_hp_en); if (state->ssc_rate) fhctl_set_ssc_regs(pll, regs, data, state->ssc_rate); return ret; } static unsigned int __get_postdiv(struct mtk_clk_pll *pll) { unsigned int regval; regval = readl(pll->pd_addr) >> pll->data->pd_shift; regval &= POSTDIV_MASK; return BIT(regval); } static void __set_postdiv(struct mtk_clk_pll *pll, unsigned int postdiv) { unsigned int regval; regval = readl(pll->pd_addr); regval &= ~(POSTDIV_MASK << pll->data->pd_shift); regval |= (ffs(postdiv) - 1) << pll->data->pd_shift; writel(regval, pll->pd_addr); } static int fhctl_hopping(struct mtk_fh *fh, unsigned int new_dds, unsigned int postdiv) { const struct fh_pll_data *data = &fh->pllfh_data->data; struct fh_pll_state *state = &fh->pllfh_data->state; struct fh_pll_regs *regs = &fh->regs; struct mtk_clk_pll *pll = &fh->clk_pll; spinlock_t *lock = fh->lock; unsigned int pll_postdiv; unsigned long flags = 0; int ret; if (postdiv) { pll_postdiv = __get_postdiv(pll); if (postdiv > pll_postdiv) __set_postdiv(pll, postdiv); } spin_lock_irqsave(lock, flags); ret = hopping_hw_flow(pll, regs, data, state, new_dds); spin_unlock_irqrestore(lock, flags); if (postdiv && postdiv < pll_postdiv) __set_postdiv(pll, postdiv); return ret; } static int fhctl_ssc_enable(struct mtk_fh *fh, u32 rate) { const struct fh_pll_data *data = &fh->pllfh_data->data; struct fh_pll_state *state = &fh->pllfh_data->state; struct fh_pll_regs *regs = &fh->regs; struct mtk_clk_pll *pll = &fh->clk_pll; spinlock_t *lock = fh->lock; unsigned long flags = 0; spin_lock_irqsave(lock, flags); fhctl_set_ssc_regs(pll, regs, data, rate); state->ssc_rate = rate; spin_unlock_irqrestore(lock, flags); return 0; } static const struct fh_operation fhctl_ops = { .hopping = fhctl_hopping, .ssc_enable = fhctl_ssc_enable, }; const struct fh_operation *fhctl_get_ops(void) { return &fhctl_ops; } void fhctl_hw_init(struct mtk_fh *fh) { const struct fh_pll_data data = fh->pllfh_data->data; struct fh_pll_state state = fh->pllfh_data->state; struct fh_pll_regs regs = fh->regs; u32 val; /* initial hw register */ val = readl(regs.reg_clk_con) | BIT(data.fh_id); writel(val, regs.reg_clk_con); val = readl(regs.reg_rst_con) & ~BIT(data.fh_id); writel(val, regs.reg_rst_con); val = readl(regs.reg_rst_con) | BIT(data.fh_id); writel(val, regs.reg_rst_con); writel(0x0, regs.reg_cfg); writel(0x0, regs.reg_updnlmt); writel(0x0, regs.reg_dds); /* enable ssc if needed */ if (state.ssc_rate) fh->ops->ssc_enable(fh, state.ssc_rate); }