diff options
author | Chen-Yu Tsai <wens@csie.org> | 2014-09-16 06:04:01 -0400 |
---|---|---|
committer | Maxime Ripard <maxime.ripard@free-electrons.com> | 2014-09-27 03:05:43 -0400 |
commit | 9c8176bfb67f98ed9a521b624dcb6ab7fa254aa7 (patch) | |
tree | 52bf5988af67aa1c7423010c09dbccfe9ab40671 /drivers/clk/sunxi/clk-sun8i-mbus.c | |
parent | 37e1041f04717d726931c8688cbf425071aeb9c1 (diff) |
clk: sunxi: Add sun8i MBUS clock support
The MBUS clock on sun8i is slightly different from the old mod0 clocks.
The divider is 3 bits wider, while also needing a divider table for the
higher 4 values, which all set the same divider.
Signed-off-by: Chen-Yu Tsai <wens@csie.org>
Signed-off-by: Maxime Ripard <maxime.ripard@free-electrons.com>
Diffstat (limited to 'drivers/clk/sunxi/clk-sun8i-mbus.c')
-rw-r--r-- | drivers/clk/sunxi/clk-sun8i-mbus.c | 78 |
1 files changed, 78 insertions, 0 deletions
diff --git a/drivers/clk/sunxi/clk-sun8i-mbus.c b/drivers/clk/sunxi/clk-sun8i-mbus.c new file mode 100644 index 000000000000..8e49b44cee41 --- /dev/null +++ b/drivers/clk/sunxi/clk-sun8i-mbus.c | |||
@@ -0,0 +1,78 @@ | |||
1 | /* | ||
2 | * Copyright 2014 Chen-Yu Tsai | ||
3 | * | ||
4 | * Chen-Yu Tsai <wens@csie.org> | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation; either version 2 of the License, or | ||
9 | * (at your option) any later version. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | */ | ||
16 | |||
17 | #include <linux/clk-provider.h> | ||
18 | #include <linux/clkdev.h> | ||
19 | #include <linux/of_address.h> | ||
20 | |||
21 | #include "clk-factors.h" | ||
22 | |||
23 | /** | ||
24 | * sun8i_a23_get_mbus_factors() - calculates m factor for MBUS clocks | ||
25 | * MBUS rate is calculated as follows | ||
26 | * rate = parent_rate / (m + 1); | ||
27 | */ | ||
28 | |||
29 | static void sun8i_a23_get_mbus_factors(u32 *freq, u32 parent_rate, | ||
30 | u8 *n, u8 *k, u8 *m, u8 *p) | ||
31 | { | ||
32 | u8 div; | ||
33 | |||
34 | /* | ||
35 | * These clocks can only divide, so we will never be able to | ||
36 | * achieve frequencies higher than the parent frequency | ||
37 | */ | ||
38 | if (*freq > parent_rate) | ||
39 | *freq = parent_rate; | ||
40 | |||
41 | div = DIV_ROUND_UP(parent_rate, *freq); | ||
42 | |||
43 | if (div > 8) | ||
44 | div = 8; | ||
45 | |||
46 | *freq = parent_rate / div; | ||
47 | |||
48 | /* we were called to round the frequency, we can now return */ | ||
49 | if (m == NULL) | ||
50 | return; | ||
51 | |||
52 | *m = div - 1; | ||
53 | } | ||
54 | |||
55 | static struct clk_factors_config sun8i_a23_mbus_config = { | ||
56 | .mshift = 0, | ||
57 | .mwidth = 3, | ||
58 | }; | ||
59 | |||
60 | static const struct factors_data sun8i_a23_mbus_data __initconst = { | ||
61 | .enable = 31, | ||
62 | .mux = 24, | ||
63 | .table = &sun8i_a23_mbus_config, | ||
64 | .getter = sun8i_a23_get_mbus_factors, | ||
65 | }; | ||
66 | |||
67 | static DEFINE_SPINLOCK(sun8i_a23_mbus_lock); | ||
68 | |||
69 | static void __init sun8i_a23_mbus_setup(struct device_node *node) | ||
70 | { | ||
71 | struct clk *mbus = sunxi_factors_register(node, &sun8i_a23_mbus_data, | ||
72 | &sun8i_a23_mbus_lock); | ||
73 | |||
74 | /* The MBUS clocks needs to be always enabled */ | ||
75 | __clk_get(mbus); | ||
76 | clk_prepare_enable(mbus); | ||
77 | } | ||
78 | CLK_OF_DECLARE(sun8i_a23_mbus, "allwinner,sun8i-a23-mbus-clk", sun8i_a23_mbus_setup); | ||