162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * Copyright (c) 2014 MediaTek Inc.
462306a36Sopenharmony_ci * Author: James Liao <jamesjj.liao@mediatek.com>
562306a36Sopenharmony_ci */
662306a36Sopenharmony_ci
762306a36Sopenharmony_ci#include <linux/clk-provider.h>
862306a36Sopenharmony_ci#include <linux/container_of.h>
962306a36Sopenharmony_ci#include <linux/delay.h>
1062306a36Sopenharmony_ci#include <linux/err.h>
1162306a36Sopenharmony_ci#include <linux/io.h>
1262306a36Sopenharmony_ci#include <linux/module.h>
1362306a36Sopenharmony_ci#include <linux/of_address.h>
1462306a36Sopenharmony_ci#include <linux/slab.h>
1562306a36Sopenharmony_ci
1662306a36Sopenharmony_ci#include "clk-pll.h"
1762306a36Sopenharmony_ci
1862306a36Sopenharmony_ci#define MHZ			(1000 * 1000)
1962306a36Sopenharmony_ci
2062306a36Sopenharmony_ci#define REG_CON0		0
2162306a36Sopenharmony_ci#define REG_CON1		4
2262306a36Sopenharmony_ci
2362306a36Sopenharmony_ci#define CON0_BASE_EN		BIT(0)
2462306a36Sopenharmony_ci#define CON0_PWR_ON		BIT(0)
2562306a36Sopenharmony_ci#define CON0_ISO_EN		BIT(1)
2662306a36Sopenharmony_ci#define PCW_CHG_MASK		BIT(31)
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci#define AUDPLL_TUNER_EN		BIT(31)
2962306a36Sopenharmony_ci
3062306a36Sopenharmony_ci/* default 7 bits integer, can be overridden with pcwibits. */
3162306a36Sopenharmony_ci#define INTEGER_BITS		7
3262306a36Sopenharmony_ci
3362306a36Sopenharmony_ciint mtk_pll_is_prepared(struct clk_hw *hw)
3462306a36Sopenharmony_ci{
3562306a36Sopenharmony_ci	struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
3662306a36Sopenharmony_ci
3762306a36Sopenharmony_ci	return (readl(pll->en_addr) & BIT(pll->data->pll_en_bit)) != 0;
3862306a36Sopenharmony_ci}
3962306a36Sopenharmony_ci
4062306a36Sopenharmony_cistatic unsigned long __mtk_pll_recalc_rate(struct mtk_clk_pll *pll, u32 fin,
4162306a36Sopenharmony_ci		u32 pcw, int postdiv)
4262306a36Sopenharmony_ci{
4362306a36Sopenharmony_ci	int pcwbits = pll->data->pcwbits;
4462306a36Sopenharmony_ci	int pcwfbits = 0;
4562306a36Sopenharmony_ci	int ibits;
4662306a36Sopenharmony_ci	u64 vco;
4762306a36Sopenharmony_ci	u8 c = 0;
4862306a36Sopenharmony_ci
4962306a36Sopenharmony_ci	/* The fractional part of the PLL divider. */
5062306a36Sopenharmony_ci	ibits = pll->data->pcwibits ? pll->data->pcwibits : INTEGER_BITS;
5162306a36Sopenharmony_ci	if (pcwbits > ibits)
5262306a36Sopenharmony_ci		pcwfbits = pcwbits - ibits;
5362306a36Sopenharmony_ci
5462306a36Sopenharmony_ci	vco = (u64)fin * pcw;
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci	if (pcwfbits && (vco & GENMASK(pcwfbits - 1, 0)))
5762306a36Sopenharmony_ci		c = 1;
5862306a36Sopenharmony_ci
5962306a36Sopenharmony_ci	vco >>= pcwfbits;
6062306a36Sopenharmony_ci
6162306a36Sopenharmony_ci	if (c)
6262306a36Sopenharmony_ci		vco++;
6362306a36Sopenharmony_ci
6462306a36Sopenharmony_ci	return ((unsigned long)vco + postdiv - 1) / postdiv;
6562306a36Sopenharmony_ci}
6662306a36Sopenharmony_ci
6762306a36Sopenharmony_cistatic void __mtk_pll_tuner_enable(struct mtk_clk_pll *pll)
6862306a36Sopenharmony_ci{
6962306a36Sopenharmony_ci	u32 r;
7062306a36Sopenharmony_ci
7162306a36Sopenharmony_ci	if (pll->tuner_en_addr) {
7262306a36Sopenharmony_ci		r = readl(pll->tuner_en_addr) | BIT(pll->data->tuner_en_bit);
7362306a36Sopenharmony_ci		writel(r, pll->tuner_en_addr);
7462306a36Sopenharmony_ci	} else if (pll->tuner_addr) {
7562306a36Sopenharmony_ci		r = readl(pll->tuner_addr) | AUDPLL_TUNER_EN;
7662306a36Sopenharmony_ci		writel(r, pll->tuner_addr);
7762306a36Sopenharmony_ci	}
7862306a36Sopenharmony_ci}
7962306a36Sopenharmony_ci
8062306a36Sopenharmony_cistatic void __mtk_pll_tuner_disable(struct mtk_clk_pll *pll)
8162306a36Sopenharmony_ci{
8262306a36Sopenharmony_ci	u32 r;
8362306a36Sopenharmony_ci
8462306a36Sopenharmony_ci	if (pll->tuner_en_addr) {
8562306a36Sopenharmony_ci		r = readl(pll->tuner_en_addr) & ~BIT(pll->data->tuner_en_bit);
8662306a36Sopenharmony_ci		writel(r, pll->tuner_en_addr);
8762306a36Sopenharmony_ci	} else if (pll->tuner_addr) {
8862306a36Sopenharmony_ci		r = readl(pll->tuner_addr) & ~AUDPLL_TUNER_EN;
8962306a36Sopenharmony_ci		writel(r, pll->tuner_addr);
9062306a36Sopenharmony_ci	}
9162306a36Sopenharmony_ci}
9262306a36Sopenharmony_ci
9362306a36Sopenharmony_cistatic void mtk_pll_set_rate_regs(struct mtk_clk_pll *pll, u32 pcw,
9462306a36Sopenharmony_ci		int postdiv)
9562306a36Sopenharmony_ci{
9662306a36Sopenharmony_ci	u32 chg, val;
9762306a36Sopenharmony_ci
9862306a36Sopenharmony_ci	/* disable tuner */
9962306a36Sopenharmony_ci	__mtk_pll_tuner_disable(pll);
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ci	/* set postdiv */
10262306a36Sopenharmony_ci	val = readl(pll->pd_addr);
10362306a36Sopenharmony_ci	val &= ~(POSTDIV_MASK << pll->data->pd_shift);
10462306a36Sopenharmony_ci	val |= (ffs(postdiv) - 1) << pll->data->pd_shift;
10562306a36Sopenharmony_ci
10662306a36Sopenharmony_ci	/* postdiv and pcw need to set at the same time if on same register */
10762306a36Sopenharmony_ci	if (pll->pd_addr != pll->pcw_addr) {
10862306a36Sopenharmony_ci		writel(val, pll->pd_addr);
10962306a36Sopenharmony_ci		val = readl(pll->pcw_addr);
11062306a36Sopenharmony_ci	}
11162306a36Sopenharmony_ci
11262306a36Sopenharmony_ci	/* set pcw */
11362306a36Sopenharmony_ci	val &= ~GENMASK(pll->data->pcw_shift + pll->data->pcwbits - 1,
11462306a36Sopenharmony_ci			pll->data->pcw_shift);
11562306a36Sopenharmony_ci	val |= pcw << pll->data->pcw_shift;
11662306a36Sopenharmony_ci	writel(val, pll->pcw_addr);
11762306a36Sopenharmony_ci	chg = readl(pll->pcw_chg_addr) | PCW_CHG_MASK;
11862306a36Sopenharmony_ci	writel(chg, pll->pcw_chg_addr);
11962306a36Sopenharmony_ci	if (pll->tuner_addr)
12062306a36Sopenharmony_ci		writel(val + 1, pll->tuner_addr);
12162306a36Sopenharmony_ci
12262306a36Sopenharmony_ci	/* restore tuner_en */
12362306a36Sopenharmony_ci	__mtk_pll_tuner_enable(pll);
12462306a36Sopenharmony_ci
12562306a36Sopenharmony_ci	udelay(20);
12662306a36Sopenharmony_ci}
12762306a36Sopenharmony_ci
12862306a36Sopenharmony_ci/*
12962306a36Sopenharmony_ci * mtk_pll_calc_values - calculate good values for a given input frequency.
13062306a36Sopenharmony_ci * @pll:	The pll
13162306a36Sopenharmony_ci * @pcw:	The pcw value (output)
13262306a36Sopenharmony_ci * @postdiv:	The post divider (output)
13362306a36Sopenharmony_ci * @freq:	The desired target frequency
13462306a36Sopenharmony_ci * @fin:	The input frequency
13562306a36Sopenharmony_ci *
13662306a36Sopenharmony_ci */
13762306a36Sopenharmony_civoid mtk_pll_calc_values(struct mtk_clk_pll *pll, u32 *pcw, u32 *postdiv,
13862306a36Sopenharmony_ci			 u32 freq, u32 fin)
13962306a36Sopenharmony_ci{
14062306a36Sopenharmony_ci	unsigned long fmin = pll->data->fmin ? pll->data->fmin : (1000 * MHZ);
14162306a36Sopenharmony_ci	const struct mtk_pll_div_table *div_table = pll->data->div_table;
14262306a36Sopenharmony_ci	u64 _pcw;
14362306a36Sopenharmony_ci	int ibits;
14462306a36Sopenharmony_ci	u32 val;
14562306a36Sopenharmony_ci
14662306a36Sopenharmony_ci	if (freq > pll->data->fmax)
14762306a36Sopenharmony_ci		freq = pll->data->fmax;
14862306a36Sopenharmony_ci
14962306a36Sopenharmony_ci	if (div_table) {
15062306a36Sopenharmony_ci		if (freq > div_table[0].freq)
15162306a36Sopenharmony_ci			freq = div_table[0].freq;
15262306a36Sopenharmony_ci
15362306a36Sopenharmony_ci		for (val = 0; div_table[val + 1].freq != 0; val++) {
15462306a36Sopenharmony_ci			if (freq > div_table[val + 1].freq)
15562306a36Sopenharmony_ci				break;
15662306a36Sopenharmony_ci		}
15762306a36Sopenharmony_ci		*postdiv = 1 << val;
15862306a36Sopenharmony_ci	} else {
15962306a36Sopenharmony_ci		for (val = 0; val < 5; val++) {
16062306a36Sopenharmony_ci			*postdiv = 1 << val;
16162306a36Sopenharmony_ci			if ((u64)freq * *postdiv >= fmin)
16262306a36Sopenharmony_ci				break;
16362306a36Sopenharmony_ci		}
16462306a36Sopenharmony_ci	}
16562306a36Sopenharmony_ci
16662306a36Sopenharmony_ci	/* _pcw = freq * postdiv / fin * 2^pcwfbits */
16762306a36Sopenharmony_ci	ibits = pll->data->pcwibits ? pll->data->pcwibits : INTEGER_BITS;
16862306a36Sopenharmony_ci	_pcw = ((u64)freq << val) << (pll->data->pcwbits - ibits);
16962306a36Sopenharmony_ci	do_div(_pcw, fin);
17062306a36Sopenharmony_ci
17162306a36Sopenharmony_ci	*pcw = (u32)_pcw;
17262306a36Sopenharmony_ci}
17362306a36Sopenharmony_ci
17462306a36Sopenharmony_ciint mtk_pll_set_rate(struct clk_hw *hw, unsigned long rate,
17562306a36Sopenharmony_ci		     unsigned long parent_rate)
17662306a36Sopenharmony_ci{
17762306a36Sopenharmony_ci	struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
17862306a36Sopenharmony_ci	u32 pcw = 0;
17962306a36Sopenharmony_ci	u32 postdiv;
18062306a36Sopenharmony_ci
18162306a36Sopenharmony_ci	mtk_pll_calc_values(pll, &pcw, &postdiv, rate, parent_rate);
18262306a36Sopenharmony_ci	mtk_pll_set_rate_regs(pll, pcw, postdiv);
18362306a36Sopenharmony_ci
18462306a36Sopenharmony_ci	return 0;
18562306a36Sopenharmony_ci}
18662306a36Sopenharmony_ci
18762306a36Sopenharmony_ciunsigned long mtk_pll_recalc_rate(struct clk_hw *hw, unsigned long parent_rate)
18862306a36Sopenharmony_ci{
18962306a36Sopenharmony_ci	struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
19062306a36Sopenharmony_ci	u32 postdiv;
19162306a36Sopenharmony_ci	u32 pcw;
19262306a36Sopenharmony_ci
19362306a36Sopenharmony_ci	postdiv = (readl(pll->pd_addr) >> pll->data->pd_shift) & POSTDIV_MASK;
19462306a36Sopenharmony_ci	postdiv = 1 << postdiv;
19562306a36Sopenharmony_ci
19662306a36Sopenharmony_ci	pcw = readl(pll->pcw_addr) >> pll->data->pcw_shift;
19762306a36Sopenharmony_ci	pcw &= GENMASK(pll->data->pcwbits - 1, 0);
19862306a36Sopenharmony_ci
19962306a36Sopenharmony_ci	return __mtk_pll_recalc_rate(pll, parent_rate, pcw, postdiv);
20062306a36Sopenharmony_ci}
20162306a36Sopenharmony_ci
20262306a36Sopenharmony_cilong mtk_pll_round_rate(struct clk_hw *hw, unsigned long rate,
20362306a36Sopenharmony_ci			unsigned long *prate)
20462306a36Sopenharmony_ci{
20562306a36Sopenharmony_ci	struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
20662306a36Sopenharmony_ci	u32 pcw = 0;
20762306a36Sopenharmony_ci	int postdiv;
20862306a36Sopenharmony_ci
20962306a36Sopenharmony_ci	mtk_pll_calc_values(pll, &pcw, &postdiv, rate, *prate);
21062306a36Sopenharmony_ci
21162306a36Sopenharmony_ci	return __mtk_pll_recalc_rate(pll, *prate, pcw, postdiv);
21262306a36Sopenharmony_ci}
21362306a36Sopenharmony_ci
21462306a36Sopenharmony_ciint mtk_pll_prepare(struct clk_hw *hw)
21562306a36Sopenharmony_ci{
21662306a36Sopenharmony_ci	struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
21762306a36Sopenharmony_ci	u32 r;
21862306a36Sopenharmony_ci
21962306a36Sopenharmony_ci	r = readl(pll->pwr_addr) | CON0_PWR_ON;
22062306a36Sopenharmony_ci	writel(r, pll->pwr_addr);
22162306a36Sopenharmony_ci	udelay(1);
22262306a36Sopenharmony_ci
22362306a36Sopenharmony_ci	r = readl(pll->pwr_addr) & ~CON0_ISO_EN;
22462306a36Sopenharmony_ci	writel(r, pll->pwr_addr);
22562306a36Sopenharmony_ci	udelay(1);
22662306a36Sopenharmony_ci
22762306a36Sopenharmony_ci	r = readl(pll->en_addr) | BIT(pll->data->pll_en_bit);
22862306a36Sopenharmony_ci	writel(r, pll->en_addr);
22962306a36Sopenharmony_ci
23062306a36Sopenharmony_ci	if (pll->data->en_mask) {
23162306a36Sopenharmony_ci		r = readl(pll->base_addr + REG_CON0) | pll->data->en_mask;
23262306a36Sopenharmony_ci		writel(r, pll->base_addr + REG_CON0);
23362306a36Sopenharmony_ci	}
23462306a36Sopenharmony_ci
23562306a36Sopenharmony_ci	__mtk_pll_tuner_enable(pll);
23662306a36Sopenharmony_ci
23762306a36Sopenharmony_ci	udelay(20);
23862306a36Sopenharmony_ci
23962306a36Sopenharmony_ci	if (pll->data->flags & HAVE_RST_BAR) {
24062306a36Sopenharmony_ci		r = readl(pll->base_addr + REG_CON0);
24162306a36Sopenharmony_ci		r |= pll->data->rst_bar_mask;
24262306a36Sopenharmony_ci		writel(r, pll->base_addr + REG_CON0);
24362306a36Sopenharmony_ci	}
24462306a36Sopenharmony_ci
24562306a36Sopenharmony_ci	return 0;
24662306a36Sopenharmony_ci}
24762306a36Sopenharmony_ci
24862306a36Sopenharmony_civoid mtk_pll_unprepare(struct clk_hw *hw)
24962306a36Sopenharmony_ci{
25062306a36Sopenharmony_ci	struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
25162306a36Sopenharmony_ci	u32 r;
25262306a36Sopenharmony_ci
25362306a36Sopenharmony_ci	if (pll->data->flags & HAVE_RST_BAR) {
25462306a36Sopenharmony_ci		r = readl(pll->base_addr + REG_CON0);
25562306a36Sopenharmony_ci		r &= ~pll->data->rst_bar_mask;
25662306a36Sopenharmony_ci		writel(r, pll->base_addr + REG_CON0);
25762306a36Sopenharmony_ci	}
25862306a36Sopenharmony_ci
25962306a36Sopenharmony_ci	__mtk_pll_tuner_disable(pll);
26062306a36Sopenharmony_ci
26162306a36Sopenharmony_ci	if (pll->data->en_mask) {
26262306a36Sopenharmony_ci		r = readl(pll->base_addr + REG_CON0) & ~pll->data->en_mask;
26362306a36Sopenharmony_ci		writel(r, pll->base_addr + REG_CON0);
26462306a36Sopenharmony_ci	}
26562306a36Sopenharmony_ci
26662306a36Sopenharmony_ci	r = readl(pll->en_addr) & ~BIT(pll->data->pll_en_bit);
26762306a36Sopenharmony_ci	writel(r, pll->en_addr);
26862306a36Sopenharmony_ci
26962306a36Sopenharmony_ci	r = readl(pll->pwr_addr) | CON0_ISO_EN;
27062306a36Sopenharmony_ci	writel(r, pll->pwr_addr);
27162306a36Sopenharmony_ci
27262306a36Sopenharmony_ci	r = readl(pll->pwr_addr) & ~CON0_PWR_ON;
27362306a36Sopenharmony_ci	writel(r, pll->pwr_addr);
27462306a36Sopenharmony_ci}
27562306a36Sopenharmony_ci
27662306a36Sopenharmony_ciconst struct clk_ops mtk_pll_ops = {
27762306a36Sopenharmony_ci	.is_prepared	= mtk_pll_is_prepared,
27862306a36Sopenharmony_ci	.prepare	= mtk_pll_prepare,
27962306a36Sopenharmony_ci	.unprepare	= mtk_pll_unprepare,
28062306a36Sopenharmony_ci	.recalc_rate	= mtk_pll_recalc_rate,
28162306a36Sopenharmony_ci	.round_rate	= mtk_pll_round_rate,
28262306a36Sopenharmony_ci	.set_rate	= mtk_pll_set_rate,
28362306a36Sopenharmony_ci};
28462306a36Sopenharmony_ci
28562306a36Sopenharmony_cistruct clk_hw *mtk_clk_register_pll_ops(struct mtk_clk_pll *pll,
28662306a36Sopenharmony_ci					const struct mtk_pll_data *data,
28762306a36Sopenharmony_ci					void __iomem *base,
28862306a36Sopenharmony_ci					const struct clk_ops *pll_ops)
28962306a36Sopenharmony_ci{
29062306a36Sopenharmony_ci	struct clk_init_data init = {};
29162306a36Sopenharmony_ci	int ret;
29262306a36Sopenharmony_ci	const char *parent_name = "clk26m";
29362306a36Sopenharmony_ci
29462306a36Sopenharmony_ci	pll->base_addr = base + data->reg;
29562306a36Sopenharmony_ci	pll->pwr_addr = base + data->pwr_reg;
29662306a36Sopenharmony_ci	pll->pd_addr = base + data->pd_reg;
29762306a36Sopenharmony_ci	pll->pcw_addr = base + data->pcw_reg;
29862306a36Sopenharmony_ci	if (data->pcw_chg_reg)
29962306a36Sopenharmony_ci		pll->pcw_chg_addr = base + data->pcw_chg_reg;
30062306a36Sopenharmony_ci	else
30162306a36Sopenharmony_ci		pll->pcw_chg_addr = pll->base_addr + REG_CON1;
30262306a36Sopenharmony_ci	if (data->tuner_reg)
30362306a36Sopenharmony_ci		pll->tuner_addr = base + data->tuner_reg;
30462306a36Sopenharmony_ci	if (data->tuner_en_reg || data->tuner_en_bit)
30562306a36Sopenharmony_ci		pll->tuner_en_addr = base + data->tuner_en_reg;
30662306a36Sopenharmony_ci	if (data->en_reg)
30762306a36Sopenharmony_ci		pll->en_addr = base + data->en_reg;
30862306a36Sopenharmony_ci	else
30962306a36Sopenharmony_ci		pll->en_addr = pll->base_addr + REG_CON0;
31062306a36Sopenharmony_ci	pll->hw.init = &init;
31162306a36Sopenharmony_ci	pll->data = data;
31262306a36Sopenharmony_ci
31362306a36Sopenharmony_ci	init.name = data->name;
31462306a36Sopenharmony_ci	init.flags = (data->flags & PLL_AO) ? CLK_IS_CRITICAL : 0;
31562306a36Sopenharmony_ci	init.ops = pll_ops;
31662306a36Sopenharmony_ci	if (data->parent_name)
31762306a36Sopenharmony_ci		init.parent_names = &data->parent_name;
31862306a36Sopenharmony_ci	else
31962306a36Sopenharmony_ci		init.parent_names = &parent_name;
32062306a36Sopenharmony_ci	init.num_parents = 1;
32162306a36Sopenharmony_ci
32262306a36Sopenharmony_ci	ret = clk_hw_register(NULL, &pll->hw);
32362306a36Sopenharmony_ci
32462306a36Sopenharmony_ci	if (ret)
32562306a36Sopenharmony_ci		return ERR_PTR(ret);
32662306a36Sopenharmony_ci
32762306a36Sopenharmony_ci	return &pll->hw;
32862306a36Sopenharmony_ci}
32962306a36Sopenharmony_ci
33062306a36Sopenharmony_cistruct clk_hw *mtk_clk_register_pll(const struct mtk_pll_data *data,
33162306a36Sopenharmony_ci				    void __iomem *base)
33262306a36Sopenharmony_ci{
33362306a36Sopenharmony_ci	struct mtk_clk_pll *pll;
33462306a36Sopenharmony_ci	struct clk_hw *hw;
33562306a36Sopenharmony_ci
33662306a36Sopenharmony_ci	pll = kzalloc(sizeof(*pll), GFP_KERNEL);
33762306a36Sopenharmony_ci	if (!pll)
33862306a36Sopenharmony_ci		return ERR_PTR(-ENOMEM);
33962306a36Sopenharmony_ci
34062306a36Sopenharmony_ci	hw = mtk_clk_register_pll_ops(pll, data, base, &mtk_pll_ops);
34162306a36Sopenharmony_ci	if (IS_ERR(hw))
34262306a36Sopenharmony_ci		kfree(pll);
34362306a36Sopenharmony_ci
34462306a36Sopenharmony_ci	return hw;
34562306a36Sopenharmony_ci}
34662306a36Sopenharmony_ci
34762306a36Sopenharmony_civoid mtk_clk_unregister_pll(struct clk_hw *hw)
34862306a36Sopenharmony_ci{
34962306a36Sopenharmony_ci	struct mtk_clk_pll *pll;
35062306a36Sopenharmony_ci
35162306a36Sopenharmony_ci	if (!hw)
35262306a36Sopenharmony_ci		return;
35362306a36Sopenharmony_ci
35462306a36Sopenharmony_ci	pll = to_mtk_clk_pll(hw);
35562306a36Sopenharmony_ci
35662306a36Sopenharmony_ci	clk_hw_unregister(hw);
35762306a36Sopenharmony_ci	kfree(pll);
35862306a36Sopenharmony_ci}
35962306a36Sopenharmony_ci
36062306a36Sopenharmony_ciint mtk_clk_register_plls(struct device_node *node,
36162306a36Sopenharmony_ci			  const struct mtk_pll_data *plls, int num_plls,
36262306a36Sopenharmony_ci			  struct clk_hw_onecell_data *clk_data)
36362306a36Sopenharmony_ci{
36462306a36Sopenharmony_ci	void __iomem *base;
36562306a36Sopenharmony_ci	int i;
36662306a36Sopenharmony_ci	struct clk_hw *hw;
36762306a36Sopenharmony_ci
36862306a36Sopenharmony_ci	base = of_iomap(node, 0);
36962306a36Sopenharmony_ci	if (!base) {
37062306a36Sopenharmony_ci		pr_err("%s(): ioremap failed\n", __func__);
37162306a36Sopenharmony_ci		return -EINVAL;
37262306a36Sopenharmony_ci	}
37362306a36Sopenharmony_ci
37462306a36Sopenharmony_ci	for (i = 0; i < num_plls; i++) {
37562306a36Sopenharmony_ci		const struct mtk_pll_data *pll = &plls[i];
37662306a36Sopenharmony_ci
37762306a36Sopenharmony_ci		if (!IS_ERR_OR_NULL(clk_data->hws[pll->id])) {
37862306a36Sopenharmony_ci			pr_warn("%pOF: Trying to register duplicate clock ID: %d\n",
37962306a36Sopenharmony_ci				node, pll->id);
38062306a36Sopenharmony_ci			continue;
38162306a36Sopenharmony_ci		}
38262306a36Sopenharmony_ci
38362306a36Sopenharmony_ci		hw = mtk_clk_register_pll(pll, base);
38462306a36Sopenharmony_ci
38562306a36Sopenharmony_ci		if (IS_ERR(hw)) {
38662306a36Sopenharmony_ci			pr_err("Failed to register clk %s: %pe\n", pll->name,
38762306a36Sopenharmony_ci			       hw);
38862306a36Sopenharmony_ci			goto err;
38962306a36Sopenharmony_ci		}
39062306a36Sopenharmony_ci
39162306a36Sopenharmony_ci		clk_data->hws[pll->id] = hw;
39262306a36Sopenharmony_ci	}
39362306a36Sopenharmony_ci
39462306a36Sopenharmony_ci	return 0;
39562306a36Sopenharmony_ci
39662306a36Sopenharmony_cierr:
39762306a36Sopenharmony_ci	while (--i >= 0) {
39862306a36Sopenharmony_ci		const struct mtk_pll_data *pll = &plls[i];
39962306a36Sopenharmony_ci
40062306a36Sopenharmony_ci		mtk_clk_unregister_pll(clk_data->hws[pll->id]);
40162306a36Sopenharmony_ci		clk_data->hws[pll->id] = ERR_PTR(-ENOENT);
40262306a36Sopenharmony_ci	}
40362306a36Sopenharmony_ci
40462306a36Sopenharmony_ci	iounmap(base);
40562306a36Sopenharmony_ci
40662306a36Sopenharmony_ci	return PTR_ERR(hw);
40762306a36Sopenharmony_ci}
40862306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mtk_clk_register_plls);
40962306a36Sopenharmony_ci
41062306a36Sopenharmony_ci__iomem void *mtk_clk_pll_get_base(struct clk_hw *hw,
41162306a36Sopenharmony_ci				   const struct mtk_pll_data *data)
41262306a36Sopenharmony_ci{
41362306a36Sopenharmony_ci	struct mtk_clk_pll *pll = to_mtk_clk_pll(hw);
41462306a36Sopenharmony_ci
41562306a36Sopenharmony_ci	return pll->base_addr - data->reg;
41662306a36Sopenharmony_ci}
41762306a36Sopenharmony_ci
41862306a36Sopenharmony_civoid mtk_clk_unregister_plls(const struct mtk_pll_data *plls, int num_plls,
41962306a36Sopenharmony_ci			     struct clk_hw_onecell_data *clk_data)
42062306a36Sopenharmony_ci{
42162306a36Sopenharmony_ci	__iomem void *base = NULL;
42262306a36Sopenharmony_ci	int i;
42362306a36Sopenharmony_ci
42462306a36Sopenharmony_ci	if (!clk_data)
42562306a36Sopenharmony_ci		return;
42662306a36Sopenharmony_ci
42762306a36Sopenharmony_ci	for (i = num_plls; i > 0; i--) {
42862306a36Sopenharmony_ci		const struct mtk_pll_data *pll = &plls[i - 1];
42962306a36Sopenharmony_ci
43062306a36Sopenharmony_ci		if (IS_ERR_OR_NULL(clk_data->hws[pll->id]))
43162306a36Sopenharmony_ci			continue;
43262306a36Sopenharmony_ci
43362306a36Sopenharmony_ci		/*
43462306a36Sopenharmony_ci		 * This is quite ugly but unfortunately the clks don't have
43562306a36Sopenharmony_ci		 * any device tied to them, so there's no place to store the
43662306a36Sopenharmony_ci		 * pointer to the I/O region base address. We have to fetch
43762306a36Sopenharmony_ci		 * it from one of the registered clks.
43862306a36Sopenharmony_ci		 */
43962306a36Sopenharmony_ci		base = mtk_clk_pll_get_base(clk_data->hws[pll->id], pll);
44062306a36Sopenharmony_ci
44162306a36Sopenharmony_ci		mtk_clk_unregister_pll(clk_data->hws[pll->id]);
44262306a36Sopenharmony_ci		clk_data->hws[pll->id] = ERR_PTR(-ENOENT);
44362306a36Sopenharmony_ci	}
44462306a36Sopenharmony_ci
44562306a36Sopenharmony_ci	iounmap(base);
44662306a36Sopenharmony_ci}
44762306a36Sopenharmony_ciEXPORT_SYMBOL_GPL(mtk_clk_unregister_plls);
44862306a36Sopenharmony_ci
44962306a36Sopenharmony_ciMODULE_LICENSE("GPL");
450