xref: /openbmc/linux/drivers/clk/mmp/clk-mix.c (revision 3bb16560)
1*3bb16560SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2ee81f4eeSChao Xie /*
3ee81f4eeSChao Xie  * mmp mix(div and mux) clock operation source file
4ee81f4eeSChao Xie  *
5ee81f4eeSChao Xie  * Copyright (C) 2014 Marvell
6ee81f4eeSChao Xie  * Chao Xie <chao.xie@marvell.com>
7ee81f4eeSChao Xie  */
8ee81f4eeSChao Xie 
9ee81f4eeSChao Xie #include <linux/clk-provider.h>
10ee81f4eeSChao Xie #include <linux/slab.h>
11ee81f4eeSChao Xie #include <linux/io.h>
12ee81f4eeSChao Xie #include <linux/err.h>
13ee81f4eeSChao Xie 
14ee81f4eeSChao Xie #include "clk.h"
15ee81f4eeSChao Xie 
16ee81f4eeSChao Xie /*
17ee81f4eeSChao Xie  * The mix clock is a clock combined mux and div type clock.
18ee81f4eeSChao Xie  * Because the div field and mux field need to be set at same
19ee81f4eeSChao Xie  * time, we can not divide it into 2 types of clock
20ee81f4eeSChao Xie  */
21ee81f4eeSChao Xie 
22ee81f4eeSChao Xie #define to_clk_mix(hw)	container_of(hw, struct mmp_clk_mix, hw)
23ee81f4eeSChao Xie 
_get_maxdiv(struct mmp_clk_mix * mix)24ee81f4eeSChao Xie static unsigned int _get_maxdiv(struct mmp_clk_mix *mix)
25ee81f4eeSChao Xie {
26ee81f4eeSChao Xie 	unsigned int div_mask = (1 << mix->reg_info.width_div) - 1;
27ee81f4eeSChao Xie 	unsigned int maxdiv = 0;
28ee81f4eeSChao Xie 	struct clk_div_table *clkt;
29ee81f4eeSChao Xie 
30ee81f4eeSChao Xie 	if (mix->div_flags & CLK_DIVIDER_ONE_BASED)
31ee81f4eeSChao Xie 		return div_mask;
32ee81f4eeSChao Xie 	if (mix->div_flags & CLK_DIVIDER_POWER_OF_TWO)
33ee81f4eeSChao Xie 		return 1 << div_mask;
34ee81f4eeSChao Xie 	if (mix->div_table) {
35ee81f4eeSChao Xie 		for (clkt = mix->div_table; clkt->div; clkt++)
36ee81f4eeSChao Xie 			if (clkt->div > maxdiv)
37ee81f4eeSChao Xie 				maxdiv = clkt->div;
38ee81f4eeSChao Xie 		return maxdiv;
39ee81f4eeSChao Xie 	}
40ee81f4eeSChao Xie 	return div_mask + 1;
41ee81f4eeSChao Xie }
42ee81f4eeSChao Xie 
_get_div(struct mmp_clk_mix * mix,unsigned int val)43ee81f4eeSChao Xie static unsigned int _get_div(struct mmp_clk_mix *mix, unsigned int val)
44ee81f4eeSChao Xie {
45ee81f4eeSChao Xie 	struct clk_div_table *clkt;
46ee81f4eeSChao Xie 
47ee81f4eeSChao Xie 	if (mix->div_flags & CLK_DIVIDER_ONE_BASED)
48ee81f4eeSChao Xie 		return val;
49ee81f4eeSChao Xie 	if (mix->div_flags & CLK_DIVIDER_POWER_OF_TWO)
50ee81f4eeSChao Xie 		return 1 << val;
51ee81f4eeSChao Xie 	if (mix->div_table) {
52ee81f4eeSChao Xie 		for (clkt = mix->div_table; clkt->div; clkt++)
53ee81f4eeSChao Xie 			if (clkt->val == val)
54ee81f4eeSChao Xie 				return clkt->div;
55ee81f4eeSChao Xie 		if (clkt->div == 0)
56ee81f4eeSChao Xie 			return 0;
57ee81f4eeSChao Xie 	}
58ee81f4eeSChao Xie 	return val + 1;
59ee81f4eeSChao Xie }
60ee81f4eeSChao Xie 
_get_mux(struct mmp_clk_mix * mix,unsigned int val)61ee81f4eeSChao Xie static unsigned int _get_mux(struct mmp_clk_mix *mix, unsigned int val)
62ee81f4eeSChao Xie {
63497295afSStephen Boyd 	int num_parents = clk_hw_get_num_parents(&mix->hw);
64ee81f4eeSChao Xie 	int i;
65ee81f4eeSChao Xie 
66ee81f4eeSChao Xie 	if (mix->mux_flags & CLK_MUX_INDEX_BIT)
67ee81f4eeSChao Xie 		return ffs(val) - 1;
68ee81f4eeSChao Xie 	if (mix->mux_flags & CLK_MUX_INDEX_ONE)
69ee81f4eeSChao Xie 		return val - 1;
70ee81f4eeSChao Xie 	if (mix->mux_table) {
71ee81f4eeSChao Xie 		for (i = 0; i < num_parents; i++)
72ee81f4eeSChao Xie 			if (mix->mux_table[i] == val)
73ee81f4eeSChao Xie 				return i;
74ee81f4eeSChao Xie 		if (i == num_parents)
75ee81f4eeSChao Xie 			return 0;
76ee81f4eeSChao Xie 	}
77ee81f4eeSChao Xie 
78ee81f4eeSChao Xie 	return val;
79ee81f4eeSChao Xie }
_get_div_val(struct mmp_clk_mix * mix,unsigned int div)80ee81f4eeSChao Xie static unsigned int _get_div_val(struct mmp_clk_mix *mix, unsigned int div)
81ee81f4eeSChao Xie {
82ee81f4eeSChao Xie 	struct clk_div_table *clkt;
83ee81f4eeSChao Xie 
84ee81f4eeSChao Xie 	if (mix->div_flags & CLK_DIVIDER_ONE_BASED)
85ee81f4eeSChao Xie 		return div;
86ee81f4eeSChao Xie 	if (mix->div_flags & CLK_DIVIDER_POWER_OF_TWO)
87ee81f4eeSChao Xie 		return __ffs(div);
88ee81f4eeSChao Xie 	if (mix->div_table) {
89ee81f4eeSChao Xie 		for (clkt = mix->div_table; clkt->div; clkt++)
90ee81f4eeSChao Xie 			if (clkt->div == div)
91ee81f4eeSChao Xie 				return clkt->val;
92ee81f4eeSChao Xie 		if (clkt->div == 0)
93ee81f4eeSChao Xie 			return 0;
94ee81f4eeSChao Xie 	}
95ee81f4eeSChao Xie 
96ee81f4eeSChao Xie 	return div - 1;
97ee81f4eeSChao Xie }
98ee81f4eeSChao Xie 
_get_mux_val(struct mmp_clk_mix * mix,unsigned int mux)99ee81f4eeSChao Xie static unsigned int _get_mux_val(struct mmp_clk_mix *mix, unsigned int mux)
100ee81f4eeSChao Xie {
101ee81f4eeSChao Xie 	if (mix->mux_table)
102ee81f4eeSChao Xie 		return mix->mux_table[mux];
103ee81f4eeSChao Xie 
104ee81f4eeSChao Xie 	return mux;
105ee81f4eeSChao Xie }
106ee81f4eeSChao Xie 
_filter_clk_table(struct mmp_clk_mix * mix,struct mmp_clk_mix_clk_table * table,unsigned int table_size)107ee81f4eeSChao Xie static void _filter_clk_table(struct mmp_clk_mix *mix,
108ee81f4eeSChao Xie 				struct mmp_clk_mix_clk_table *table,
109ee81f4eeSChao Xie 				unsigned int table_size)
110ee81f4eeSChao Xie {
111ee81f4eeSChao Xie 	int i;
112ee81f4eeSChao Xie 	struct mmp_clk_mix_clk_table *item;
113aef28cb6SStephen Boyd 	struct clk_hw *parent, *hw;
114ee81f4eeSChao Xie 	unsigned long parent_rate;
115ee81f4eeSChao Xie 
116aef28cb6SStephen Boyd 	hw = &mix->hw;
117ee81f4eeSChao Xie 
118ee81f4eeSChao Xie 	for (i = 0; i < table_size; i++) {
119ee81f4eeSChao Xie 		item = &table[i];
120aef28cb6SStephen Boyd 		parent = clk_hw_get_parent_by_index(hw, item->parent_index);
121aef28cb6SStephen Boyd 		parent_rate = clk_hw_get_rate(parent);
122ee81f4eeSChao Xie 		if (parent_rate % item->rate) {
123ee81f4eeSChao Xie 			item->valid = 0;
124ee81f4eeSChao Xie 		} else {
125ee81f4eeSChao Xie 			item->divisor = parent_rate / item->rate;
126ee81f4eeSChao Xie 			item->valid = 1;
127ee81f4eeSChao Xie 		}
128ee81f4eeSChao Xie 	}
129ee81f4eeSChao Xie }
130ee81f4eeSChao Xie 
_set_rate(struct mmp_clk_mix * mix,u32 mux_val,u32 div_val,unsigned int change_mux,unsigned int change_div)131ee81f4eeSChao Xie static int _set_rate(struct mmp_clk_mix *mix, u32 mux_val, u32 div_val,
132ee81f4eeSChao Xie 			unsigned int change_mux, unsigned int change_div)
133ee81f4eeSChao Xie {
134ee81f4eeSChao Xie 	struct mmp_clk_mix_reg_info *ri = &mix->reg_info;
135ee81f4eeSChao Xie 	u8 width, shift;
136ee81f4eeSChao Xie 	u32 mux_div, fc_req;
137ee81f4eeSChao Xie 	int ret, timeout = 50;
138ee81f4eeSChao Xie 	unsigned long flags = 0;
139ee81f4eeSChao Xie 
140ee81f4eeSChao Xie 	if (!change_mux && !change_div)
141ee81f4eeSChao Xie 		return -EINVAL;
142ee81f4eeSChao Xie 
143ee81f4eeSChao Xie 	if (mix->lock)
144ee81f4eeSChao Xie 		spin_lock_irqsave(mix->lock, flags);
145ee81f4eeSChao Xie 
146ee81f4eeSChao Xie 	if (mix->type == MMP_CLK_MIX_TYPE_V1
147ee81f4eeSChao Xie 		|| mix->type == MMP_CLK_MIX_TYPE_V2)
148ee81f4eeSChao Xie 		mux_div = readl(ri->reg_clk_ctrl);
149ee81f4eeSChao Xie 	else
150ee81f4eeSChao Xie 		mux_div = readl(ri->reg_clk_sel);
151ee81f4eeSChao Xie 
152ee81f4eeSChao Xie 	if (change_div) {
153ee81f4eeSChao Xie 		width = ri->width_div;
154ee81f4eeSChao Xie 		shift = ri->shift_div;
155ee81f4eeSChao Xie 		mux_div &= ~MMP_CLK_BITS_MASK(width, shift);
156ee81f4eeSChao Xie 		mux_div |= MMP_CLK_BITS_SET_VAL(div_val, width, shift);
157ee81f4eeSChao Xie 	}
158ee81f4eeSChao Xie 
159ee81f4eeSChao Xie 	if (change_mux) {
160ee81f4eeSChao Xie 		width = ri->width_mux;
161ee81f4eeSChao Xie 		shift = ri->shift_mux;
162ee81f4eeSChao Xie 		mux_div &= ~MMP_CLK_BITS_MASK(width, shift);
163ee81f4eeSChao Xie 		mux_div |= MMP_CLK_BITS_SET_VAL(mux_val, width, shift);
164ee81f4eeSChao Xie 	}
165ee81f4eeSChao Xie 
166ee81f4eeSChao Xie 	if (mix->type == MMP_CLK_MIX_TYPE_V1) {
167ee81f4eeSChao Xie 		writel(mux_div, ri->reg_clk_ctrl);
168ee81f4eeSChao Xie 	} else if (mix->type == MMP_CLK_MIX_TYPE_V2) {
169ee81f4eeSChao Xie 		mux_div |= (1 << ri->bit_fc);
170ee81f4eeSChao Xie 		writel(mux_div, ri->reg_clk_ctrl);
171ee81f4eeSChao Xie 
172ee81f4eeSChao Xie 		do {
173ee81f4eeSChao Xie 			fc_req = readl(ri->reg_clk_ctrl);
174ee81f4eeSChao Xie 			timeout--;
175ee81f4eeSChao Xie 			if (!(fc_req & (1 << ri->bit_fc)))
176ee81f4eeSChao Xie 				break;
177ee81f4eeSChao Xie 		} while (timeout);
178ee81f4eeSChao Xie 
179ee81f4eeSChao Xie 		if (timeout == 0) {
180ee81f4eeSChao Xie 			pr_err("%s:%s cannot do frequency change\n",
181aef28cb6SStephen Boyd 				__func__, clk_hw_get_name(&mix->hw));
182ee81f4eeSChao Xie 			ret = -EBUSY;
183ee81f4eeSChao Xie 			goto error;
184ee81f4eeSChao Xie 		}
185ee81f4eeSChao Xie 	} else {
186ee81f4eeSChao Xie 		fc_req = readl(ri->reg_clk_ctrl);
187ee81f4eeSChao Xie 		fc_req |= 1 << ri->bit_fc;
188ee81f4eeSChao Xie 		writel(fc_req, ri->reg_clk_ctrl);
189ee81f4eeSChao Xie 		writel(mux_div, ri->reg_clk_sel);
190ee81f4eeSChao Xie 		fc_req &= ~(1 << ri->bit_fc);
191ee81f4eeSChao Xie 	}
192ee81f4eeSChao Xie 
193ee81f4eeSChao Xie 	ret = 0;
194ee81f4eeSChao Xie error:
195ee81f4eeSChao Xie 	if (mix->lock)
196ee81f4eeSChao Xie 		spin_unlock_irqrestore(mix->lock, flags);
197ee81f4eeSChao Xie 
198ee81f4eeSChao Xie 	return ret;
199ee81f4eeSChao Xie }
200ee81f4eeSChao Xie 
mmp_clk_mix_determine_rate(struct clk_hw * hw,struct clk_rate_request * req)2010817b62cSBoris Brezillon static int mmp_clk_mix_determine_rate(struct clk_hw *hw,
2020817b62cSBoris Brezillon 				      struct clk_rate_request *req)
203ee81f4eeSChao Xie {
204ee81f4eeSChao Xie 	struct mmp_clk_mix *mix = to_clk_mix(hw);
205ee81f4eeSChao Xie 	struct mmp_clk_mix_clk_table *item;
206aef28cb6SStephen Boyd 	struct clk_hw *parent, *parent_best;
207ee81f4eeSChao Xie 	unsigned long parent_rate, mix_rate, mix_rate_best, parent_rate_best;
208ee81f4eeSChao Xie 	unsigned long gap, gap_best;
209ee81f4eeSChao Xie 	u32 div_val_max;
210ee81f4eeSChao Xie 	unsigned int div;
211ee81f4eeSChao Xie 	int i, j;
212ee81f4eeSChao Xie 
213ee81f4eeSChao Xie 
214ee81f4eeSChao Xie 	mix_rate_best = 0;
215ee81f4eeSChao Xie 	parent_rate_best = 0;
21657d866e6SBoris Brezillon 	gap_best = ULONG_MAX;
217ee81f4eeSChao Xie 	parent_best = NULL;
218ee81f4eeSChao Xie 
219ee81f4eeSChao Xie 	if (mix->table) {
220ee81f4eeSChao Xie 		for (i = 0; i < mix->table_size; i++) {
221ee81f4eeSChao Xie 			item = &mix->table[i];
222ee81f4eeSChao Xie 			if (item->valid == 0)
223ee81f4eeSChao Xie 				continue;
224aef28cb6SStephen Boyd 			parent = clk_hw_get_parent_by_index(hw,
225ee81f4eeSChao Xie 							item->parent_index);
226aef28cb6SStephen Boyd 			parent_rate = clk_hw_get_rate(parent);
227ee81f4eeSChao Xie 			mix_rate = parent_rate / item->divisor;
2280817b62cSBoris Brezillon 			gap = abs(mix_rate - req->rate);
2297a3aad90SMarkus Elfring 			if (!parent_best || gap < gap_best) {
230ee81f4eeSChao Xie 				parent_best = parent;
231ee81f4eeSChao Xie 				parent_rate_best = parent_rate;
232ee81f4eeSChao Xie 				mix_rate_best = mix_rate;
233ee81f4eeSChao Xie 				gap_best = gap;
234ee81f4eeSChao Xie 				if (gap_best == 0)
235ee81f4eeSChao Xie 					goto found;
236ee81f4eeSChao Xie 			}
237ee81f4eeSChao Xie 		}
238ee81f4eeSChao Xie 	} else {
239497295afSStephen Boyd 		for (i = 0; i < clk_hw_get_num_parents(hw); i++) {
240aef28cb6SStephen Boyd 			parent = clk_hw_get_parent_by_index(hw, i);
241aef28cb6SStephen Boyd 			parent_rate = clk_hw_get_rate(parent);
242ee81f4eeSChao Xie 			div_val_max = _get_maxdiv(mix);
243ee81f4eeSChao Xie 			for (j = 0; j < div_val_max; j++) {
244ee81f4eeSChao Xie 				div = _get_div(mix, j);
245ee81f4eeSChao Xie 				mix_rate = parent_rate / div;
2460817b62cSBoris Brezillon 				gap = abs(mix_rate - req->rate);
2477a3aad90SMarkus Elfring 				if (!parent_best || gap < gap_best) {
248ee81f4eeSChao Xie 					parent_best = parent;
249ee81f4eeSChao Xie 					parent_rate_best = parent_rate;
250ee81f4eeSChao Xie 					mix_rate_best = mix_rate;
251ee81f4eeSChao Xie 					gap_best = gap;
252ee81f4eeSChao Xie 					if (gap_best == 0)
253ee81f4eeSChao Xie 						goto found;
254ee81f4eeSChao Xie 				}
255ee81f4eeSChao Xie 			}
256ee81f4eeSChao Xie 		}
257ee81f4eeSChao Xie 	}
258ee81f4eeSChao Xie 
259ee81f4eeSChao Xie found:
26057d866e6SBoris Brezillon 	if (!parent_best)
26157d866e6SBoris Brezillon 		return -EINVAL;
26257d866e6SBoris Brezillon 
2630817b62cSBoris Brezillon 	req->best_parent_rate = parent_rate_best;
264aef28cb6SStephen Boyd 	req->best_parent_hw = parent_best;
2650817b62cSBoris Brezillon 	req->rate = mix_rate_best;
266ee81f4eeSChao Xie 
2670817b62cSBoris Brezillon 	return 0;
268ee81f4eeSChao Xie }
269ee81f4eeSChao Xie 
mmp_clk_mix_set_rate_and_parent(struct clk_hw * hw,unsigned long rate,unsigned long parent_rate,u8 index)270ee81f4eeSChao Xie static int mmp_clk_mix_set_rate_and_parent(struct clk_hw *hw,
271ee81f4eeSChao Xie 						unsigned long rate,
272ee81f4eeSChao Xie 						unsigned long parent_rate,
273ee81f4eeSChao Xie 						u8 index)
274ee81f4eeSChao Xie {
275ee81f4eeSChao Xie 	struct mmp_clk_mix *mix = to_clk_mix(hw);
276ee81f4eeSChao Xie 	unsigned int div;
277ee81f4eeSChao Xie 	u32 div_val, mux_val;
278ee81f4eeSChao Xie 
279ee81f4eeSChao Xie 	div = parent_rate / rate;
280ee81f4eeSChao Xie 	div_val = _get_div_val(mix, div);
281ee81f4eeSChao Xie 	mux_val = _get_mux_val(mix, index);
282ee81f4eeSChao Xie 
283ee81f4eeSChao Xie 	return _set_rate(mix, mux_val, div_val, 1, 1);
284ee81f4eeSChao Xie }
285ee81f4eeSChao Xie 
mmp_clk_mix_get_parent(struct clk_hw * hw)286ee81f4eeSChao Xie static u8 mmp_clk_mix_get_parent(struct clk_hw *hw)
287ee81f4eeSChao Xie {
288ee81f4eeSChao Xie 	struct mmp_clk_mix *mix = to_clk_mix(hw);
289ee81f4eeSChao Xie 	struct mmp_clk_mix_reg_info *ri = &mix->reg_info;
290ee81f4eeSChao Xie 	unsigned long flags = 0;
291ee81f4eeSChao Xie 	u32 mux_div = 0;
292ee81f4eeSChao Xie 	u8 width, shift;
293ee81f4eeSChao Xie 	u32 mux_val;
294ee81f4eeSChao Xie 
295ee81f4eeSChao Xie 	if (mix->lock)
296ee81f4eeSChao Xie 		spin_lock_irqsave(mix->lock, flags);
297ee81f4eeSChao Xie 
298ee81f4eeSChao Xie 	if (mix->type == MMP_CLK_MIX_TYPE_V1
299ee81f4eeSChao Xie 		|| mix->type == MMP_CLK_MIX_TYPE_V2)
300ee81f4eeSChao Xie 		mux_div = readl(ri->reg_clk_ctrl);
301ee81f4eeSChao Xie 	else
302ee81f4eeSChao Xie 		mux_div = readl(ri->reg_clk_sel);
303ee81f4eeSChao Xie 
304ee81f4eeSChao Xie 	if (mix->lock)
305ee81f4eeSChao Xie 		spin_unlock_irqrestore(mix->lock, flags);
306ee81f4eeSChao Xie 
307ee81f4eeSChao Xie 	width = mix->reg_info.width_mux;
308ee81f4eeSChao Xie 	shift = mix->reg_info.shift_mux;
309ee81f4eeSChao Xie 
310ee81f4eeSChao Xie 	mux_val = MMP_CLK_BITS_GET_VAL(mux_div, width, shift);
311ee81f4eeSChao Xie 
312ee81f4eeSChao Xie 	return _get_mux(mix, mux_val);
313ee81f4eeSChao Xie }
314ee81f4eeSChao Xie 
mmp_clk_mix_recalc_rate(struct clk_hw * hw,unsigned long parent_rate)315ee81f4eeSChao Xie static unsigned long mmp_clk_mix_recalc_rate(struct clk_hw *hw,
316ee81f4eeSChao Xie 					unsigned long parent_rate)
317ee81f4eeSChao Xie {
318ee81f4eeSChao Xie 	struct mmp_clk_mix *mix = to_clk_mix(hw);
319ee81f4eeSChao Xie 	struct mmp_clk_mix_reg_info *ri = &mix->reg_info;
320ee81f4eeSChao Xie 	unsigned long flags = 0;
321ee81f4eeSChao Xie 	u32 mux_div = 0;
322ee81f4eeSChao Xie 	u8 width, shift;
323ee81f4eeSChao Xie 	unsigned int div;
324ee81f4eeSChao Xie 
325ee81f4eeSChao Xie 	if (mix->lock)
326ee81f4eeSChao Xie 		spin_lock_irqsave(mix->lock, flags);
327ee81f4eeSChao Xie 
328ee81f4eeSChao Xie 	if (mix->type == MMP_CLK_MIX_TYPE_V1
329ee81f4eeSChao Xie 		|| mix->type == MMP_CLK_MIX_TYPE_V2)
330ee81f4eeSChao Xie 		mux_div = readl(ri->reg_clk_ctrl);
331ee81f4eeSChao Xie 	else
332ee81f4eeSChao Xie 		mux_div = readl(ri->reg_clk_sel);
333ee81f4eeSChao Xie 
334ee81f4eeSChao Xie 	if (mix->lock)
335ee81f4eeSChao Xie 		spin_unlock_irqrestore(mix->lock, flags);
336ee81f4eeSChao Xie 
337ee81f4eeSChao Xie 	width = mix->reg_info.width_div;
338ee81f4eeSChao Xie 	shift = mix->reg_info.shift_div;
339ee81f4eeSChao Xie 
340ee81f4eeSChao Xie 	div = _get_div(mix, MMP_CLK_BITS_GET_VAL(mux_div, width, shift));
341ee81f4eeSChao Xie 
342ee81f4eeSChao Xie 	return parent_rate / div;
343ee81f4eeSChao Xie }
344ee81f4eeSChao Xie 
mmp_clk_set_parent(struct clk_hw * hw,u8 index)345ee81f4eeSChao Xie static int mmp_clk_set_parent(struct clk_hw *hw, u8 index)
346ee81f4eeSChao Xie {
347ee81f4eeSChao Xie 	struct mmp_clk_mix *mix = to_clk_mix(hw);
348ee81f4eeSChao Xie 	struct mmp_clk_mix_clk_table *item;
349ee81f4eeSChao Xie 	int i;
350ee81f4eeSChao Xie 	u32 div_val, mux_val;
351ee81f4eeSChao Xie 
352ee81f4eeSChao Xie 	if (mix->table) {
353ee81f4eeSChao Xie 		for (i = 0; i < mix->table_size; i++) {
354ee81f4eeSChao Xie 			item = &mix->table[i];
355ee81f4eeSChao Xie 			if (item->valid == 0)
356ee81f4eeSChao Xie 				continue;
357ee81f4eeSChao Xie 			if (item->parent_index == index)
358ee81f4eeSChao Xie 				break;
359ee81f4eeSChao Xie 		}
360ee81f4eeSChao Xie 		if (i < mix->table_size) {
361ee81f4eeSChao Xie 			div_val = _get_div_val(mix, item->divisor);
362ee81f4eeSChao Xie 			mux_val = _get_mux_val(mix, item->parent_index);
363ee81f4eeSChao Xie 		} else
364ee81f4eeSChao Xie 			return -EINVAL;
365ee81f4eeSChao Xie 	} else {
366ee81f4eeSChao Xie 		mux_val = _get_mux_val(mix, index);
367ee81f4eeSChao Xie 		div_val = 0;
368ee81f4eeSChao Xie 	}
369ee81f4eeSChao Xie 
370ee81f4eeSChao Xie 	return _set_rate(mix, mux_val, div_val, 1, div_val ? 1 : 0);
371ee81f4eeSChao Xie }
372ee81f4eeSChao Xie 
mmp_clk_set_rate(struct clk_hw * hw,unsigned long rate,unsigned long best_parent_rate)373ee81f4eeSChao Xie static int mmp_clk_set_rate(struct clk_hw *hw, unsigned long rate,
374ee81f4eeSChao Xie 				unsigned long best_parent_rate)
375ee81f4eeSChao Xie {
376ee81f4eeSChao Xie 	struct mmp_clk_mix *mix = to_clk_mix(hw);
377ee81f4eeSChao Xie 	struct mmp_clk_mix_clk_table *item;
378ee81f4eeSChao Xie 	unsigned long parent_rate;
379ee81f4eeSChao Xie 	unsigned int best_divisor;
380aef28cb6SStephen Boyd 	struct clk_hw *parent;
381ee81f4eeSChao Xie 	int i;
382ee81f4eeSChao Xie 
383ee81f4eeSChao Xie 	best_divisor = best_parent_rate / rate;
384ee81f4eeSChao Xie 
385ee81f4eeSChao Xie 	if (mix->table) {
386ee81f4eeSChao Xie 		for (i = 0; i < mix->table_size; i++) {
387ee81f4eeSChao Xie 			item = &mix->table[i];
388ee81f4eeSChao Xie 			if (item->valid == 0)
389ee81f4eeSChao Xie 				continue;
390aef28cb6SStephen Boyd 			parent = clk_hw_get_parent_by_index(hw,
391ee81f4eeSChao Xie 							item->parent_index);
392aef28cb6SStephen Boyd 			parent_rate = clk_hw_get_rate(parent);
393ee81f4eeSChao Xie 			if (parent_rate == best_parent_rate
394ee81f4eeSChao Xie 				&& item->divisor == best_divisor)
395ee81f4eeSChao Xie 				break;
396ee81f4eeSChao Xie 		}
397ee81f4eeSChao Xie 		if (i < mix->table_size)
398ee81f4eeSChao Xie 			return _set_rate(mix,
399ee81f4eeSChao Xie 					_get_mux_val(mix, item->parent_index),
400ee81f4eeSChao Xie 					_get_div_val(mix, item->divisor),
401ee81f4eeSChao Xie 					1, 1);
402ee81f4eeSChao Xie 		else
403ee81f4eeSChao Xie 			return -EINVAL;
404ee81f4eeSChao Xie 	} else {
405497295afSStephen Boyd 		for (i = 0; i < clk_hw_get_num_parents(hw); i++) {
406aef28cb6SStephen Boyd 			parent = clk_hw_get_parent_by_index(hw, i);
407aef28cb6SStephen Boyd 			parent_rate = clk_hw_get_rate(parent);
408ee81f4eeSChao Xie 			if (parent_rate == best_parent_rate)
409ee81f4eeSChao Xie 				break;
410ee81f4eeSChao Xie 		}
411497295afSStephen Boyd 		if (i < clk_hw_get_num_parents(hw))
412ee81f4eeSChao Xie 			return _set_rate(mix, _get_mux_val(mix, i),
413ee81f4eeSChao Xie 					_get_div_val(mix, best_divisor), 1, 1);
414ee81f4eeSChao Xie 		else
415ee81f4eeSChao Xie 			return -EINVAL;
416ee81f4eeSChao Xie 	}
417ee81f4eeSChao Xie }
418ee81f4eeSChao Xie 
mmp_clk_mix_init(struct clk_hw * hw)41989d079dcSJerome Brunet static int mmp_clk_mix_init(struct clk_hw *hw)
420ee81f4eeSChao Xie {
421ee81f4eeSChao Xie 	struct mmp_clk_mix *mix = to_clk_mix(hw);
422ee81f4eeSChao Xie 
423ee81f4eeSChao Xie 	if (mix->table)
424ee81f4eeSChao Xie 		_filter_clk_table(mix, mix->table, mix->table_size);
42589d079dcSJerome Brunet 
42689d079dcSJerome Brunet 	return 0;
427ee81f4eeSChao Xie }
428ee81f4eeSChao Xie 
429ee81f4eeSChao Xie const struct clk_ops mmp_clk_mix_ops = {
430ee81f4eeSChao Xie 	.determine_rate = mmp_clk_mix_determine_rate,
431ee81f4eeSChao Xie 	.set_rate_and_parent = mmp_clk_mix_set_rate_and_parent,
432ee81f4eeSChao Xie 	.set_rate = mmp_clk_set_rate,
433ee81f4eeSChao Xie 	.set_parent = mmp_clk_set_parent,
434ee81f4eeSChao Xie 	.get_parent = mmp_clk_mix_get_parent,
435ee81f4eeSChao Xie 	.recalc_rate = mmp_clk_mix_recalc_rate,
436ee81f4eeSChao Xie 	.init = mmp_clk_mix_init,
437ee81f4eeSChao Xie };
438ee81f4eeSChao Xie 
mmp_clk_register_mix(struct device * dev,const char * name,const char * const * parent_names,u8 num_parents,unsigned long flags,struct mmp_clk_mix_config * config,spinlock_t * lock)439ee81f4eeSChao Xie struct clk *mmp_clk_register_mix(struct device *dev,
440ee81f4eeSChao Xie 					const char *name,
441cb8dbfe8SLubomir Rintel 					const char * const *parent_names,
442ee81f4eeSChao Xie 					u8 num_parents,
443ee81f4eeSChao Xie 					unsigned long flags,
444ee81f4eeSChao Xie 					struct mmp_clk_mix_config *config,
445ee81f4eeSChao Xie 					spinlock_t *lock)
446ee81f4eeSChao Xie {
447ee81f4eeSChao Xie 	struct mmp_clk_mix *mix;
448ee81f4eeSChao Xie 	struct clk *clk;
449ee81f4eeSChao Xie 	struct clk_init_data init;
450ee81f4eeSChao Xie 	size_t table_bytes;
451ee81f4eeSChao Xie 
452ee81f4eeSChao Xie 	mix = kzalloc(sizeof(*mix), GFP_KERNEL);
4531cc36f73SMarkus Elfring 	if (!mix)
454ee81f4eeSChao Xie 		return ERR_PTR(-ENOMEM);
455ee81f4eeSChao Xie 
456ee81f4eeSChao Xie 	init.name = name;
457ee81f4eeSChao Xie 	init.flags = flags | CLK_GET_RATE_NOCACHE;
458ee81f4eeSChao Xie 	init.parent_names = parent_names;
459ee81f4eeSChao Xie 	init.num_parents = num_parents;
460ee81f4eeSChao Xie 	init.ops = &mmp_clk_mix_ops;
461ee81f4eeSChao Xie 
462ee81f4eeSChao Xie 	memcpy(&mix->reg_info, &config->reg_info, sizeof(config->reg_info));
463ee81f4eeSChao Xie 	if (config->table) {
464ee81f4eeSChao Xie 		table_bytes = sizeof(*config->table) * config->table_size;
465e8f35aabSAndrzej Hajda 		mix->table = kmemdup(config->table, table_bytes, GFP_KERNEL);
466e9baa279SMarkus Elfring 		if (!mix->table)
467e9baa279SMarkus Elfring 			goto free_mix;
468e9baa279SMarkus Elfring 
469ee81f4eeSChao Xie 		mix->table_size = config->table_size;
470ee81f4eeSChao Xie 	}
471ee81f4eeSChao Xie 
472ee81f4eeSChao Xie 	if (config->mux_table) {
473ee81f4eeSChao Xie 		table_bytes = sizeof(u32) * num_parents;
474e8f35aabSAndrzej Hajda 		mix->mux_table = kmemdup(config->mux_table, table_bytes,
475e8f35aabSAndrzej Hajda 					 GFP_KERNEL);
476ee81f4eeSChao Xie 		if (!mix->mux_table) {
477ee81f4eeSChao Xie 			kfree(mix->table);
478e9baa279SMarkus Elfring 			goto free_mix;
479ee81f4eeSChao Xie 		}
480ee81f4eeSChao Xie 	}
481ee81f4eeSChao Xie 
482ee81f4eeSChao Xie 	mix->div_flags = config->div_flags;
483ee81f4eeSChao Xie 	mix->mux_flags = config->mux_flags;
484ee81f4eeSChao Xie 	mix->lock = lock;
485ee81f4eeSChao Xie 	mix->hw.init = &init;
486ee81f4eeSChao Xie 
487ee81f4eeSChao Xie 	if (config->reg_info.bit_fc >= 32)
488ee81f4eeSChao Xie 		mix->type = MMP_CLK_MIX_TYPE_V1;
489ee81f4eeSChao Xie 	else if (config->reg_info.reg_clk_sel)
490ee81f4eeSChao Xie 		mix->type = MMP_CLK_MIX_TYPE_V3;
491ee81f4eeSChao Xie 	else
492ee81f4eeSChao Xie 		mix->type = MMP_CLK_MIX_TYPE_V2;
493ee81f4eeSChao Xie 	clk = clk_register(dev, &mix->hw);
494ee81f4eeSChao Xie 
495ee81f4eeSChao Xie 	if (IS_ERR(clk)) {
496ee81f4eeSChao Xie 		kfree(mix->mux_table);
497ee81f4eeSChao Xie 		kfree(mix->table);
498ee81f4eeSChao Xie 		kfree(mix);
499ee81f4eeSChao Xie 	}
500ee81f4eeSChao Xie 
501ee81f4eeSChao Xie 	return clk;
502e9baa279SMarkus Elfring 
503e9baa279SMarkus Elfring free_mix:
504e9baa279SMarkus Elfring 	kfree(mix);
505e9baa279SMarkus Elfring 	return ERR_PTR(-ENOMEM);
506ee81f4eeSChao Xie }
507