xref: /openbmc/linux/drivers/media/i2c/aptina-pll.c (revision 27b0a9c2)
11802d0beSThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
2cb7a01acSMauro Carvalho Chehab /*
3cb7a01acSMauro Carvalho Chehab  * Aptina Sensor PLL Configuration
4cb7a01acSMauro Carvalho Chehab  *
5cb7a01acSMauro Carvalho Chehab  * Copyright (C) 2012 Laurent Pinchart <laurent.pinchart@ideasonboard.com>
6cb7a01acSMauro Carvalho Chehab  */
7cb7a01acSMauro Carvalho Chehab 
8cb7a01acSMauro Carvalho Chehab #include <linux/device.h>
9cb7a01acSMauro Carvalho Chehab #include <linux/gcd.h>
10cb7a01acSMauro Carvalho Chehab #include <linux/kernel.h>
11cb7a01acSMauro Carvalho Chehab #include <linux/module.h>
12cb7a01acSMauro Carvalho Chehab 
13cb7a01acSMauro Carvalho Chehab #include "aptina-pll.h"
14cb7a01acSMauro Carvalho Chehab 
aptina_pll_calculate(struct device * dev,const struct aptina_pll_limits * limits,struct aptina_pll * pll)15cb7a01acSMauro Carvalho Chehab int aptina_pll_calculate(struct device *dev,
16cb7a01acSMauro Carvalho Chehab 			 const struct aptina_pll_limits *limits,
17cb7a01acSMauro Carvalho Chehab 			 struct aptina_pll *pll)
18cb7a01acSMauro Carvalho Chehab {
19cb7a01acSMauro Carvalho Chehab 	unsigned int mf_min;
20cb7a01acSMauro Carvalho Chehab 	unsigned int mf_max;
21cb7a01acSMauro Carvalho Chehab 	unsigned int p1_min;
22cb7a01acSMauro Carvalho Chehab 	unsigned int p1_max;
23cb7a01acSMauro Carvalho Chehab 	unsigned int p1;
24cb7a01acSMauro Carvalho Chehab 	unsigned int div;
25cb7a01acSMauro Carvalho Chehab 
26cb7a01acSMauro Carvalho Chehab 	dev_dbg(dev, "PLL: ext clock %u pix clock %u\n",
27cb7a01acSMauro Carvalho Chehab 		pll->ext_clock, pll->pix_clock);
28cb7a01acSMauro Carvalho Chehab 
29cb7a01acSMauro Carvalho Chehab 	if (pll->ext_clock < limits->ext_clock_min ||
30cb7a01acSMauro Carvalho Chehab 	    pll->ext_clock > limits->ext_clock_max) {
31cb7a01acSMauro Carvalho Chehab 		dev_err(dev, "pll: invalid external clock frequency.\n");
32cb7a01acSMauro Carvalho Chehab 		return -EINVAL;
33cb7a01acSMauro Carvalho Chehab 	}
34cb7a01acSMauro Carvalho Chehab 
35cb7a01acSMauro Carvalho Chehab 	if (pll->pix_clock == 0 || pll->pix_clock > limits->pix_clock_max) {
36cb7a01acSMauro Carvalho Chehab 		dev_err(dev, "pll: invalid pixel clock frequency.\n");
37cb7a01acSMauro Carvalho Chehab 		return -EINVAL;
38cb7a01acSMauro Carvalho Chehab 	}
39cb7a01acSMauro Carvalho Chehab 
40cb7a01acSMauro Carvalho Chehab 	/* Compute the multiplier M and combined N*P1 divisor. */
41cb7a01acSMauro Carvalho Chehab 	div = gcd(pll->pix_clock, pll->ext_clock);
42cb7a01acSMauro Carvalho Chehab 	pll->m = pll->pix_clock / div;
43cb7a01acSMauro Carvalho Chehab 	div = pll->ext_clock / div;
44cb7a01acSMauro Carvalho Chehab 
45cb7a01acSMauro Carvalho Chehab 	/* We now have the smallest M and N*P1 values that will result in the
46cb7a01acSMauro Carvalho Chehab 	 * desired pixel clock frequency, but they might be out of the valid
47cb7a01acSMauro Carvalho Chehab 	 * range. Compute the factor by which we should multiply them given the
48cb7a01acSMauro Carvalho Chehab 	 * following constraints:
49cb7a01acSMauro Carvalho Chehab 	 *
50cb7a01acSMauro Carvalho Chehab 	 * - minimum/maximum multiplier
51cb7a01acSMauro Carvalho Chehab 	 * - minimum/maximum multiplier output clock frequency assuming the
52cb7a01acSMauro Carvalho Chehab 	 *   minimum/maximum N value
53cb7a01acSMauro Carvalho Chehab 	 * - minimum/maximum combined N*P1 divisor
54cb7a01acSMauro Carvalho Chehab 	 */
55cb7a01acSMauro Carvalho Chehab 	mf_min = DIV_ROUND_UP(limits->m_min, pll->m);
56cb7a01acSMauro Carvalho Chehab 	mf_min = max(mf_min, limits->out_clock_min /
57cb7a01acSMauro Carvalho Chehab 		     (pll->ext_clock / limits->n_min * pll->m));
58cb7a01acSMauro Carvalho Chehab 	mf_min = max(mf_min, limits->n_min * limits->p1_min / div);
59cb7a01acSMauro Carvalho Chehab 	mf_max = limits->m_max / pll->m;
60cb7a01acSMauro Carvalho Chehab 	mf_max = min(mf_max, limits->out_clock_max /
61cb7a01acSMauro Carvalho Chehab 		    (pll->ext_clock / limits->n_max * pll->m));
62cb7a01acSMauro Carvalho Chehab 	mf_max = min(mf_max, DIV_ROUND_UP(limits->n_max * limits->p1_max, div));
63cb7a01acSMauro Carvalho Chehab 
64cb7a01acSMauro Carvalho Chehab 	dev_dbg(dev, "pll: mf min %u max %u\n", mf_min, mf_max);
65cb7a01acSMauro Carvalho Chehab 	if (mf_min > mf_max) {
66cb7a01acSMauro Carvalho Chehab 		dev_err(dev, "pll: no valid combined N*P1 divisor.\n");
67cb7a01acSMauro Carvalho Chehab 		return -EINVAL;
68cb7a01acSMauro Carvalho Chehab 	}
69cb7a01acSMauro Carvalho Chehab 
70cb7a01acSMauro Carvalho Chehab 	/*
71cb7a01acSMauro Carvalho Chehab 	 * We're looking for the highest acceptable P1 value for which a
72cb7a01acSMauro Carvalho Chehab 	 * multiplier factor MF exists that fulfills the following conditions:
73cb7a01acSMauro Carvalho Chehab 	 *
74cb7a01acSMauro Carvalho Chehab 	 * 1. p1 is in the [p1_min, p1_max] range given by the limits and is
75cb7a01acSMauro Carvalho Chehab 	 *    even
76cb7a01acSMauro Carvalho Chehab 	 * 2. mf is in the [mf_min, mf_max] range computed above
77cb7a01acSMauro Carvalho Chehab 	 * 3. div * mf is a multiple of p1, in order to compute
78cb7a01acSMauro Carvalho Chehab 	 *	n = div * mf / p1
79cb7a01acSMauro Carvalho Chehab 	 *	m = pll->m * mf
80cb7a01acSMauro Carvalho Chehab 	 * 4. the internal clock frequency, given by ext_clock / n, is in the
81cb7a01acSMauro Carvalho Chehab 	 *    [int_clock_min, int_clock_max] range given by the limits
82cb7a01acSMauro Carvalho Chehab 	 * 5. the output clock frequency, given by ext_clock / n * m, is in the
83cb7a01acSMauro Carvalho Chehab 	 *    [out_clock_min, out_clock_max] range given by the limits
84cb7a01acSMauro Carvalho Chehab 	 *
85cb7a01acSMauro Carvalho Chehab 	 * The first naive approach is to iterate over all p1 values acceptable
86cb7a01acSMauro Carvalho Chehab 	 * according to (1) and all mf values acceptable according to (2), and
87cb7a01acSMauro Carvalho Chehab 	 * stop at the first combination that fulfills (3), (4) and (5). This
88cb7a01acSMauro Carvalho Chehab 	 * has a O(n^2) complexity.
89cb7a01acSMauro Carvalho Chehab 	 *
90cb7a01acSMauro Carvalho Chehab 	 * Instead of iterating over all mf values in the [mf_min, mf_max] range
91cb7a01acSMauro Carvalho Chehab 	 * we can compute the mf increment between two acceptable values
92cb7a01acSMauro Carvalho Chehab 	 * according to (3) with
93cb7a01acSMauro Carvalho Chehab 	 *
94cb7a01acSMauro Carvalho Chehab 	 *	mf_inc = p1 / gcd(div, p1)			(6)
95cb7a01acSMauro Carvalho Chehab 	 *
96cb7a01acSMauro Carvalho Chehab 	 * and round the minimum up to the nearest multiple of mf_inc. This will
97cb7a01acSMauro Carvalho Chehab 	 * restrict the number of mf values to be checked.
98cb7a01acSMauro Carvalho Chehab 	 *
99cb7a01acSMauro Carvalho Chehab 	 * Furthermore, conditions (4) and (5) only restrict the range of
100cb7a01acSMauro Carvalho Chehab 	 * acceptable p1 and mf values by modifying the minimum and maximum
101cb7a01acSMauro Carvalho Chehab 	 * limits. (5) can be expressed as
102cb7a01acSMauro Carvalho Chehab 	 *
103cb7a01acSMauro Carvalho Chehab 	 *	ext_clock / (div * mf / p1) * m * mf >= out_clock_min
104cb7a01acSMauro Carvalho Chehab 	 *	ext_clock / (div * mf / p1) * m * mf <= out_clock_max
105cb7a01acSMauro Carvalho Chehab 	 *
106cb7a01acSMauro Carvalho Chehab 	 * or
107cb7a01acSMauro Carvalho Chehab 	 *
108cb7a01acSMauro Carvalho Chehab 	 *	p1 >= out_clock_min * div / (ext_clock * m)	(7)
109cb7a01acSMauro Carvalho Chehab 	 *	p1 <= out_clock_max * div / (ext_clock * m)
110cb7a01acSMauro Carvalho Chehab 	 *
111cb7a01acSMauro Carvalho Chehab 	 * Similarly, (4) can be expressed as
112cb7a01acSMauro Carvalho Chehab 	 *
113cb7a01acSMauro Carvalho Chehab 	 *	mf >= ext_clock * p1 / (int_clock_max * div)	(8)
114cb7a01acSMauro Carvalho Chehab 	 *	mf <= ext_clock * p1 / (int_clock_min * div)
115cb7a01acSMauro Carvalho Chehab 	 *
116cb7a01acSMauro Carvalho Chehab 	 * We can thus iterate over the restricted p1 range defined by the
117cb7a01acSMauro Carvalho Chehab 	 * combination of (1) and (7), and then compute the restricted mf range
118cb7a01acSMauro Carvalho Chehab 	 * defined by the combination of (2), (6) and (8). If the resulting mf
119cb7a01acSMauro Carvalho Chehab 	 * range is not empty, any value in the mf range is acceptable. We thus
120cb7a01acSMauro Carvalho Chehab 	 * select the mf lwoer bound and the corresponding p1 value.
121cb7a01acSMauro Carvalho Chehab 	 */
122cb7a01acSMauro Carvalho Chehab 	if (limits->p1_min == 0) {
123cb7a01acSMauro Carvalho Chehab 		dev_err(dev, "pll: P1 minimum value must be >0.\n");
124cb7a01acSMauro Carvalho Chehab 		return -EINVAL;
125cb7a01acSMauro Carvalho Chehab 	}
126cb7a01acSMauro Carvalho Chehab 
127cb7a01acSMauro Carvalho Chehab 	p1_min = max(limits->p1_min, DIV_ROUND_UP(limits->out_clock_min * div,
128cb7a01acSMauro Carvalho Chehab 		     pll->ext_clock * pll->m));
129cb7a01acSMauro Carvalho Chehab 	p1_max = min(limits->p1_max, limits->out_clock_max * div /
130cb7a01acSMauro Carvalho Chehab 		     (pll->ext_clock * pll->m));
131cb7a01acSMauro Carvalho Chehab 
132cb7a01acSMauro Carvalho Chehab 	for (p1 = p1_max & ~1; p1 >= p1_min; p1 -= 2) {
133cb7a01acSMauro Carvalho Chehab 		unsigned int mf_inc = p1 / gcd(div, p1);
134cb7a01acSMauro Carvalho Chehab 		unsigned int mf_high;
135cb7a01acSMauro Carvalho Chehab 		unsigned int mf_low;
136cb7a01acSMauro Carvalho Chehab 
137cb7a01acSMauro Carvalho Chehab 		mf_low = roundup(max(mf_min, DIV_ROUND_UP(pll->ext_clock * p1,
138cb7a01acSMauro Carvalho Chehab 					limits->int_clock_max * div)), mf_inc);
139cb7a01acSMauro Carvalho Chehab 		mf_high = min(mf_max, pll->ext_clock * p1 /
140cb7a01acSMauro Carvalho Chehab 			      (limits->int_clock_min * div));
141cb7a01acSMauro Carvalho Chehab 
142cb7a01acSMauro Carvalho Chehab 		if (mf_low > mf_high)
143cb7a01acSMauro Carvalho Chehab 			continue;
144cb7a01acSMauro Carvalho Chehab 
145cb7a01acSMauro Carvalho Chehab 		pll->n = div * mf_low / p1;
146cb7a01acSMauro Carvalho Chehab 		pll->m *= mf_low;
147cb7a01acSMauro Carvalho Chehab 		pll->p1 = p1;
148cb7a01acSMauro Carvalho Chehab 		dev_dbg(dev, "PLL: N %u M %u P1 %u\n", pll->n, pll->m, pll->p1);
149cb7a01acSMauro Carvalho Chehab 		return 0;
150cb7a01acSMauro Carvalho Chehab 	}
151cb7a01acSMauro Carvalho Chehab 
152cb7a01acSMauro Carvalho Chehab 	dev_err(dev, "pll: no valid N and P1 divisors found.\n");
153cb7a01acSMauro Carvalho Chehab 	return -EINVAL;
154cb7a01acSMauro Carvalho Chehab }
155cb7a01acSMauro Carvalho Chehab EXPORT_SYMBOL_GPL(aptina_pll_calculate);
156cb7a01acSMauro Carvalho Chehab 
157cb7a01acSMauro Carvalho Chehab MODULE_DESCRIPTION("Aptina PLL Helpers");
158cb7a01acSMauro Carvalho Chehab MODULE_AUTHOR("Laurent Pinchart <laurent.pinchart@ideasonboard.com>");
159cb7a01acSMauro Carvalho Chehab MODULE_LICENSE("GPL v2");
160