2 * rl6231.c - RL6231 class device shared support
4 * Copyright 2014 Realtek Semiconductor Corp.
6 * Author: Oder Chiou <oder_chiou@realtek.com>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
13 #include <linux/module.h>
14 #include <linux/moduleparam.h>
15 #include <linux/init.h>
16 #include <linux/delay.h>
18 #include <linux/gpio.h>
19 #include <linux/i2c.h>
20 #include <linux/regmap.h>
22 #include <linux/of_gpio.h>
23 #include <linux/platform_device.h>
24 #include <linux/spi/spi.h>
25 #include <linux/acpi.h>
26 #include <sound/core.h>
27 #include <sound/pcm.h>
28 #include <sound/pcm_params.h>
29 #include <sound/soc.h>
30 #include <sound/soc-dapm.h>
31 #include <sound/initval.h>
32 #include <sound/tlv.h>
37 * rl6231_calc_dmic_clk - Calculate the parameter of dmic.
39 * @rate: base clock rate.
41 * Choose dmic clock between 1MHz and 3MHz.
42 * It is better for clock to approximate 3MHz.
44 int rl6231_calc_dmic_clk(int rate
)
46 int div
[] = {2, 3, 4, 6, 8, 12}, idx
= -EINVAL
;
47 int i
, red
, bound
, temp
;
50 for (i
= 0; i
< ARRAY_SIZE(div
); i
++) {
51 bound
= div
[i
] * 3000000;
63 EXPORT_SYMBOL_GPL(rl6231_calc_dmic_clk
);
66 * rl6231_pll_calc - Calcualte PLL M/N/K code.
67 * @freq_in: external clock provided to codec.
68 * @freq_out: target clock which codec works on.
69 * @pll_code: Pointer to structure with M, N, K and bypass flag.
71 * Calcualte M/N/K code to configure PLL for codec.
73 * Returns 0 for success or negative error code.
75 int rl6231_pll_calc(const unsigned int freq_in
,
76 const unsigned int freq_out
, struct rl6231_pll_code
*pll_code
)
78 int max_n
= RL6231_PLL_N_MAX
, max_m
= RL6231_PLL_M_MAX
;
79 int k
, red
, n_t
, pll_out
, in_t
, out_t
;
80 int n
= 0, m
= 0, m_t
= 0;
81 int red_t
= abs(freq_out
- freq_in
);
84 if (RL6231_PLL_INP_MAX
< freq_in
|| RL6231_PLL_INP_MIN
> freq_in
)
87 k
= 100000000 / freq_out
- 2;
88 if (k
> RL6231_PLL_K_MAX
)
90 for (n_t
= 0; n_t
<= max_n
; n_t
++) {
91 in_t
= freq_in
/ (k
+ 2);
92 pll_out
= freq_out
/ (n_t
+ 2);
95 if (in_t
== pll_out
) {
100 red
= abs(in_t
- pll_out
);
109 for (m_t
= 0; m_t
<= max_m
; m_t
++) {
110 out_t
= in_t
/ (m_t
+ 2);
111 red
= abs(out_t
- pll_out
);
122 pr_debug("Only get approximation about PLL\n");
126 pll_code
->m_bp
= bypass
;
127 pll_code
->m_code
= m
;
128 pll_code
->n_code
= n
;
129 pll_code
->k_code
= k
;
132 EXPORT_SYMBOL_GPL(rl6231_pll_calc
);
134 int rl6231_get_clk_info(int sclk
, int rate
)
136 int i
, pd
[] = {1, 2, 3, 4, 6, 8, 12, 16};
138 if (sclk
<= 0 || rate
<= 0)
142 for (i
= 0; i
< ARRAY_SIZE(pd
); i
++)
143 if (sclk
== rate
* pd
[i
])
148 EXPORT_SYMBOL_GPL(rl6231_get_clk_info
);
150 MODULE_DESCRIPTION("RL6231 class device shared support");
151 MODULE_AUTHOR("Oder Chiou <oder_chiou@realtek.com>");
152 MODULE_LICENSE("GPL v2");