xref: /openbmc/linux/drivers/clk/clk-fractional-divider.c (revision c9933d494c54f72290831191c09bb8488bfd5905)
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Copyright (C) 2014 Intel Corporation
4  *
5  * Adjustable fractional divider clock implementation.
6  * Uses rational best approximation algorithm.
7  *
8  * Output is calculated as
9  *
10  *	rate = (m / n) * parent_rate				(1)
11  *
12  * This is useful when we have a prescaler block which asks for
13  * m (numerator) and n (denominator) values to be provided to satisfy
14  * the (1) as much as possible.
15  *
16  * Since m and n have the limitation by a range, e.g.
17  *
18  *	n >= 1, n < N_width, where N_width = 2^nwidth		(2)
19  *
20  * for some cases the output may be saturated. Hence, from (1) and (2),
21  * assuming the worst case when m = 1, the inequality
22  *
23  *	floor(log2(parent_rate / rate)) <= nwidth		(3)
24  *
25  * may be derived. Thus, in cases when
26  *
27  *	(parent_rate / rate) >> N_width				(4)
28  *
29  * we might scale up the rate by 2^scale (see the description of
30  * CLK_FRAC_DIVIDER_POWER_OF_TWO_PS for additional information), where
31  *
32  *	scale = floor(log2(parent_rate / rate)) - nwidth	(5)
33  *
34  * and assume that the IP, that needs m and n, has also its own
35  * prescaler, which is capable to divide by 2^scale. In this way
36  * we get the denominator to satisfy the desired range (2) and
37  * at the same time a much better result of m and n than simple
38  * saturated values.
39  */
40 
41 #include <linux/clk-provider.h>
42 #include <linux/io.h>
43 #include <linux/module.h>
44 #include <linux/device.h>
45 #include <linux/slab.h>
46 #include <linux/rational.h>
47 
48 #include "clk-fractional-divider.h"
49 
50 static inline u32 clk_fd_readl(struct clk_fractional_divider *fd)
51 {
52 	if (fd->flags & CLK_FRAC_DIVIDER_BIG_ENDIAN)
53 		return ioread32be(fd->reg);
54 
55 	return readl(fd->reg);
56 }
57 
58 static inline void clk_fd_writel(struct clk_fractional_divider *fd, u32 val)
59 {
60 	if (fd->flags & CLK_FRAC_DIVIDER_BIG_ENDIAN)
61 		iowrite32be(val, fd->reg);
62 	else
63 		writel(val, fd->reg);
64 }
65 
66 static unsigned long clk_fd_recalc_rate(struct clk_hw *hw,
67 					unsigned long parent_rate)
68 {
69 	struct clk_fractional_divider *fd = to_clk_fd(hw);
70 	unsigned long flags = 0;
71 	unsigned long m, n;
72 	u32 val;
73 	u64 ret;
74 
75 	if (fd->lock)
76 		spin_lock_irqsave(fd->lock, flags);
77 	else
78 		__acquire(fd->lock);
79 
80 	val = clk_fd_readl(fd);
81 
82 	if (fd->lock)
83 		spin_unlock_irqrestore(fd->lock, flags);
84 	else
85 		__release(fd->lock);
86 
87 	m = (val & fd->mmask) >> fd->mshift;
88 	n = (val & fd->nmask) >> fd->nshift;
89 
90 	if (fd->flags & CLK_FRAC_DIVIDER_ZERO_BASED) {
91 		m++;
92 		n++;
93 	}
94 
95 	if (!n || !m)
96 		return parent_rate;
97 
98 	ret = (u64)parent_rate * m;
99 	do_div(ret, n);
100 
101 	return ret;
102 }
103 
104 void clk_fractional_divider_general_approximation(struct clk_hw *hw,
105 						  unsigned long rate,
106 						  unsigned long *parent_rate,
107 						  unsigned long *m, unsigned long *n)
108 {
109 	struct clk_fractional_divider *fd = to_clk_fd(hw);
110 
111 	/*
112 	 * Get rate closer to *parent_rate to guarantee there is no overflow
113 	 * for m and n. In the result it will be the nearest rate left shifted
114 	 * by (scale - fd->nwidth) bits.
115 	 *
116 	 * For the detailed explanation see the top comment in this file.
117 	 */
118 	if (fd->flags & CLK_FRAC_DIVIDER_POWER_OF_TWO_PS) {
119 		unsigned long scale = fls_long(*parent_rate / rate - 1);
120 
121 		if (scale > fd->nwidth)
122 			rate <<= scale - fd->nwidth;
123 	}
124 
125 	rational_best_approximation(rate, *parent_rate,
126 			GENMASK(fd->mwidth - 1, 0), GENMASK(fd->nwidth - 1, 0),
127 			m, n);
128 }
129 
130 static long clk_fd_round_rate(struct clk_hw *hw, unsigned long rate,
131 			      unsigned long *parent_rate)
132 {
133 	struct clk_fractional_divider *fd = to_clk_fd(hw);
134 	unsigned long m, n;
135 	u64 ret;
136 
137 	if (!rate || (!clk_hw_can_set_rate_parent(hw) && rate >= *parent_rate))
138 		return *parent_rate;
139 
140 	if (fd->approximation)
141 		fd->approximation(hw, rate, parent_rate, &m, &n);
142 	else
143 		clk_fractional_divider_general_approximation(hw, rate, parent_rate, &m, &n);
144 
145 	ret = (u64)*parent_rate * m;
146 	do_div(ret, n);
147 
148 	return ret;
149 }
150 
151 static int clk_fd_set_rate(struct clk_hw *hw, unsigned long rate,
152 			   unsigned long parent_rate)
153 {
154 	struct clk_fractional_divider *fd = to_clk_fd(hw);
155 	unsigned long flags = 0;
156 	unsigned long m, n;
157 	u32 val;
158 
159 	rational_best_approximation(rate, parent_rate,
160 			GENMASK(fd->mwidth - 1, 0), GENMASK(fd->nwidth - 1, 0),
161 			&m, &n);
162 
163 	if (fd->flags & CLK_FRAC_DIVIDER_ZERO_BASED) {
164 		m--;
165 		n--;
166 	}
167 
168 	if (fd->lock)
169 		spin_lock_irqsave(fd->lock, flags);
170 	else
171 		__acquire(fd->lock);
172 
173 	val = clk_fd_readl(fd);
174 	val &= ~(fd->mmask | fd->nmask);
175 	val |= (m << fd->mshift) | (n << fd->nshift);
176 	clk_fd_writel(fd, val);
177 
178 	if (fd->lock)
179 		spin_unlock_irqrestore(fd->lock, flags);
180 	else
181 		__release(fd->lock);
182 
183 	return 0;
184 }
185 
186 const struct clk_ops clk_fractional_divider_ops = {
187 	.recalc_rate = clk_fd_recalc_rate,
188 	.round_rate = clk_fd_round_rate,
189 	.set_rate = clk_fd_set_rate,
190 };
191 EXPORT_SYMBOL_GPL(clk_fractional_divider_ops);
192 
193 struct clk_hw *clk_hw_register_fractional_divider(struct device *dev,
194 		const char *name, const char *parent_name, unsigned long flags,
195 		void __iomem *reg, u8 mshift, u8 mwidth, u8 nshift, u8 nwidth,
196 		u8 clk_divider_flags, spinlock_t *lock)
197 {
198 	struct clk_fractional_divider *fd;
199 	struct clk_init_data init;
200 	struct clk_hw *hw;
201 	int ret;
202 
203 	fd = kzalloc(sizeof(*fd), GFP_KERNEL);
204 	if (!fd)
205 		return ERR_PTR(-ENOMEM);
206 
207 	init.name = name;
208 	init.ops = &clk_fractional_divider_ops;
209 	init.flags = flags;
210 	init.parent_names = parent_name ? &parent_name : NULL;
211 	init.num_parents = parent_name ? 1 : 0;
212 
213 	fd->reg = reg;
214 	fd->mshift = mshift;
215 	fd->mwidth = mwidth;
216 	fd->mmask = GENMASK(mwidth - 1, 0) << mshift;
217 	fd->nshift = nshift;
218 	fd->nwidth = nwidth;
219 	fd->nmask = GENMASK(nwidth - 1, 0) << nshift;
220 	fd->flags = clk_divider_flags;
221 	fd->lock = lock;
222 	fd->hw.init = &init;
223 
224 	hw = &fd->hw;
225 	ret = clk_hw_register(dev, hw);
226 	if (ret) {
227 		kfree(fd);
228 		hw = ERR_PTR(ret);
229 	}
230 
231 	return hw;
232 }
233 EXPORT_SYMBOL_GPL(clk_hw_register_fractional_divider);
234 
235 struct clk *clk_register_fractional_divider(struct device *dev,
236 		const char *name, const char *parent_name, unsigned long flags,
237 		void __iomem *reg, u8 mshift, u8 mwidth, u8 nshift, u8 nwidth,
238 		u8 clk_divider_flags, spinlock_t *lock)
239 {
240 	struct clk_hw *hw;
241 
242 	hw = clk_hw_register_fractional_divider(dev, name, parent_name, flags,
243 			reg, mshift, mwidth, nshift, nwidth, clk_divider_flags,
244 			lock);
245 	if (IS_ERR(hw))
246 		return ERR_CAST(hw);
247 	return hw->clk;
248 }
249 EXPORT_SYMBOL_GPL(clk_register_fractional_divider);
250 
251 void clk_hw_unregister_fractional_divider(struct clk_hw *hw)
252 {
253 	struct clk_fractional_divider *fd;
254 
255 	fd = to_clk_fd(hw);
256 
257 	clk_hw_unregister(hw);
258 	kfree(fd);
259 }
260