4 * Copyright (C) 2014 Atmel
6 * Alexandre Belloni <alexandre.belloni@free-electrons.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 as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
15 #include <linux/clk-provider.h>
16 #include <linux/clkdev.h>
17 #include <linux/clk/at91_pmc.h>
19 #include <linux/regmap.h>
20 #include <linux/mfd/syscon.h>
24 #define H32MX_MAX_FREQ 90000000
26 struct clk_sama5d4_h32mx
{
28 struct regmap
*regmap
;
31 #define to_clk_sama5d4_h32mx(hw) container_of(hw, struct clk_sama5d4_h32mx, hw)
33 static unsigned long clk_sama5d4_h32mx_recalc_rate(struct clk_hw
*hw
,
34 unsigned long parent_rate
)
36 struct clk_sama5d4_h32mx
*h32mxclk
= to_clk_sama5d4_h32mx(hw
);
39 regmap_read(h32mxclk
->regmap
, AT91_PMC_MCKR
, &mckr
);
40 if (mckr
& AT91_PMC_H32MXDIV
)
41 return parent_rate
/ 2;
43 if (parent_rate
> H32MX_MAX_FREQ
)
44 pr_warn("H32MX clock is too fast\n");
48 static long clk_sama5d4_h32mx_round_rate(struct clk_hw
*hw
, unsigned long rate
,
49 unsigned long *parent_rate
)
53 if (rate
> *parent_rate
)
55 div
= *parent_rate
/ 2;
59 if (rate
- div
< *parent_rate
- rate
)
65 static int clk_sama5d4_h32mx_set_rate(struct clk_hw
*hw
, unsigned long rate
,
66 unsigned long parent_rate
)
68 struct clk_sama5d4_h32mx
*h32mxclk
= to_clk_sama5d4_h32mx(hw
);
71 if (parent_rate
!= rate
&& (parent_rate
/ 2) != rate
)
74 if ((parent_rate
/ 2) == rate
)
75 mckr
= AT91_PMC_H32MXDIV
;
77 regmap_update_bits(h32mxclk
->regmap
, AT91_PMC_MCKR
,
78 AT91_PMC_H32MXDIV
, mckr
);
83 static const struct clk_ops h32mx_ops
= {
84 .recalc_rate
= clk_sama5d4_h32mx_recalc_rate
,
85 .round_rate
= clk_sama5d4_h32mx_round_rate
,
86 .set_rate
= clk_sama5d4_h32mx_set_rate
,
89 static void __init
of_sama5d4_clk_h32mx_setup(struct device_node
*np
)
91 struct clk_sama5d4_h32mx
*h32mxclk
;
92 struct clk_init_data init
;
93 const char *parent_name
;
94 struct regmap
*regmap
;
97 regmap
= syscon_node_to_regmap(of_get_parent(np
));
101 h32mxclk
= kzalloc(sizeof(*h32mxclk
), GFP_KERNEL
);
105 parent_name
= of_clk_get_parent_name(np
, 0);
107 init
.name
= np
->name
;
108 init
.ops
= &h32mx_ops
;
109 init
.parent_names
= parent_name
? &parent_name
: NULL
;
110 init
.num_parents
= parent_name
? 1 : 0;
111 init
.flags
= CLK_SET_RATE_GATE
;
113 h32mxclk
->hw
.init
= &init
;
114 h32mxclk
->regmap
= regmap
;
116 ret
= clk_hw_register(NULL
, &h32mxclk
->hw
);
122 of_clk_add_hw_provider(np
, of_clk_hw_simple_get
, &h32mxclk
->hw
);
124 CLK_OF_DECLARE(of_sama5d4_clk_h32mx_setup
, "atmel,sama5d4-clk-h32mx",
125 of_sama5d4_clk_h32mx_setup
);