4 * Copyright (C) 2013 Ideas On Board SPRL
6 * Contact: Laurent Pinchart <laurent.pinchart@ideasonboard.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; version 2 of the License.
13 #include <linux/clk-provider.h>
14 #include <linux/clkdev.h>
17 #include <linux/of_address.h>
18 #include <linux/spinlock.h>
21 * MSTP clocks. We can't use standard gate clocks as we need to poll on the
22 * status register when enabling the clock.
25 #define MSTP_MAX_CLOCKS 32
28 * struct mstp_clock_group - MSTP gating clocks group
30 * @data: clocks in this group
31 * @smstpcr: module stop control register
32 * @mstpsr: module stop status register (optional)
33 * @lock: protects writes to SMSTPCR
35 struct mstp_clock_group
{
36 struct clk_onecell_data data
;
37 void __iomem
*smstpcr
;
43 * struct mstp_clock - MSTP gating clock
44 * @hw: handle between common and hardware-specific interfaces
45 * @bit_index: control bit index
46 * @group: MSTP clocks group
51 struct mstp_clock_group
*group
;
54 #define to_mstp_clock(_hw) container_of(_hw, struct mstp_clock, hw)
56 static int cpg_mstp_clock_endisable(struct clk_hw
*hw
, bool enable
)
58 struct mstp_clock
*clock
= to_mstp_clock(hw
);
59 struct mstp_clock_group
*group
= clock
->group
;
60 u32 bitmask
= BIT(clock
->bit_index
);
65 spin_lock_irqsave(&group
->lock
, flags
);
67 value
= clk_readl(group
->smstpcr
);
72 clk_writel(value
, group
->smstpcr
);
74 spin_unlock_irqrestore(&group
->lock
, flags
);
76 if (!enable
|| !group
->mstpsr
)
79 for (i
= 1000; i
> 0; --i
) {
80 if (!(clk_readl(group
->mstpsr
) & bitmask
))
86 pr_err("%s: failed to enable %p[%d]\n", __func__
,
87 group
->smstpcr
, clock
->bit_index
);
94 static int cpg_mstp_clock_enable(struct clk_hw
*hw
)
96 return cpg_mstp_clock_endisable(hw
, true);
99 static void cpg_mstp_clock_disable(struct clk_hw
*hw
)
101 cpg_mstp_clock_endisable(hw
, false);
104 static int cpg_mstp_clock_is_enabled(struct clk_hw
*hw
)
106 struct mstp_clock
*clock
= to_mstp_clock(hw
);
107 struct mstp_clock_group
*group
= clock
->group
;
111 value
= clk_readl(group
->mstpsr
);
113 value
= clk_readl(group
->smstpcr
);
115 return !!(value
& BIT(clock
->bit_index
));
118 static const struct clk_ops cpg_mstp_clock_ops
= {
119 .enable
= cpg_mstp_clock_enable
,
120 .disable
= cpg_mstp_clock_disable
,
121 .is_enabled
= cpg_mstp_clock_is_enabled
,
124 static struct clk
* __init
125 cpg_mstp_clock_register(const char *name
, const char *parent_name
,
126 unsigned int index
, struct mstp_clock_group
*group
)
128 struct clk_init_data init
;
129 struct mstp_clock
*clock
;
132 clock
= kzalloc(sizeof(*clock
), GFP_KERNEL
);
134 pr_err("%s: failed to allocate MSTP clock.\n", __func__
);
135 return ERR_PTR(-ENOMEM
);
139 init
.ops
= &cpg_mstp_clock_ops
;
140 init
.flags
= CLK_IS_BASIC
;
141 init
.parent_names
= &parent_name
;
142 init
.num_parents
= 1;
144 clock
->bit_index
= index
;
145 clock
->group
= group
;
146 clock
->hw
.init
= &init
;
148 clk
= clk_register(NULL
, &clock
->hw
);
156 static void __init
cpg_mstp_clocks_init(struct device_node
*np
)
158 struct mstp_clock_group
*group
;
162 group
= kzalloc(sizeof(*group
), GFP_KERNEL
);
163 clks
= kmalloc(MSTP_MAX_CLOCKS
* sizeof(*clks
), GFP_KERNEL
);
164 if (group
== NULL
|| clks
== NULL
) {
167 pr_err("%s: failed to allocate group\n", __func__
);
171 spin_lock_init(&group
->lock
);
172 group
->data
.clks
= clks
;
174 group
->smstpcr
= of_iomap(np
, 0);
175 group
->mstpsr
= of_iomap(np
, 1);
177 if (group
->smstpcr
== NULL
) {
178 pr_err("%s: failed to remap SMSTPCR\n", __func__
);
184 for (i
= 0; i
< MSTP_MAX_CLOCKS
; ++i
)
185 clks
[i
] = ERR_PTR(-ENOENT
);
187 for (i
= 0; i
< MSTP_MAX_CLOCKS
; ++i
) {
188 const char *parent_name
;
193 /* Skip clocks with no name. */
194 ret
= of_property_read_string_index(np
, "clock-output-names",
196 if (ret
< 0 || strlen(name
) == 0)
199 parent_name
= of_clk_get_parent_name(np
, i
);
200 ret
= of_property_read_u32_index(np
, "renesas,clock-indices", i
,
202 if (parent_name
== NULL
|| ret
< 0)
205 if (clkidx
>= MSTP_MAX_CLOCKS
) {
206 pr_err("%s: invalid clock %s %s index %u)\n",
207 __func__
, np
->name
, name
, clkidx
);
211 clks
[clkidx
] = cpg_mstp_clock_register(name
, parent_name
,
213 if (!IS_ERR(clks
[clkidx
])) {
214 group
->data
.clk_num
= max(group
->data
.clk_num
,
217 * Register a clkdev to let board code retrieve the
218 * clock by name and register aliases for non-DT
221 * FIXME: Remove this when all devices that require a
222 * clock will be instantiated from DT.
224 clk_register_clkdev(clks
[clkidx
], name
, NULL
);
226 pr_err("%s: failed to register %s %s clock (%ld)\n",
227 __func__
, np
->name
, name
, PTR_ERR(clks
[clkidx
]));
231 of_clk_add_provider(np
, of_clk_src_onecell_get
, &group
->data
);
233 CLK_OF_DECLARE(cpg_mstp_clks
, "renesas,cpg-mstp-clocks", cpg_mstp_clocks_init
);