162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci// Copyright (C) 2014 Broadcom Corporation 362306a36Sopenharmony_ci 462306a36Sopenharmony_ci#include <linux/kernel.h> 562306a36Sopenharmony_ci#include <linux/err.h> 662306a36Sopenharmony_ci#include <linux/clk-provider.h> 762306a36Sopenharmony_ci#include <linux/io.h> 862306a36Sopenharmony_ci#include <linux/of.h> 962306a36Sopenharmony_ci#include <linux/clkdev.h> 1062306a36Sopenharmony_ci#include <linux/of_address.h> 1162306a36Sopenharmony_ci#include <linux/delay.h> 1262306a36Sopenharmony_ci 1362306a36Sopenharmony_ci#include "clk-iproc.h" 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_ci#define PLL_VCO_HIGH_SHIFT 19 1662306a36Sopenharmony_ci#define PLL_VCO_LOW_SHIFT 30 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci/* 1962306a36Sopenharmony_ci * PLL MACRO_SELECT modes 0 to 5 choose pre-calculated PLL output frequencies 2062306a36Sopenharmony_ci * from a look-up table. Mode 7 allows user to manipulate PLL clock dividers 2162306a36Sopenharmony_ci */ 2262306a36Sopenharmony_ci#define PLL_USER_MODE 7 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_ci/* number of delay loops waiting for PLL to lock */ 2562306a36Sopenharmony_ci#define LOCK_DELAY 100 2662306a36Sopenharmony_ci 2762306a36Sopenharmony_ci/* number of VCO frequency bands */ 2862306a36Sopenharmony_ci#define NUM_FREQ_BANDS 8 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci#define NUM_KP_BANDS 3 3162306a36Sopenharmony_cienum kp_band { 3262306a36Sopenharmony_ci KP_BAND_MID = 0, 3362306a36Sopenharmony_ci KP_BAND_HIGH, 3462306a36Sopenharmony_ci KP_BAND_HIGH_HIGH 3562306a36Sopenharmony_ci}; 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_cistatic const unsigned int kp_table[NUM_KP_BANDS][NUM_FREQ_BANDS] = { 3862306a36Sopenharmony_ci { 5, 6, 6, 7, 7, 8, 9, 10 }, 3962306a36Sopenharmony_ci { 4, 4, 5, 5, 6, 7, 8, 9 }, 4062306a36Sopenharmony_ci { 4, 5, 5, 6, 7, 8, 9, 10 }, 4162306a36Sopenharmony_ci}; 4262306a36Sopenharmony_ci 4362306a36Sopenharmony_cistatic const unsigned long ref_freq_table[NUM_FREQ_BANDS][2] = { 4462306a36Sopenharmony_ci { 10000000, 12500000 }, 4562306a36Sopenharmony_ci { 12500000, 15000000 }, 4662306a36Sopenharmony_ci { 15000000, 20000000 }, 4762306a36Sopenharmony_ci { 20000000, 25000000 }, 4862306a36Sopenharmony_ci { 25000000, 50000000 }, 4962306a36Sopenharmony_ci { 50000000, 75000000 }, 5062306a36Sopenharmony_ci { 75000000, 100000000 }, 5162306a36Sopenharmony_ci { 100000000, 125000000 }, 5262306a36Sopenharmony_ci}; 5362306a36Sopenharmony_ci 5462306a36Sopenharmony_cienum vco_freq_range { 5562306a36Sopenharmony_ci VCO_LOW = 700000000U, 5662306a36Sopenharmony_ci VCO_MID = 1200000000U, 5762306a36Sopenharmony_ci VCO_HIGH = 2200000000U, 5862306a36Sopenharmony_ci VCO_HIGH_HIGH = 3100000000U, 5962306a36Sopenharmony_ci VCO_MAX = 4000000000U, 6062306a36Sopenharmony_ci}; 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_cistruct iproc_pll { 6362306a36Sopenharmony_ci void __iomem *status_base; 6462306a36Sopenharmony_ci void __iomem *control_base; 6562306a36Sopenharmony_ci void __iomem *pwr_base; 6662306a36Sopenharmony_ci void __iomem *asiu_base; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl; 6962306a36Sopenharmony_ci const struct iproc_pll_vco_param *vco_param; 7062306a36Sopenharmony_ci unsigned int num_vco_entries; 7162306a36Sopenharmony_ci}; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_cistruct iproc_clk { 7462306a36Sopenharmony_ci struct clk_hw hw; 7562306a36Sopenharmony_ci struct iproc_pll *pll; 7662306a36Sopenharmony_ci const struct iproc_clk_ctrl *ctrl; 7762306a36Sopenharmony_ci}; 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci#define to_iproc_clk(hw) container_of(hw, struct iproc_clk, hw) 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_cistatic int pll_calc_param(unsigned long target_rate, 8262306a36Sopenharmony_ci unsigned long parent_rate, 8362306a36Sopenharmony_ci struct iproc_pll_vco_param *vco_out) 8462306a36Sopenharmony_ci{ 8562306a36Sopenharmony_ci u64 ndiv_int, ndiv_frac, residual; 8662306a36Sopenharmony_ci 8762306a36Sopenharmony_ci ndiv_int = target_rate / parent_rate; 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_ci if (!ndiv_int || (ndiv_int > 255)) 9062306a36Sopenharmony_ci return -EINVAL; 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_ci residual = target_rate - (ndiv_int * parent_rate); 9362306a36Sopenharmony_ci residual <<= 20; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci /* 9662306a36Sopenharmony_ci * Add half of the divisor so the result will be rounded to closest 9762306a36Sopenharmony_ci * instead of rounded down. 9862306a36Sopenharmony_ci */ 9962306a36Sopenharmony_ci residual += (parent_rate / 2); 10062306a36Sopenharmony_ci ndiv_frac = div64_u64((u64)residual, (u64)parent_rate); 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci vco_out->ndiv_int = ndiv_int; 10362306a36Sopenharmony_ci vco_out->ndiv_frac = ndiv_frac; 10462306a36Sopenharmony_ci vco_out->pdiv = 1; 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_ci vco_out->rate = vco_out->ndiv_int * parent_rate; 10762306a36Sopenharmony_ci residual = (u64)vco_out->ndiv_frac * (u64)parent_rate; 10862306a36Sopenharmony_ci residual >>= 20; 10962306a36Sopenharmony_ci vco_out->rate += residual; 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci return 0; 11262306a36Sopenharmony_ci} 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_ci/* 11562306a36Sopenharmony_ci * Based on the target frequency, find a match from the VCO frequency parameter 11662306a36Sopenharmony_ci * table and return its index 11762306a36Sopenharmony_ci */ 11862306a36Sopenharmony_cistatic int pll_get_rate_index(struct iproc_pll *pll, unsigned int target_rate) 11962306a36Sopenharmony_ci{ 12062306a36Sopenharmony_ci int i; 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_ci for (i = 0; i < pll->num_vco_entries; i++) 12362306a36Sopenharmony_ci if (target_rate == pll->vco_param[i].rate) 12462306a36Sopenharmony_ci break; 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_ci if (i >= pll->num_vco_entries) 12762306a36Sopenharmony_ci return -EINVAL; 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_ci return i; 13062306a36Sopenharmony_ci} 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_cistatic int get_kp(unsigned long ref_freq, enum kp_band kp_index) 13362306a36Sopenharmony_ci{ 13462306a36Sopenharmony_ci int i; 13562306a36Sopenharmony_ci 13662306a36Sopenharmony_ci if (ref_freq < ref_freq_table[0][0]) 13762306a36Sopenharmony_ci return -EINVAL; 13862306a36Sopenharmony_ci 13962306a36Sopenharmony_ci for (i = 0; i < NUM_FREQ_BANDS; i++) { 14062306a36Sopenharmony_ci if (ref_freq >= ref_freq_table[i][0] && 14162306a36Sopenharmony_ci ref_freq < ref_freq_table[i][1]) 14262306a36Sopenharmony_ci return kp_table[kp_index][i]; 14362306a36Sopenharmony_ci } 14462306a36Sopenharmony_ci return -EINVAL; 14562306a36Sopenharmony_ci} 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_cistatic int pll_wait_for_lock(struct iproc_pll *pll) 14862306a36Sopenharmony_ci{ 14962306a36Sopenharmony_ci int i; 15062306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci for (i = 0; i < LOCK_DELAY; i++) { 15362306a36Sopenharmony_ci u32 val = readl(pll->status_base + ctrl->status.offset); 15462306a36Sopenharmony_ci 15562306a36Sopenharmony_ci if (val & (1 << ctrl->status.shift)) 15662306a36Sopenharmony_ci return 0; 15762306a36Sopenharmony_ci udelay(10); 15862306a36Sopenharmony_ci } 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci return -EIO; 16162306a36Sopenharmony_ci} 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_cistatic void iproc_pll_write(const struct iproc_pll *pll, void __iomem *base, 16462306a36Sopenharmony_ci const u32 offset, u32 val) 16562306a36Sopenharmony_ci{ 16662306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci writel(val, base + offset); 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci if (unlikely(ctrl->flags & IPROC_CLK_NEEDS_READ_BACK && 17162306a36Sopenharmony_ci (base == pll->status_base || base == pll->control_base))) 17262306a36Sopenharmony_ci val = readl(base + offset); 17362306a36Sopenharmony_ci} 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_cistatic void __pll_disable(struct iproc_pll *pll) 17662306a36Sopenharmony_ci{ 17762306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 17862306a36Sopenharmony_ci u32 val; 17962306a36Sopenharmony_ci 18062306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_ASIU) { 18162306a36Sopenharmony_ci val = readl(pll->asiu_base + ctrl->asiu.offset); 18262306a36Sopenharmony_ci val &= ~(1 << ctrl->asiu.en_shift); 18362306a36Sopenharmony_ci iproc_pll_write(pll, pll->asiu_base, ctrl->asiu.offset, val); 18462306a36Sopenharmony_ci } 18562306a36Sopenharmony_ci 18662306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_EMBED_PWRCTRL) { 18762306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->aon.offset); 18862306a36Sopenharmony_ci val |= bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift; 18962306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->aon.offset, val); 19062306a36Sopenharmony_ci } 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_ci if (pll->pwr_base) { 19362306a36Sopenharmony_ci /* latch input value so core power can be shut down */ 19462306a36Sopenharmony_ci val = readl(pll->pwr_base + ctrl->aon.offset); 19562306a36Sopenharmony_ci val |= 1 << ctrl->aon.iso_shift; 19662306a36Sopenharmony_ci iproc_pll_write(pll, pll->pwr_base, ctrl->aon.offset, val); 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ci /* power down the core */ 19962306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift); 20062306a36Sopenharmony_ci iproc_pll_write(pll, pll->pwr_base, ctrl->aon.offset, val); 20162306a36Sopenharmony_ci } 20262306a36Sopenharmony_ci} 20362306a36Sopenharmony_ci 20462306a36Sopenharmony_cistatic int __pll_enable(struct iproc_pll *pll) 20562306a36Sopenharmony_ci{ 20662306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 20762306a36Sopenharmony_ci u32 val; 20862306a36Sopenharmony_ci 20962306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_EMBED_PWRCTRL) { 21062306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->aon.offset); 21162306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift); 21262306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->aon.offset, val); 21362306a36Sopenharmony_ci } 21462306a36Sopenharmony_ci 21562306a36Sopenharmony_ci if (pll->pwr_base) { 21662306a36Sopenharmony_ci /* power up the PLL and make sure it's not latched */ 21762306a36Sopenharmony_ci val = readl(pll->pwr_base + ctrl->aon.offset); 21862306a36Sopenharmony_ci val |= bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift; 21962306a36Sopenharmony_ci val &= ~(1 << ctrl->aon.iso_shift); 22062306a36Sopenharmony_ci iproc_pll_write(pll, pll->pwr_base, ctrl->aon.offset, val); 22162306a36Sopenharmony_ci } 22262306a36Sopenharmony_ci 22362306a36Sopenharmony_ci /* certain PLLs also need to be ungated from the ASIU top level */ 22462306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_ASIU) { 22562306a36Sopenharmony_ci val = readl(pll->asiu_base + ctrl->asiu.offset); 22662306a36Sopenharmony_ci val |= (1 << ctrl->asiu.en_shift); 22762306a36Sopenharmony_ci iproc_pll_write(pll, pll->asiu_base, ctrl->asiu.offset, val); 22862306a36Sopenharmony_ci } 22962306a36Sopenharmony_ci 23062306a36Sopenharmony_ci return 0; 23162306a36Sopenharmony_ci} 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_cistatic void __pll_put_in_reset(struct iproc_pll *pll) 23462306a36Sopenharmony_ci{ 23562306a36Sopenharmony_ci u32 val; 23662306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 23762306a36Sopenharmony_ci const struct iproc_pll_reset_ctrl *reset = &ctrl->reset; 23862306a36Sopenharmony_ci 23962306a36Sopenharmony_ci val = readl(pll->control_base + reset->offset); 24062306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_RESET_ACTIVE_LOW) 24162306a36Sopenharmony_ci val |= BIT(reset->reset_shift) | BIT(reset->p_reset_shift); 24262306a36Sopenharmony_ci else 24362306a36Sopenharmony_ci val &= ~(BIT(reset->reset_shift) | BIT(reset->p_reset_shift)); 24462306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, reset->offset, val); 24562306a36Sopenharmony_ci} 24662306a36Sopenharmony_ci 24762306a36Sopenharmony_cistatic void __pll_bring_out_reset(struct iproc_pll *pll, unsigned int kp, 24862306a36Sopenharmony_ci unsigned int ka, unsigned int ki) 24962306a36Sopenharmony_ci{ 25062306a36Sopenharmony_ci u32 val; 25162306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 25262306a36Sopenharmony_ci const struct iproc_pll_reset_ctrl *reset = &ctrl->reset; 25362306a36Sopenharmony_ci const struct iproc_pll_dig_filter_ctrl *dig_filter = &ctrl->dig_filter; 25462306a36Sopenharmony_ci 25562306a36Sopenharmony_ci val = readl(pll->control_base + dig_filter->offset); 25662306a36Sopenharmony_ci val &= ~(bit_mask(dig_filter->ki_width) << dig_filter->ki_shift | 25762306a36Sopenharmony_ci bit_mask(dig_filter->kp_width) << dig_filter->kp_shift | 25862306a36Sopenharmony_ci bit_mask(dig_filter->ka_width) << dig_filter->ka_shift); 25962306a36Sopenharmony_ci val |= ki << dig_filter->ki_shift | kp << dig_filter->kp_shift | 26062306a36Sopenharmony_ci ka << dig_filter->ka_shift; 26162306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, dig_filter->offset, val); 26262306a36Sopenharmony_ci 26362306a36Sopenharmony_ci val = readl(pll->control_base + reset->offset); 26462306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_RESET_ACTIVE_LOW) 26562306a36Sopenharmony_ci val &= ~(BIT(reset->reset_shift) | BIT(reset->p_reset_shift)); 26662306a36Sopenharmony_ci else 26762306a36Sopenharmony_ci val |= BIT(reset->reset_shift) | BIT(reset->p_reset_shift); 26862306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, reset->offset, val); 26962306a36Sopenharmony_ci} 27062306a36Sopenharmony_ci 27162306a36Sopenharmony_ci/* 27262306a36Sopenharmony_ci * Determines if the change to be applied to the PLL is minor (just an update 27362306a36Sopenharmony_ci * or the fractional divider). If so, then we can avoid going through a 27462306a36Sopenharmony_ci * disruptive reset and lock sequence. 27562306a36Sopenharmony_ci */ 27662306a36Sopenharmony_cistatic bool pll_fractional_change_only(struct iproc_pll *pll, 27762306a36Sopenharmony_ci struct iproc_pll_vco_param *vco) 27862306a36Sopenharmony_ci{ 27962306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 28062306a36Sopenharmony_ci u32 val; 28162306a36Sopenharmony_ci u32 ndiv_int; 28262306a36Sopenharmony_ci unsigned int pdiv; 28362306a36Sopenharmony_ci 28462306a36Sopenharmony_ci /* PLL needs to be locked */ 28562306a36Sopenharmony_ci val = readl(pll->status_base + ctrl->status.offset); 28662306a36Sopenharmony_ci if ((val & (1 << ctrl->status.shift)) == 0) 28762306a36Sopenharmony_ci return false; 28862306a36Sopenharmony_ci 28962306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->ndiv_int.offset); 29062306a36Sopenharmony_ci ndiv_int = (val >> ctrl->ndiv_int.shift) & 29162306a36Sopenharmony_ci bit_mask(ctrl->ndiv_int.width); 29262306a36Sopenharmony_ci 29362306a36Sopenharmony_ci if (ndiv_int != vco->ndiv_int) 29462306a36Sopenharmony_ci return false; 29562306a36Sopenharmony_ci 29662306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->pdiv.offset); 29762306a36Sopenharmony_ci pdiv = (val >> ctrl->pdiv.shift) & bit_mask(ctrl->pdiv.width); 29862306a36Sopenharmony_ci 29962306a36Sopenharmony_ci if (pdiv != vco->pdiv) 30062306a36Sopenharmony_ci return false; 30162306a36Sopenharmony_ci 30262306a36Sopenharmony_ci return true; 30362306a36Sopenharmony_ci} 30462306a36Sopenharmony_ci 30562306a36Sopenharmony_cistatic int pll_set_rate(struct iproc_clk *clk, struct iproc_pll_vco_param *vco, 30662306a36Sopenharmony_ci unsigned long parent_rate) 30762306a36Sopenharmony_ci{ 30862306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 30962306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 31062306a36Sopenharmony_ci int ka = 0, ki, kp, ret; 31162306a36Sopenharmony_ci unsigned long rate = vco->rate; 31262306a36Sopenharmony_ci u32 val; 31362306a36Sopenharmony_ci enum kp_band kp_index; 31462306a36Sopenharmony_ci unsigned long ref_freq; 31562306a36Sopenharmony_ci const char *clk_name = clk_hw_get_name(&clk->hw); 31662306a36Sopenharmony_ci 31762306a36Sopenharmony_ci /* 31862306a36Sopenharmony_ci * reference frequency = parent frequency / PDIV 31962306a36Sopenharmony_ci * If PDIV = 0, then it becomes a multiplier (x2) 32062306a36Sopenharmony_ci */ 32162306a36Sopenharmony_ci if (vco->pdiv == 0) 32262306a36Sopenharmony_ci ref_freq = parent_rate * 2; 32362306a36Sopenharmony_ci else 32462306a36Sopenharmony_ci ref_freq = parent_rate / vco->pdiv; 32562306a36Sopenharmony_ci 32662306a36Sopenharmony_ci /* determine Ki and Kp index based on target VCO frequency */ 32762306a36Sopenharmony_ci if (rate >= VCO_LOW && rate < VCO_HIGH) { 32862306a36Sopenharmony_ci ki = 4; 32962306a36Sopenharmony_ci kp_index = KP_BAND_MID; 33062306a36Sopenharmony_ci } else if (rate >= VCO_HIGH && rate < VCO_HIGH_HIGH) { 33162306a36Sopenharmony_ci ki = 3; 33262306a36Sopenharmony_ci kp_index = KP_BAND_HIGH; 33362306a36Sopenharmony_ci } else if (rate >= VCO_HIGH_HIGH && rate < VCO_MAX) { 33462306a36Sopenharmony_ci ki = 3; 33562306a36Sopenharmony_ci kp_index = KP_BAND_HIGH_HIGH; 33662306a36Sopenharmony_ci } else { 33762306a36Sopenharmony_ci pr_err("%s: pll: %s has invalid rate: %lu\n", __func__, 33862306a36Sopenharmony_ci clk_name, rate); 33962306a36Sopenharmony_ci return -EINVAL; 34062306a36Sopenharmony_ci } 34162306a36Sopenharmony_ci 34262306a36Sopenharmony_ci kp = get_kp(ref_freq, kp_index); 34362306a36Sopenharmony_ci if (kp < 0) { 34462306a36Sopenharmony_ci pr_err("%s: pll: %s has invalid kp\n", __func__, clk_name); 34562306a36Sopenharmony_ci return kp; 34662306a36Sopenharmony_ci } 34762306a36Sopenharmony_ci 34862306a36Sopenharmony_ci ret = __pll_enable(pll); 34962306a36Sopenharmony_ci if (ret) { 35062306a36Sopenharmony_ci pr_err("%s: pll: %s fails to enable\n", __func__, clk_name); 35162306a36Sopenharmony_ci return ret; 35262306a36Sopenharmony_ci } 35362306a36Sopenharmony_ci 35462306a36Sopenharmony_ci if (pll_fractional_change_only(clk->pll, vco)) { 35562306a36Sopenharmony_ci /* program fractional part of NDIV */ 35662306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_HAS_NDIV_FRAC) { 35762306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->ndiv_frac.offset); 35862306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->ndiv_frac.width) << 35962306a36Sopenharmony_ci ctrl->ndiv_frac.shift); 36062306a36Sopenharmony_ci val |= vco->ndiv_frac << ctrl->ndiv_frac.shift; 36162306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, 36262306a36Sopenharmony_ci ctrl->ndiv_frac.offset, val); 36362306a36Sopenharmony_ci return 0; 36462306a36Sopenharmony_ci } 36562306a36Sopenharmony_ci } 36662306a36Sopenharmony_ci 36762306a36Sopenharmony_ci /* put PLL in reset */ 36862306a36Sopenharmony_ci __pll_put_in_reset(pll); 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_ci /* set PLL in user mode before modifying PLL controls */ 37162306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_USER_MODE_ON) { 37262306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->macro_mode.offset); 37362306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->macro_mode.width) << 37462306a36Sopenharmony_ci ctrl->macro_mode.shift); 37562306a36Sopenharmony_ci val |= PLL_USER_MODE << ctrl->macro_mode.shift; 37662306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, 37762306a36Sopenharmony_ci ctrl->macro_mode.offset, val); 37862306a36Sopenharmony_ci } 37962306a36Sopenharmony_ci 38062306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->vco_ctrl.u_offset, 0); 38162306a36Sopenharmony_ci 38262306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->vco_ctrl.l_offset); 38362306a36Sopenharmony_ci 38462306a36Sopenharmony_ci if (rate >= VCO_LOW && rate < VCO_MID) 38562306a36Sopenharmony_ci val |= (1 << PLL_VCO_LOW_SHIFT); 38662306a36Sopenharmony_ci 38762306a36Sopenharmony_ci if (rate < VCO_HIGH) 38862306a36Sopenharmony_ci val &= ~(1 << PLL_VCO_HIGH_SHIFT); 38962306a36Sopenharmony_ci else 39062306a36Sopenharmony_ci val |= (1 << PLL_VCO_HIGH_SHIFT); 39162306a36Sopenharmony_ci 39262306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->vco_ctrl.l_offset, val); 39362306a36Sopenharmony_ci 39462306a36Sopenharmony_ci /* program integer part of NDIV */ 39562306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->ndiv_int.offset); 39662306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->ndiv_int.width) << ctrl->ndiv_int.shift); 39762306a36Sopenharmony_ci val |= vco->ndiv_int << ctrl->ndiv_int.shift; 39862306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->ndiv_int.offset, val); 39962306a36Sopenharmony_ci 40062306a36Sopenharmony_ci /* program fractional part of NDIV */ 40162306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_HAS_NDIV_FRAC) { 40262306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->ndiv_frac.offset); 40362306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->ndiv_frac.width) << 40462306a36Sopenharmony_ci ctrl->ndiv_frac.shift); 40562306a36Sopenharmony_ci val |= vco->ndiv_frac << ctrl->ndiv_frac.shift; 40662306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->ndiv_frac.offset, 40762306a36Sopenharmony_ci val); 40862306a36Sopenharmony_ci } 40962306a36Sopenharmony_ci 41062306a36Sopenharmony_ci /* program PDIV */ 41162306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->pdiv.offset); 41262306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->pdiv.width) << ctrl->pdiv.shift); 41362306a36Sopenharmony_ci val |= vco->pdiv << ctrl->pdiv.shift; 41462306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->pdiv.offset, val); 41562306a36Sopenharmony_ci 41662306a36Sopenharmony_ci __pll_bring_out_reset(pll, kp, ka, ki); 41762306a36Sopenharmony_ci 41862306a36Sopenharmony_ci ret = pll_wait_for_lock(pll); 41962306a36Sopenharmony_ci if (ret < 0) { 42062306a36Sopenharmony_ci pr_err("%s: pll: %s failed to lock\n", __func__, clk_name); 42162306a36Sopenharmony_ci return ret; 42262306a36Sopenharmony_ci } 42362306a36Sopenharmony_ci 42462306a36Sopenharmony_ci return 0; 42562306a36Sopenharmony_ci} 42662306a36Sopenharmony_ci 42762306a36Sopenharmony_cistatic int iproc_pll_enable(struct clk_hw *hw) 42862306a36Sopenharmony_ci{ 42962306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 43062306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 43162306a36Sopenharmony_ci 43262306a36Sopenharmony_ci return __pll_enable(pll); 43362306a36Sopenharmony_ci} 43462306a36Sopenharmony_ci 43562306a36Sopenharmony_cistatic void iproc_pll_disable(struct clk_hw *hw) 43662306a36Sopenharmony_ci{ 43762306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 43862306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 43962306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 44062306a36Sopenharmony_ci 44162306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_AON) 44262306a36Sopenharmony_ci return; 44362306a36Sopenharmony_ci 44462306a36Sopenharmony_ci __pll_disable(pll); 44562306a36Sopenharmony_ci} 44662306a36Sopenharmony_ci 44762306a36Sopenharmony_cistatic unsigned long iproc_pll_recalc_rate(struct clk_hw *hw, 44862306a36Sopenharmony_ci unsigned long parent_rate) 44962306a36Sopenharmony_ci{ 45062306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 45162306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 45262306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 45362306a36Sopenharmony_ci u32 val; 45462306a36Sopenharmony_ci u64 ndiv, ndiv_int, ndiv_frac; 45562306a36Sopenharmony_ci unsigned int pdiv; 45662306a36Sopenharmony_ci unsigned long rate; 45762306a36Sopenharmony_ci 45862306a36Sopenharmony_ci if (parent_rate == 0) 45962306a36Sopenharmony_ci return 0; 46062306a36Sopenharmony_ci 46162306a36Sopenharmony_ci /* PLL needs to be locked */ 46262306a36Sopenharmony_ci val = readl(pll->status_base + ctrl->status.offset); 46362306a36Sopenharmony_ci if ((val & (1 << ctrl->status.shift)) == 0) 46462306a36Sopenharmony_ci return 0; 46562306a36Sopenharmony_ci 46662306a36Sopenharmony_ci /* 46762306a36Sopenharmony_ci * PLL output frequency = 46862306a36Sopenharmony_ci * 46962306a36Sopenharmony_ci * ((ndiv_int + ndiv_frac / 2^20) * (parent clock rate / pdiv) 47062306a36Sopenharmony_ci */ 47162306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->ndiv_int.offset); 47262306a36Sopenharmony_ci ndiv_int = (val >> ctrl->ndiv_int.shift) & 47362306a36Sopenharmony_ci bit_mask(ctrl->ndiv_int.width); 47462306a36Sopenharmony_ci ndiv = ndiv_int << 20; 47562306a36Sopenharmony_ci 47662306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_HAS_NDIV_FRAC) { 47762306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->ndiv_frac.offset); 47862306a36Sopenharmony_ci ndiv_frac = (val >> ctrl->ndiv_frac.shift) & 47962306a36Sopenharmony_ci bit_mask(ctrl->ndiv_frac.width); 48062306a36Sopenharmony_ci ndiv += ndiv_frac; 48162306a36Sopenharmony_ci } 48262306a36Sopenharmony_ci 48362306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->pdiv.offset); 48462306a36Sopenharmony_ci pdiv = (val >> ctrl->pdiv.shift) & bit_mask(ctrl->pdiv.width); 48562306a36Sopenharmony_ci 48662306a36Sopenharmony_ci rate = (ndiv * parent_rate) >> 20; 48762306a36Sopenharmony_ci 48862306a36Sopenharmony_ci if (pdiv == 0) 48962306a36Sopenharmony_ci rate *= 2; 49062306a36Sopenharmony_ci else 49162306a36Sopenharmony_ci rate /= pdiv; 49262306a36Sopenharmony_ci 49362306a36Sopenharmony_ci return rate; 49462306a36Sopenharmony_ci} 49562306a36Sopenharmony_ci 49662306a36Sopenharmony_cistatic int iproc_pll_determine_rate(struct clk_hw *hw, 49762306a36Sopenharmony_ci struct clk_rate_request *req) 49862306a36Sopenharmony_ci{ 49962306a36Sopenharmony_ci unsigned int i; 50062306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 50162306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 50262306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 50362306a36Sopenharmony_ci unsigned long diff, best_diff; 50462306a36Sopenharmony_ci unsigned int best_idx = 0; 50562306a36Sopenharmony_ci int ret; 50662306a36Sopenharmony_ci 50762306a36Sopenharmony_ci if (req->rate == 0 || req->best_parent_rate == 0) 50862306a36Sopenharmony_ci return -EINVAL; 50962306a36Sopenharmony_ci 51062306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_CALC_PARAM) { 51162306a36Sopenharmony_ci struct iproc_pll_vco_param vco_param; 51262306a36Sopenharmony_ci 51362306a36Sopenharmony_ci ret = pll_calc_param(req->rate, req->best_parent_rate, 51462306a36Sopenharmony_ci &vco_param); 51562306a36Sopenharmony_ci if (ret) 51662306a36Sopenharmony_ci return ret; 51762306a36Sopenharmony_ci 51862306a36Sopenharmony_ci req->rate = vco_param.rate; 51962306a36Sopenharmony_ci return 0; 52062306a36Sopenharmony_ci } 52162306a36Sopenharmony_ci 52262306a36Sopenharmony_ci if (!pll->vco_param) 52362306a36Sopenharmony_ci return -EINVAL; 52462306a36Sopenharmony_ci 52562306a36Sopenharmony_ci best_diff = ULONG_MAX; 52662306a36Sopenharmony_ci for (i = 0; i < pll->num_vco_entries; i++) { 52762306a36Sopenharmony_ci diff = abs(req->rate - pll->vco_param[i].rate); 52862306a36Sopenharmony_ci if (diff <= best_diff) { 52962306a36Sopenharmony_ci best_diff = diff; 53062306a36Sopenharmony_ci best_idx = i; 53162306a36Sopenharmony_ci } 53262306a36Sopenharmony_ci /* break now if perfect match */ 53362306a36Sopenharmony_ci if (diff == 0) 53462306a36Sopenharmony_ci break; 53562306a36Sopenharmony_ci } 53662306a36Sopenharmony_ci 53762306a36Sopenharmony_ci req->rate = pll->vco_param[best_idx].rate; 53862306a36Sopenharmony_ci 53962306a36Sopenharmony_ci return 0; 54062306a36Sopenharmony_ci} 54162306a36Sopenharmony_ci 54262306a36Sopenharmony_cistatic int iproc_pll_set_rate(struct clk_hw *hw, unsigned long rate, 54362306a36Sopenharmony_ci unsigned long parent_rate) 54462306a36Sopenharmony_ci{ 54562306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 54662306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 54762306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 54862306a36Sopenharmony_ci struct iproc_pll_vco_param vco_param; 54962306a36Sopenharmony_ci int rate_index, ret; 55062306a36Sopenharmony_ci 55162306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_CALC_PARAM) { 55262306a36Sopenharmony_ci ret = pll_calc_param(rate, parent_rate, &vco_param); 55362306a36Sopenharmony_ci if (ret) 55462306a36Sopenharmony_ci return ret; 55562306a36Sopenharmony_ci } else { 55662306a36Sopenharmony_ci rate_index = pll_get_rate_index(pll, rate); 55762306a36Sopenharmony_ci if (rate_index < 0) 55862306a36Sopenharmony_ci return rate_index; 55962306a36Sopenharmony_ci 56062306a36Sopenharmony_ci vco_param = pll->vco_param[rate_index]; 56162306a36Sopenharmony_ci } 56262306a36Sopenharmony_ci 56362306a36Sopenharmony_ci ret = pll_set_rate(clk, &vco_param, parent_rate); 56462306a36Sopenharmony_ci return ret; 56562306a36Sopenharmony_ci} 56662306a36Sopenharmony_ci 56762306a36Sopenharmony_cistatic const struct clk_ops iproc_pll_ops = { 56862306a36Sopenharmony_ci .enable = iproc_pll_enable, 56962306a36Sopenharmony_ci .disable = iproc_pll_disable, 57062306a36Sopenharmony_ci .recalc_rate = iproc_pll_recalc_rate, 57162306a36Sopenharmony_ci .determine_rate = iproc_pll_determine_rate, 57262306a36Sopenharmony_ci .set_rate = iproc_pll_set_rate, 57362306a36Sopenharmony_ci}; 57462306a36Sopenharmony_ci 57562306a36Sopenharmony_cistatic int iproc_clk_enable(struct clk_hw *hw) 57662306a36Sopenharmony_ci{ 57762306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 57862306a36Sopenharmony_ci const struct iproc_clk_ctrl *ctrl = clk->ctrl; 57962306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 58062306a36Sopenharmony_ci u32 val; 58162306a36Sopenharmony_ci 58262306a36Sopenharmony_ci /* channel enable is active low */ 58362306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->enable.offset); 58462306a36Sopenharmony_ci val &= ~(1 << ctrl->enable.enable_shift); 58562306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->enable.offset, val); 58662306a36Sopenharmony_ci 58762306a36Sopenharmony_ci /* also make sure channel is not held */ 58862306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->enable.offset); 58962306a36Sopenharmony_ci val &= ~(1 << ctrl->enable.hold_shift); 59062306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->enable.offset, val); 59162306a36Sopenharmony_ci 59262306a36Sopenharmony_ci return 0; 59362306a36Sopenharmony_ci} 59462306a36Sopenharmony_ci 59562306a36Sopenharmony_cistatic void iproc_clk_disable(struct clk_hw *hw) 59662306a36Sopenharmony_ci{ 59762306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 59862306a36Sopenharmony_ci const struct iproc_clk_ctrl *ctrl = clk->ctrl; 59962306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 60062306a36Sopenharmony_ci u32 val; 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_AON) 60362306a36Sopenharmony_ci return; 60462306a36Sopenharmony_ci 60562306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->enable.offset); 60662306a36Sopenharmony_ci val |= 1 << ctrl->enable.enable_shift; 60762306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->enable.offset, val); 60862306a36Sopenharmony_ci} 60962306a36Sopenharmony_ci 61062306a36Sopenharmony_cistatic unsigned long iproc_clk_recalc_rate(struct clk_hw *hw, 61162306a36Sopenharmony_ci unsigned long parent_rate) 61262306a36Sopenharmony_ci{ 61362306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 61462306a36Sopenharmony_ci const struct iproc_clk_ctrl *ctrl = clk->ctrl; 61562306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 61662306a36Sopenharmony_ci u32 val; 61762306a36Sopenharmony_ci unsigned int mdiv; 61862306a36Sopenharmony_ci unsigned long rate; 61962306a36Sopenharmony_ci 62062306a36Sopenharmony_ci if (parent_rate == 0) 62162306a36Sopenharmony_ci return 0; 62262306a36Sopenharmony_ci 62362306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->mdiv.offset); 62462306a36Sopenharmony_ci mdiv = (val >> ctrl->mdiv.shift) & bit_mask(ctrl->mdiv.width); 62562306a36Sopenharmony_ci if (mdiv == 0) 62662306a36Sopenharmony_ci mdiv = 256; 62762306a36Sopenharmony_ci 62862306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_MCLK_DIV_BY_2) 62962306a36Sopenharmony_ci rate = parent_rate / (mdiv * 2); 63062306a36Sopenharmony_ci else 63162306a36Sopenharmony_ci rate = parent_rate / mdiv; 63262306a36Sopenharmony_ci 63362306a36Sopenharmony_ci return rate; 63462306a36Sopenharmony_ci} 63562306a36Sopenharmony_ci 63662306a36Sopenharmony_cistatic int iproc_clk_determine_rate(struct clk_hw *hw, 63762306a36Sopenharmony_ci struct clk_rate_request *req) 63862306a36Sopenharmony_ci{ 63962306a36Sopenharmony_ci unsigned int bestdiv; 64062306a36Sopenharmony_ci 64162306a36Sopenharmony_ci if (req->rate == 0) 64262306a36Sopenharmony_ci return -EINVAL; 64362306a36Sopenharmony_ci if (req->rate == req->best_parent_rate) 64462306a36Sopenharmony_ci return 0; 64562306a36Sopenharmony_ci 64662306a36Sopenharmony_ci bestdiv = DIV_ROUND_CLOSEST(req->best_parent_rate, req->rate); 64762306a36Sopenharmony_ci if (bestdiv < 2) 64862306a36Sopenharmony_ci req->rate = req->best_parent_rate; 64962306a36Sopenharmony_ci 65062306a36Sopenharmony_ci if (bestdiv > 256) 65162306a36Sopenharmony_ci bestdiv = 256; 65262306a36Sopenharmony_ci 65362306a36Sopenharmony_ci req->rate = req->best_parent_rate / bestdiv; 65462306a36Sopenharmony_ci 65562306a36Sopenharmony_ci return 0; 65662306a36Sopenharmony_ci} 65762306a36Sopenharmony_ci 65862306a36Sopenharmony_cistatic int iproc_clk_set_rate(struct clk_hw *hw, unsigned long rate, 65962306a36Sopenharmony_ci unsigned long parent_rate) 66062306a36Sopenharmony_ci{ 66162306a36Sopenharmony_ci struct iproc_clk *clk = to_iproc_clk(hw); 66262306a36Sopenharmony_ci const struct iproc_clk_ctrl *ctrl = clk->ctrl; 66362306a36Sopenharmony_ci struct iproc_pll *pll = clk->pll; 66462306a36Sopenharmony_ci u32 val; 66562306a36Sopenharmony_ci unsigned int div; 66662306a36Sopenharmony_ci 66762306a36Sopenharmony_ci if (rate == 0 || parent_rate == 0) 66862306a36Sopenharmony_ci return -EINVAL; 66962306a36Sopenharmony_ci 67062306a36Sopenharmony_ci div = DIV_ROUND_CLOSEST(parent_rate, rate); 67162306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_MCLK_DIV_BY_2) 67262306a36Sopenharmony_ci div /= 2; 67362306a36Sopenharmony_ci 67462306a36Sopenharmony_ci if (div > 256) 67562306a36Sopenharmony_ci return -EINVAL; 67662306a36Sopenharmony_ci 67762306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->mdiv.offset); 67862306a36Sopenharmony_ci if (div == 256) { 67962306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->mdiv.width) << ctrl->mdiv.shift); 68062306a36Sopenharmony_ci } else { 68162306a36Sopenharmony_ci val &= ~(bit_mask(ctrl->mdiv.width) << ctrl->mdiv.shift); 68262306a36Sopenharmony_ci val |= div << ctrl->mdiv.shift; 68362306a36Sopenharmony_ci } 68462306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->mdiv.offset, val); 68562306a36Sopenharmony_ci 68662306a36Sopenharmony_ci return 0; 68762306a36Sopenharmony_ci} 68862306a36Sopenharmony_ci 68962306a36Sopenharmony_cistatic const struct clk_ops iproc_clk_ops = { 69062306a36Sopenharmony_ci .enable = iproc_clk_enable, 69162306a36Sopenharmony_ci .disable = iproc_clk_disable, 69262306a36Sopenharmony_ci .recalc_rate = iproc_clk_recalc_rate, 69362306a36Sopenharmony_ci .determine_rate = iproc_clk_determine_rate, 69462306a36Sopenharmony_ci .set_rate = iproc_clk_set_rate, 69562306a36Sopenharmony_ci}; 69662306a36Sopenharmony_ci 69762306a36Sopenharmony_ci/* 69862306a36Sopenharmony_ci * Some PLLs require the PLL SW override bit to be set before changes can be 69962306a36Sopenharmony_ci * applied to the PLL 70062306a36Sopenharmony_ci */ 70162306a36Sopenharmony_cistatic void iproc_pll_sw_cfg(struct iproc_pll *pll) 70262306a36Sopenharmony_ci{ 70362306a36Sopenharmony_ci const struct iproc_pll_ctrl *ctrl = pll->ctrl; 70462306a36Sopenharmony_ci 70562306a36Sopenharmony_ci if (ctrl->flags & IPROC_CLK_PLL_NEEDS_SW_CFG) { 70662306a36Sopenharmony_ci u32 val; 70762306a36Sopenharmony_ci 70862306a36Sopenharmony_ci val = readl(pll->control_base + ctrl->sw_ctrl.offset); 70962306a36Sopenharmony_ci val |= BIT(ctrl->sw_ctrl.shift); 71062306a36Sopenharmony_ci iproc_pll_write(pll, pll->control_base, ctrl->sw_ctrl.offset, 71162306a36Sopenharmony_ci val); 71262306a36Sopenharmony_ci } 71362306a36Sopenharmony_ci} 71462306a36Sopenharmony_ci 71562306a36Sopenharmony_civoid iproc_pll_clk_setup(struct device_node *node, 71662306a36Sopenharmony_ci const struct iproc_pll_ctrl *pll_ctrl, 71762306a36Sopenharmony_ci const struct iproc_pll_vco_param *vco, 71862306a36Sopenharmony_ci unsigned int num_vco_entries, 71962306a36Sopenharmony_ci const struct iproc_clk_ctrl *clk_ctrl, 72062306a36Sopenharmony_ci unsigned int num_clks) 72162306a36Sopenharmony_ci{ 72262306a36Sopenharmony_ci int i, ret; 72362306a36Sopenharmony_ci struct iproc_pll *pll; 72462306a36Sopenharmony_ci struct iproc_clk *iclk; 72562306a36Sopenharmony_ci struct clk_init_data init; 72662306a36Sopenharmony_ci const char *parent_name; 72762306a36Sopenharmony_ci struct iproc_clk *iclk_array; 72862306a36Sopenharmony_ci struct clk_hw_onecell_data *clk_data; 72962306a36Sopenharmony_ci const char *clk_name; 73062306a36Sopenharmony_ci 73162306a36Sopenharmony_ci if (WARN_ON(!pll_ctrl) || WARN_ON(!clk_ctrl)) 73262306a36Sopenharmony_ci return; 73362306a36Sopenharmony_ci 73462306a36Sopenharmony_ci pll = kzalloc(sizeof(*pll), GFP_KERNEL); 73562306a36Sopenharmony_ci if (WARN_ON(!pll)) 73662306a36Sopenharmony_ci return; 73762306a36Sopenharmony_ci 73862306a36Sopenharmony_ci clk_data = kzalloc(struct_size(clk_data, hws, num_clks), GFP_KERNEL); 73962306a36Sopenharmony_ci if (WARN_ON(!clk_data)) 74062306a36Sopenharmony_ci goto err_clk_data; 74162306a36Sopenharmony_ci clk_data->num = num_clks; 74262306a36Sopenharmony_ci 74362306a36Sopenharmony_ci iclk_array = kcalloc(num_clks, sizeof(struct iproc_clk), GFP_KERNEL); 74462306a36Sopenharmony_ci if (WARN_ON(!iclk_array)) 74562306a36Sopenharmony_ci goto err_clks; 74662306a36Sopenharmony_ci 74762306a36Sopenharmony_ci pll->control_base = of_iomap(node, 0); 74862306a36Sopenharmony_ci if (WARN_ON(!pll->control_base)) 74962306a36Sopenharmony_ci goto err_pll_iomap; 75062306a36Sopenharmony_ci 75162306a36Sopenharmony_ci /* Some SoCs do not require the pwr_base, thus failing is not fatal */ 75262306a36Sopenharmony_ci pll->pwr_base = of_iomap(node, 1); 75362306a36Sopenharmony_ci 75462306a36Sopenharmony_ci /* some PLLs require gating control at the top ASIU level */ 75562306a36Sopenharmony_ci if (pll_ctrl->flags & IPROC_CLK_PLL_ASIU) { 75662306a36Sopenharmony_ci pll->asiu_base = of_iomap(node, 2); 75762306a36Sopenharmony_ci if (WARN_ON(!pll->asiu_base)) 75862306a36Sopenharmony_ci goto err_asiu_iomap; 75962306a36Sopenharmony_ci } 76062306a36Sopenharmony_ci 76162306a36Sopenharmony_ci if (pll_ctrl->flags & IPROC_CLK_PLL_SPLIT_STAT_CTRL) { 76262306a36Sopenharmony_ci /* Some SoCs have a split status/control. If this does not 76362306a36Sopenharmony_ci * exist, assume they are unified. 76462306a36Sopenharmony_ci */ 76562306a36Sopenharmony_ci pll->status_base = of_iomap(node, 2); 76662306a36Sopenharmony_ci if (!pll->status_base) 76762306a36Sopenharmony_ci goto err_status_iomap; 76862306a36Sopenharmony_ci } else 76962306a36Sopenharmony_ci pll->status_base = pll->control_base; 77062306a36Sopenharmony_ci 77162306a36Sopenharmony_ci /* initialize and register the PLL itself */ 77262306a36Sopenharmony_ci pll->ctrl = pll_ctrl; 77362306a36Sopenharmony_ci 77462306a36Sopenharmony_ci iclk = &iclk_array[0]; 77562306a36Sopenharmony_ci iclk->pll = pll; 77662306a36Sopenharmony_ci 77762306a36Sopenharmony_ci ret = of_property_read_string_index(node, "clock-output-names", 77862306a36Sopenharmony_ci 0, &clk_name); 77962306a36Sopenharmony_ci if (WARN_ON(ret)) 78062306a36Sopenharmony_ci goto err_pll_register; 78162306a36Sopenharmony_ci 78262306a36Sopenharmony_ci init.name = clk_name; 78362306a36Sopenharmony_ci init.ops = &iproc_pll_ops; 78462306a36Sopenharmony_ci init.flags = 0; 78562306a36Sopenharmony_ci parent_name = of_clk_get_parent_name(node, 0); 78662306a36Sopenharmony_ci init.parent_names = (parent_name ? &parent_name : NULL); 78762306a36Sopenharmony_ci init.num_parents = (parent_name ? 1 : 0); 78862306a36Sopenharmony_ci iclk->hw.init = &init; 78962306a36Sopenharmony_ci 79062306a36Sopenharmony_ci if (vco) { 79162306a36Sopenharmony_ci pll->num_vco_entries = num_vco_entries; 79262306a36Sopenharmony_ci pll->vco_param = vco; 79362306a36Sopenharmony_ci } 79462306a36Sopenharmony_ci 79562306a36Sopenharmony_ci iproc_pll_sw_cfg(pll); 79662306a36Sopenharmony_ci 79762306a36Sopenharmony_ci ret = clk_hw_register(NULL, &iclk->hw); 79862306a36Sopenharmony_ci if (WARN_ON(ret)) 79962306a36Sopenharmony_ci goto err_pll_register; 80062306a36Sopenharmony_ci 80162306a36Sopenharmony_ci clk_data->hws[0] = &iclk->hw; 80262306a36Sopenharmony_ci parent_name = clk_name; 80362306a36Sopenharmony_ci 80462306a36Sopenharmony_ci /* now initialize and register all leaf clocks */ 80562306a36Sopenharmony_ci for (i = 1; i < num_clks; i++) { 80662306a36Sopenharmony_ci memset(&init, 0, sizeof(init)); 80762306a36Sopenharmony_ci 80862306a36Sopenharmony_ci ret = of_property_read_string_index(node, "clock-output-names", 80962306a36Sopenharmony_ci i, &clk_name); 81062306a36Sopenharmony_ci if (WARN_ON(ret)) 81162306a36Sopenharmony_ci goto err_clk_register; 81262306a36Sopenharmony_ci 81362306a36Sopenharmony_ci iclk = &iclk_array[i]; 81462306a36Sopenharmony_ci iclk->pll = pll; 81562306a36Sopenharmony_ci iclk->ctrl = &clk_ctrl[i]; 81662306a36Sopenharmony_ci 81762306a36Sopenharmony_ci init.name = clk_name; 81862306a36Sopenharmony_ci init.ops = &iproc_clk_ops; 81962306a36Sopenharmony_ci init.flags = 0; 82062306a36Sopenharmony_ci init.parent_names = (parent_name ? &parent_name : NULL); 82162306a36Sopenharmony_ci init.num_parents = (parent_name ? 1 : 0); 82262306a36Sopenharmony_ci iclk->hw.init = &init; 82362306a36Sopenharmony_ci 82462306a36Sopenharmony_ci ret = clk_hw_register(NULL, &iclk->hw); 82562306a36Sopenharmony_ci if (WARN_ON(ret)) 82662306a36Sopenharmony_ci goto err_clk_register; 82762306a36Sopenharmony_ci 82862306a36Sopenharmony_ci clk_data->hws[i] = &iclk->hw; 82962306a36Sopenharmony_ci } 83062306a36Sopenharmony_ci 83162306a36Sopenharmony_ci ret = of_clk_add_hw_provider(node, of_clk_hw_onecell_get, clk_data); 83262306a36Sopenharmony_ci if (WARN_ON(ret)) 83362306a36Sopenharmony_ci goto err_clk_register; 83462306a36Sopenharmony_ci 83562306a36Sopenharmony_ci return; 83662306a36Sopenharmony_ci 83762306a36Sopenharmony_cierr_clk_register: 83862306a36Sopenharmony_ci while (--i >= 0) 83962306a36Sopenharmony_ci clk_hw_unregister(clk_data->hws[i]); 84062306a36Sopenharmony_ci 84162306a36Sopenharmony_cierr_pll_register: 84262306a36Sopenharmony_ci if (pll->status_base != pll->control_base) 84362306a36Sopenharmony_ci iounmap(pll->status_base); 84462306a36Sopenharmony_ci 84562306a36Sopenharmony_cierr_status_iomap: 84662306a36Sopenharmony_ci if (pll->asiu_base) 84762306a36Sopenharmony_ci iounmap(pll->asiu_base); 84862306a36Sopenharmony_ci 84962306a36Sopenharmony_cierr_asiu_iomap: 85062306a36Sopenharmony_ci if (pll->pwr_base) 85162306a36Sopenharmony_ci iounmap(pll->pwr_base); 85262306a36Sopenharmony_ci 85362306a36Sopenharmony_ci iounmap(pll->control_base); 85462306a36Sopenharmony_ci 85562306a36Sopenharmony_cierr_pll_iomap: 85662306a36Sopenharmony_ci kfree(iclk_array); 85762306a36Sopenharmony_ci 85862306a36Sopenharmony_cierr_clks: 85962306a36Sopenharmony_ci kfree(clk_data); 86062306a36Sopenharmony_ci 86162306a36Sopenharmony_cierr_clk_data: 86262306a36Sopenharmony_ci kfree(pll); 86362306a36Sopenharmony_ci} 864