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