cregit-Linux how code gets into the kernel

Release 4.7 drivers/clk/bcm/clk-iproc-pll.c

Directory: drivers/clk/bcm
/*
 * Copyright (C) 2014 Broadcom Corporation
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation version 2.
 *
 * This program is distributed "as is" WITHOUT ANY WARRANTY of any
 * kind, whether express or implied; without even the implied warranty
 * of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 */

#include <linux/kernel.h>
#include <linux/err.h>
#include <linux/clk-provider.h>
#include <linux/io.h>
#include <linux/of.h>
#include <linux/clkdev.h>
#include <linux/of_address.h>
#include <linux/delay.h>

#include "clk-iproc.h"


#define PLL_VCO_HIGH_SHIFT 19

#define PLL_VCO_LOW_SHIFT  30

/*
 * PLL MACRO_SELECT modes 0 to 5 choose pre-calculated PLL output frequencies
 * from a look-up table. Mode 7 allows user to manipulate PLL clock dividers
 */

#define PLL_USER_MODE 7

/* number of delay loops waiting for PLL to lock */

#define LOCK_DELAY 100

/* number of VCO frequency bands */

#define NUM_FREQ_BANDS 8


#define NUM_KP_BANDS 3

enum kp_band {
	
KP_BAND_MID = 0,
	
KP_BAND_HIGH,
	
KP_BAND_HIGH_HIGH
};


static const unsigned int kp_table[NUM_KP_BANDS][NUM_FREQ_BANDS] = {
	{ 5, 6, 6, 7, 7, 8, 9, 10 },
	{ 4, 4, 5, 5, 6, 7, 8, 9  },
	{ 4, 5, 5, 6, 7, 8, 9, 10 },
};


static const unsigned long ref_freq_table[NUM_FREQ_BANDS][2] = {
	{ 10000000,  12500000  },
	{ 12500000,  15000000  },
	{ 15000000,  20000000  },
	{ 20000000,  25000000  },
	{ 25000000,  50000000  },
	{ 50000000,  75000000  },
	{ 75000000,  100000000 },
	{ 100000000, 125000000 },
};


enum vco_freq_range {
	
VCO_LOW       = 700000000U,
	
VCO_MID       = 1200000000U,
	
VCO_HIGH      = 2200000000U,
	
VCO_HIGH_HIGH = 3100000000U,
	
VCO_MAX       = 4000000000U,
};

struct iproc_pll;


struct iproc_clk {
	
struct clk_hw hw;
	
const char *name;
	
struct iproc_pll *pll;
	
unsigned long rate;
	
const struct iproc_clk_ctrl *ctrl;
};


struct iproc_pll {
	
void __iomem *status_base;
	
void __iomem *control_base;
	
void __iomem *pwr_base;
	
void __iomem *asiu_base;

	
const struct iproc_pll_ctrl *ctrl;
	
const struct iproc_pll_vco_param *vco_param;
	
unsigned int num_vco_entries;

	
struct clk_onecell_data clk_data;
	
struct iproc_clk *clks;
};


#define to_iproc_clk(hw) container_of(hw, struct iproc_clk, hw)

/*
 * Based on the target frequency, find a match from the VCO frequency parameter
 * table and return its index
 */

static int pll_get_rate_index(struct iproc_pll *pll, unsigned int target_rate) { int i; for (i = 0; i < pll->num_vco_entries; i++) if (target_rate == pll->vco_param[i].rate) break; if (i >= pll->num_vco_entries) return -EINVAL; return i; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui62100.00%1100.00%
Total62100.00%1100.00%


static int get_kp(unsigned long ref_freq, enum kp_band kp_index) { int i; if (ref_freq < ref_freq_table[0][0]) return -EINVAL; for (i = 0; i < NUM_FREQ_BANDS; i++) { if (ref_freq >= ref_freq_table[i][0] && ref_freq < ref_freq_table[i][1]) return kp_table[kp_index][i]; } return -EINVAL; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui83100.00%1100.00%
Total83100.00%1100.00%


static int pll_wait_for_lock(struct iproc_pll *pll) { int i; const struct iproc_pll_ctrl *ctrl = pll->ctrl; for (i = 0; i < LOCK_DELAY; i++) { u32 val = readl(pll->status_base + ctrl->status.offset); if (val & (1 << ctrl->status.shift)) return 0; udelay(10); } return -EIO; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui8098.77%150.00%
jon masonjon mason11.23%150.00%
Total81100.00%2100.00%


static void iproc_pll_write(const struct iproc_pll *pll, void __iomem *base, const u32 offset, u32 val) { const struct iproc_pll_ctrl *ctrl = pll->ctrl; writel(val, base + offset); if (unlikely(ctrl->flags & IPROC_CLK_NEEDS_READ_BACK && (base == pll->status_base || base == pll->control_base))) val = readl(base + offset); }

Contributors

PersonTokensPropCommitsCommitProp
jon masonjon mason77100.00%2100.00%
Total77100.00%2100.00%


static void __pll_disable(struct iproc_pll *pll) { const struct iproc_pll_ctrl *ctrl = pll->ctrl; u32 val; if (ctrl->flags & IPROC_CLK_PLL_ASIU) { val = readl(pll->asiu_base + ctrl->asiu.offset); val &= ~(1 << ctrl->asiu.en_shift); iproc_pll_write(pll, pll->asiu_base, ctrl->asiu.offset, val); } if (ctrl->flags & IPROC_CLK_EMBED_PWRCTRL) { val = readl(pll->control_base + ctrl->aon.offset); val |= bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift; iproc_pll_write(pll, pll->control_base, ctrl->aon.offset, val); } if (pll->pwr_base) { /* latch input value so core power can be shut down */ val = readl(pll->pwr_base + ctrl->aon.offset); val |= 1 << ctrl->aon.iso_shift; iproc_pll_write(pll, pll->pwr_base, ctrl->aon.offset, val); /* power down the core */ val &= ~(bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift); iproc_pll_write(pll, pll->pwr_base, ctrl->aon.offset, val); } }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui14865.20%125.00%
jon masonjon mason7934.80%375.00%
Total227100.00%4100.00%


static int __pll_enable(struct iproc_pll *pll) { const struct iproc_pll_ctrl *ctrl = pll->ctrl; u32 val; if (ctrl->flags & IPROC_CLK_EMBED_PWRCTRL) { val = readl(pll->control_base + ctrl->aon.offset); val &= ~(bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift); iproc_pll_write(pll, pll->control_base, ctrl->aon.offset, val); } if (pll->pwr_base) { /* power up the PLL and make sure it's not latched */ val = readl(pll->pwr_base + ctrl->aon.offset); val |= bit_mask(ctrl->aon.pwr_width) << ctrl->aon.pwr_shift; val &= ~(1 << ctrl->aon.iso_shift); iproc_pll_write(pll, pll->pwr_base, ctrl->aon.offset, val); } /* certain PLLs also need to be ungated from the ASIU top level */ if (ctrl->flags & IPROC_CLK_PLL_ASIU) { val = readl(pll->asiu_base + ctrl->asiu.offset); val |= (1 << ctrl->asiu.en_shift); iproc_pll_write(pll, pll->asiu_base, ctrl->asiu.offset, val); } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui13562.79%125.00%
jon masonjon mason8037.21%375.00%
Total215100.00%4100.00%


static void __pll_put_in_reset(struct iproc_pll *pll) { u32 val; const struct iproc_pll_ctrl *ctrl = pll->ctrl; const struct iproc_pll_reset_ctrl *reset = &ctrl->reset; val = readl(pll->control_base + reset->offset); if (ctrl->flags & IPROC_CLK_PLL_RESET_ACTIVE_LOW) val |= BIT(reset->reset_shift) | BIT(reset->p_reset_shift); else val &= ~(BIT(reset->reset_shift) | BIT(reset->p_reset_shift)); iproc_pll_write(pll, pll->control_base, reset->offset, val); }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui6964.49%125.00%
simran raisimran rai3128.97%125.00%
jon masonjon mason76.54%250.00%
Total107100.00%4100.00%


static void __pll_bring_out_reset(struct iproc_pll *pll, unsigned int kp, unsigned int ka, unsigned int ki) { u32 val; const struct iproc_pll_ctrl *ctrl = pll->ctrl; const struct iproc_pll_reset_ctrl *reset = &ctrl->reset; const struct iproc_pll_dig_filter_ctrl *dig_filter = &ctrl->dig_filter; val = readl(pll->control_base + dig_filter->offset); val &= ~(bit_mask(dig_filter->ki_width) << dig_filter->ki_shift | bit_mask(dig_filter->kp_width) << dig_filter->kp_shift | bit_mask(dig_filter->ka_width) << dig_filter->ka_shift); val |= ki << dig_filter->ki_shift | kp << dig_filter->kp_shift | ka << dig_filter->ka_shift; iproc_pll_write(pll, pll->control_base, dig_filter->offset, val); val = readl(pll->control_base + reset->offset); if (ctrl->flags & IPROC_CLK_PLL_RESET_ACTIVE_LOW) val &= ~(BIT(reset->reset_shift) | BIT(reset->p_reset_shift)); else val |= BIT(reset->reset_shift) | BIT(reset->p_reset_shift); iproc_pll_write(pll, pll->control_base, reset->offset, val); }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui12658.33%120.00%
jon masonjon mason5625.93%360.00%
simran raisimran rai3415.74%120.00%
Total216100.00%5100.00%


static int pll_set_rate(struct iproc_clk *clk, unsigned int rate_index, unsigned long parent_rate) { struct iproc_pll *pll = clk->pll; const struct iproc_pll_vco_param *vco = &pll->vco_param[rate_index]; const struct iproc_pll_ctrl *ctrl = pll->ctrl; int ka = 0, ki, kp, ret; unsigned long rate = vco->rate; u32 val; enum kp_band kp_index; unsigned long ref_freq; /* * reference frequency = parent frequency / PDIV * If PDIV = 0, then it becomes a multiplier (x2) */ if (vco->pdiv == 0) ref_freq = parent_rate * 2; else ref_freq = parent_rate / vco->pdiv; /* determine Ki and Kp index based on target VCO frequency */ if (rate >= VCO_LOW && rate < VCO_HIGH) { ki = 4; kp_index = KP_BAND_MID; } else if (rate >= VCO_HIGH && rate && rate < VCO_HIGH_HIGH) { ki = 3; kp_index = KP_BAND_HIGH; } else if (rate >= VCO_HIGH_HIGH && rate < VCO_MAX) { ki = 3; kp_index = KP_BAND_HIGH_HIGH; } else { pr_err("%s: pll: %s has invalid rate: %lu\n", __func__, clk->name, rate); return -EINVAL; } kp = get_kp(ref_freq, kp_index); if (kp < 0) { pr_err("%s: pll: %s has invalid kp\n", __func__, clk->name); return kp; } ret = __pll_enable(pll); if (ret) { pr_err("%s: pll: %s fails to enable\n", __func__, clk->name); return ret; } /* put PLL in reset */ __pll_put_in_reset(pll); /* set PLL in user mode before modifying PLL controls */ if (ctrl->flags & IPROC_CLK_PLL_USER_MODE_ON) { val = readl(pll->control_base + ctrl->macro_mode.offset); val &= ~(bit_mask(ctrl->macro_mode.width) << ctrl->macro_mode.shift); val |= PLL_USER_MODE << ctrl->macro_mode.shift; iproc_pll_write(pll, pll->control_base, ctrl->macro_mode.offset, val); } iproc_pll_write(pll, pll->control_base, ctrl->vco_ctrl.u_offset, 0); val = readl(pll->control_base + ctrl->vco_ctrl.l_offset); if (rate >= VCO_LOW && rate < VCO_MID) val |= (1 << PLL_VCO_LOW_SHIFT); if (rate < VCO_HIGH) val &= ~(1 << PLL_VCO_HIGH_SHIFT); else val |= (1 << PLL_VCO_HIGH_SHIFT); iproc_pll_write(pll, pll->control_base, ctrl->vco_ctrl.l_offset, val); /* program integer part of NDIV */ val = readl(pll->control_base + ctrl->ndiv_int.offset); val &= ~(bit_mask(ctrl->ndiv_int.width) << ctrl->ndiv_int.shift); val |= vco->ndiv_int << ctrl->ndiv_int.shift; iproc_pll_write(pll, pll->control_base, ctrl->ndiv_int.offset, val); /* program fractional part of NDIV */ if (ctrl->flags & IPROC_CLK_PLL_HAS_NDIV_FRAC) { val = readl(pll->control_base + ctrl->ndiv_frac.offset); val &= ~(bit_mask(ctrl->ndiv_frac.width) << ctrl->ndiv_frac.shift); val |= vco->ndiv_frac << ctrl->ndiv_frac.shift; iproc_pll_write(pll, pll->control_base, ctrl->ndiv_frac.offset, val); } /* program PDIV */ val = readl(pll->control_base + ctrl->pdiv.offset); val &= ~(bit_mask(ctrl->pdiv.width) << ctrl->pdiv.shift); val |= vco->pdiv << ctrl->pdiv.shift; iproc_pll_write(pll, pll->control_base, ctrl->pdiv.offset, val); __pll_bring_out_reset(pll, kp, ka, ki); ret = pll_wait_for_lock(pll); if (ret < 0) { pr_err("%s: pll: %s failed to lock\n", __func__, clk->name); return ret; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui56083.96%125.00%
simran raisimran rai7310.94%125.00%
jon masonjon mason345.10%250.00%
Total667100.00%4100.00%


static int iproc_pll_enable(struct clk_hw *hw) { struct iproc_clk *clk = to_iproc_clk(hw); struct iproc_pll *pll = clk->pll; return __pll_enable(pll); }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui36100.00%1100.00%
Total36100.00%1100.00%


static void iproc_pll_disable(struct clk_hw *hw) { struct iproc_clk *clk = to_iproc_clk(hw); struct iproc_pll *pll = clk->pll; const struct iproc_pll_ctrl *ctrl = pll->ctrl; if (ctrl->flags & IPROC_CLK_AON) return; __pll_disable(pll); }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui54100.00%1100.00%
Total54100.00%1100.00%


static unsigned long iproc_pll_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct iproc_clk *clk = to_iproc_clk(hw); struct iproc_pll *pll = clk->pll; const struct iproc_pll_ctrl *ctrl = pll->ctrl; u32 val; u64 ndiv, ndiv_int, ndiv_frac; unsigned int pdiv; if (parent_rate == 0) return 0; /* PLL needs to be locked */ val = readl(pll->status_base + ctrl->status.offset); if ((val & (1 << ctrl->status.shift)) == 0) { clk->rate = 0; return 0; } /* * PLL output frequency = * * ((ndiv_int + ndiv_frac / 2^20) * (parent clock rate / pdiv) */ val = readl(pll->control_base + ctrl->ndiv_int.offset); ndiv_int = (val >> ctrl->ndiv_int.shift) & bit_mask(ctrl->ndiv_int.width); ndiv = ndiv_int << 20; if (ctrl->flags & IPROC_CLK_PLL_HAS_NDIV_FRAC) { val = readl(pll->control_base + ctrl->ndiv_frac.offset); ndiv_frac = (val >> ctrl->ndiv_frac.shift) & bit_mask(ctrl->ndiv_frac.width); ndiv += ndiv_frac; } val = readl(pll->control_base + ctrl->pdiv.offset); pdiv = (val >> ctrl->pdiv.shift) & bit_mask(ctrl->pdiv.width); clk->rate = (ndiv * parent_rate) >> 20; if (pdiv == 0) clk->rate *= 2; else clk->rate /= pdiv; return clk->rate; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui26796.04%133.33%
simran raisimran rai72.52%133.33%
jon masonjon mason41.44%133.33%
Total278100.00%3100.00%


static long iproc_pll_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *parent_rate) { unsigned i; struct iproc_clk *clk = to_iproc_clk(hw); struct iproc_pll *pll = clk->pll; if (rate == 0 || *parent_rate == 0 || !pll->vco_param) return -EINVAL; for (i = 0; i < pll->num_vco_entries; i++) { if (rate <= pll->vco_param[i].rate) break; } if (i == pll->num_vco_entries) i--; return pll->vco_param[i].rate; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui114100.00%1100.00%
Total114100.00%1100.00%


static int iproc_pll_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct iproc_clk *clk = to_iproc_clk(hw); struct iproc_pll *pll = clk->pll; int rate_index, ret; rate_index = pll_get_rate_index(pll, rate); if (rate_index < 0) return rate_index; ret = pll_set_rate(clk, rate_index, parent_rate); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui75100.00%1100.00%
Total75100.00%1100.00%

static const struct clk_ops iproc_pll_ops = { .enable = iproc_pll_enable, .disable = iproc_pll_disable, .recalc_rate = iproc_pll_recalc_rate, .round_rate = iproc_pll_round_rate, .set_rate = iproc_pll_set_rate, };
static int iproc_clk_enable(struct clk_hw *hw) { struct iproc_clk *clk = to_iproc_clk(hw); const struct iproc_clk_ctrl *ctrl = clk->ctrl; struct iproc_pll *pll = clk->pll; u32 val; /* channel enable is active low */ val = readl(pll->control_base + ctrl->enable.offset); val &= ~(1 << ctrl->enable.enable_shift); iproc_pll_write(pll, pll->control_base, ctrl->enable.offset, val); /* also make sure channel is not held */ val = readl(pll->control_base + ctrl->enable.offset); val &= ~(1 << ctrl->enable.hold_shift); iproc_pll_write(pll, pll->control_base, ctrl->enable.offset, val); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui12489.86%133.33%
jon masonjon mason1410.14%266.67%
Total138100.00%3100.00%


static void iproc_clk_disable(struct clk_hw *hw) { struct iproc_clk *clk = to_iproc_clk(hw); const struct iproc_clk_ctrl *ctrl = clk->ctrl; struct iproc_pll *pll = clk->pll; u32 val; if (ctrl->flags & IPROC_CLK_AON) return; val = readl(pll->control_base + ctrl->enable.offset); val |= 1 << ctrl->enable.enable_shift; iproc_pll_write(pll, pll->control_base, ctrl->enable.offset, val); }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui8792.55%133.33%
jon masonjon mason77.45%266.67%
Total94100.00%3100.00%


static unsigned long iproc_clk_recalc_rate(struct clk_hw *hw, unsigned long parent_rate) { struct iproc_clk *clk = to_iproc_clk(hw); const struct iproc_clk_ctrl *ctrl = clk->ctrl; struct iproc_pll *pll = clk->pll; u32 val; unsigned int mdiv; if (parent_rate == 0) return 0; val = readl(pll->control_base + ctrl->mdiv.offset); mdiv = (val >> ctrl->mdiv.shift) & bit_mask(ctrl->mdiv.width); if (mdiv == 0) mdiv = 256; if (ctrl->flags & IPROC_CLK_MCLK_DIV_BY_2) clk->rate = parent_rate / (mdiv * 2); else clk->rate = parent_rate / mdiv; return clk->rate; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui11984.40%133.33%
simran raisimran rai2114.89%133.33%
jon masonjon mason10.71%133.33%
Total141100.00%3100.00%


static long iproc_clk_round_rate(struct clk_hw *hw, unsigned long rate, unsigned long *parent_rate) { unsigned int div; if (rate == 0 || *parent_rate == 0) return -EINVAL; if (rate == *parent_rate) return *parent_rate; div = DIV_ROUND_UP(*parent_rate, rate); if (div < 2) return *parent_rate; if (div > 256) div = 256; return *parent_rate / div; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui86100.00%1100.00%
Total86100.00%1100.00%


static int iproc_clk_set_rate(struct clk_hw *hw, unsigned long rate, unsigned long parent_rate) { struct iproc_clk *clk = to_iproc_clk(hw); const struct iproc_clk_ctrl *ctrl = clk->ctrl; struct iproc_pll *pll = clk->pll; u32 val; unsigned int div; if (rate == 0 || parent_rate == 0) return -EINVAL; if (ctrl->flags & IPROC_CLK_MCLK_DIV_BY_2) div = DIV_ROUND_UP(parent_rate, rate * 2); else div = DIV_ROUND_UP(parent_rate, rate); if (div > 256) return -EINVAL; val = readl(pll->control_base + ctrl->mdiv.offset); if (div == 256) { val &= ~(bit_mask(ctrl->mdiv.width) << ctrl->mdiv.shift); } else { val &= ~(bit_mask(ctrl->mdiv.width) << ctrl->mdiv.shift); val |= div << ctrl->mdiv.shift; } iproc_pll_write(pll, pll->control_base, ctrl->mdiv.offset, val); if (ctrl->flags & IPROC_CLK_MCLK_DIV_BY_2) clk->rate = parent_rate / (div * 2); else clk->rate = parent_rate / div; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui18579.40%125.00%
simran raisimran rai4117.60%125.00%
jon masonjon mason73.00%250.00%
Total233100.00%4100.00%

static const struct clk_ops iproc_clk_ops = { .enable = iproc_clk_enable, .disable = iproc_clk_disable, .recalc_rate = iproc_clk_recalc_rate, .round_rate = iproc_clk_round_rate, .set_rate = iproc_clk_set_rate, }; /** * Some PLLs require the PLL SW override bit to be set before changes can be * applied to the PLL */
static void iproc_pll_sw_cfg(struct iproc_pll *pll) { const struct iproc_pll_ctrl *ctrl = pll->ctrl; if (ctrl->flags & IPROC_CLK_PLL_NEEDS_SW_CFG) { u32 val; val = readl(pll->control_base + ctrl->sw_ctrl.offset); val |= BIT(ctrl->sw_ctrl.shift); iproc_pll_write(pll, pll->control_base, ctrl->sw_ctrl.offset, val); } }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui7090.91%133.33%
jon masonjon mason79.09%266.67%
Total77100.00%3100.00%


void __init iproc_pll_clk_setup(struct device_node *node, const struct iproc_pll_ctrl *pll_ctrl, const struct iproc_pll_vco_param *vco, unsigned int num_vco_entries, const struct iproc_clk_ctrl *clk_ctrl, unsigned int num_clks) { int i, ret; struct clk *clk; struct iproc_pll *pll; struct iproc_clk *iclk; struct clk_init_data init; const char *parent_name; if (WARN_ON(!pll_ctrl) || WARN_ON(!clk_ctrl)) return; pll = kzalloc(sizeof(*pll), GFP_KERNEL); if (WARN_ON(!pll)) return; pll->clk_data.clk_num = num_clks; pll->clk_data.clks = kcalloc(num_clks, sizeof(*pll->clk_data.clks), GFP_KERNEL); if (WARN_ON(!pll->clk_data.clks)) goto err_clk_data; pll->clks = kcalloc(num_clks, sizeof(*pll->clks), GFP_KERNEL); if (WARN_ON(!pll->clks)) goto err_clks; pll->control_base = of_iomap(node, 0); if (WARN_ON(!pll->control_base)) goto err_pll_iomap; /* Some SoCs do not require the pwr_base, thus failing is not fatal */ pll->pwr_base = of_iomap(node, 1); /* some PLLs require gating control at the top ASIU level */ if (pll_ctrl->flags & IPROC_CLK_PLL_ASIU) { pll->asiu_base = of_iomap(node, 2); if (WARN_ON(!pll->asiu_base)) goto err_asiu_iomap; } if (pll_ctrl->flags & IPROC_CLK_PLL_SPLIT_STAT_CTRL) { /* Some SoCs have a split status/control. If this does not * exist, assume they are unified. */ pll->status_base = of_iomap(node, 2); if (!pll->status_base) goto err_status_iomap; } else pll->status_base = pll->control_base; /* initialize and register the PLL itself */ pll->ctrl = pll_ctrl; iclk = &pll->clks[0]; iclk->pll = pll; iclk->name = node->name; init.name = node->name; init.ops = &iproc_pll_ops; init.flags = 0; parent_name = of_clk_get_parent_name(node, 0); init.parent_names = (parent_name ? &parent_name : NULL); init.num_parents = (parent_name ? 1 : 0); iclk->hw.init = &init; if (vco) { pll->num_vco_entries = num_vco_entries; pll->vco_param = vco; } iproc_pll_sw_cfg(pll); clk = clk_register(NULL, &iclk->hw); if (WARN_ON(IS_ERR(clk))) goto err_pll_register; pll->clk_data.clks[0] = clk; /* now initialize and register all leaf clocks */ for (i = 1; i < num_clks; i++) { const char *clk_name; memset(&init, 0, sizeof(init)); parent_name = node->name; ret = of_property_read_string_index(node, "clock-output-names", i, &clk_name); if (WARN_ON(ret)) goto err_clk_register; iclk = &pll->clks[i]; iclk->name = clk_name; iclk->pll = pll; iclk->ctrl = &clk_ctrl[i]; init.name = clk_name; init.ops = &iproc_clk_ops; init.flags = 0; init.parent_names = (parent_name ? &parent_name : NULL); init.num_parents = (parent_name ? 1 : 0); iclk->hw.init = &init; clk = clk_register(NULL, &iclk->hw); if (WARN_ON(IS_ERR(clk))) goto err_clk_register; pll->clk_data.clks[i] = clk; } ret = of_clk_add_provider(node, of_clk_src_onecell_get, &pll->clk_data); if (WARN_ON(ret)) goto err_clk_register; return; err_clk_register: for (i = 0; i < num_clks; i++) clk_unregister(pll->clk_data.clks[i]); err_pll_register: if (pll->status_base != pll->control_base) iounmap(pll->status_base); err_status_iomap: if (pll->asiu_base) iounmap(pll->asiu_base); err_asiu_iomap: if (pll->pwr_base) iounmap(pll->pwr_base); iounmap(pll->control_base); err_pll_iomap: kfree(pll->clks); err_clks: kfree(pll->clk_data.clks); err_clk_data: kfree(pll); }

Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui72494.39%133.33%
jon masonjon mason435.61%266.67%
Total767100.00%3100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
ray juiray jui357184.90%114.29%
jon masonjon mason42310.06%457.14%
simran raisimran rai2125.04%228.57%
Total4206100.00%7100.00%
Directory: drivers/clk/bcm
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
{% endraw %}