1// SPDX-License-Identifier: GPL-2.0-only
2//
3// Copyright (c) 2021 Samuel Holland <samuel@sholland.org>
4//
5
6#include <linux/clk.h>
7#include <linux/clk-provider.h>
8#include <linux/device.h>
9#include <linux/io.h>
10#include <linux/module.h>
11#include <linux/of.h>
12#include <linux/of_device.h>
13
14#include <linux/clk/sunxi-ng.h>
15
16#include "ccu_common.h"
17
18#include "ccu_div.h"
19#include "ccu_gate.h"
20#include "ccu_mux.h"
21
22#include "ccu-sun6i-rtc.h"
23
24#define IOSC_ACCURACY			300000000 /* 30% */
25#define IOSC_RATE			16000000
26
27#define LOSC_RATE			32768
28#define LOSC_RATE_SHIFT			15
29
30#define LOSC_CTRL_REG			0x0
31#define LOSC_CTRL_KEY			0x16aa0000
32
33#define IOSC_32K_CLK_DIV_REG		0x8
34#define IOSC_32K_CLK_DIV		GENMASK(4, 0)
35#define IOSC_32K_PRE_DIV		32
36
37#define IOSC_CLK_CALI_REG		0xc
38#define IOSC_CLK_CALI_DIV_ONES		22
39#define IOSC_CLK_CALI_EN		BIT(1)
40#define IOSC_CLK_CALI_SRC_SEL		BIT(0)
41
42#define LOSC_OUT_GATING_REG		0x60
43
44#define DCXO_CTRL_REG			0x160
45#define DCXO_CTRL_CLK16M_RC_EN		BIT(0)
46
47struct sun6i_rtc_match_data {
48	bool				have_ext_osc32k		: 1;
49	bool				have_iosc_calibration	: 1;
50	bool				rtc_32k_single_parent	: 1;
51	const struct clk_parent_data	*osc32k_fanout_parents;
52	u8				osc32k_fanout_nparents;
53};
54
55static bool have_iosc_calibration;
56
57static int ccu_iosc_enable(struct clk_hw *hw)
58{
59	struct ccu_common *cm = hw_to_ccu_common(hw);
60
61	return ccu_gate_helper_enable(cm, DCXO_CTRL_CLK16M_RC_EN);
62}
63
64static void ccu_iosc_disable(struct clk_hw *hw)
65{
66	struct ccu_common *cm = hw_to_ccu_common(hw);
67
68	return ccu_gate_helper_disable(cm, DCXO_CTRL_CLK16M_RC_EN);
69}
70
71static int ccu_iosc_is_enabled(struct clk_hw *hw)
72{
73	struct ccu_common *cm = hw_to_ccu_common(hw);
74
75	return ccu_gate_helper_is_enabled(cm, DCXO_CTRL_CLK16M_RC_EN);
76}
77
78static unsigned long ccu_iosc_recalc_rate(struct clk_hw *hw,
79					  unsigned long parent_rate)
80{
81	struct ccu_common *cm = hw_to_ccu_common(hw);
82
83	if (have_iosc_calibration) {
84		u32 reg = readl(cm->base + IOSC_CLK_CALI_REG);
85
86		/*
87		 * Recover the IOSC frequency by shifting the ones place of
88		 * (fixed-point divider * 32768) into bit zero.
89		 */
90		if (reg & IOSC_CLK_CALI_EN)
91			return reg >> (IOSC_CLK_CALI_DIV_ONES - LOSC_RATE_SHIFT);
92	}
93
94	return IOSC_RATE;
95}
96
97static unsigned long ccu_iosc_recalc_accuracy(struct clk_hw *hw,
98					      unsigned long parent_accuracy)
99{
100	return IOSC_ACCURACY;
101}
102
103static const struct clk_ops ccu_iosc_ops = {
104	.enable			= ccu_iosc_enable,
105	.disable		= ccu_iosc_disable,
106	.is_enabled		= ccu_iosc_is_enabled,
107	.recalc_rate		= ccu_iosc_recalc_rate,
108	.recalc_accuracy	= ccu_iosc_recalc_accuracy,
109};
110
111static struct ccu_common iosc_clk = {
112	.reg		= DCXO_CTRL_REG,
113	.hw.init	= CLK_HW_INIT_NO_PARENT("iosc", &ccu_iosc_ops,
114						CLK_GET_RATE_NOCACHE),
115};
116
117static int ccu_iosc_32k_prepare(struct clk_hw *hw)
118{
119	struct ccu_common *cm = hw_to_ccu_common(hw);
120	u32 val;
121
122	if (!have_iosc_calibration)
123		return 0;
124
125	val = readl(cm->base + IOSC_CLK_CALI_REG);
126	writel(val | IOSC_CLK_CALI_EN | IOSC_CLK_CALI_SRC_SEL,
127	       cm->base + IOSC_CLK_CALI_REG);
128
129	return 0;
130}
131
132static void ccu_iosc_32k_unprepare(struct clk_hw *hw)
133{
134	struct ccu_common *cm = hw_to_ccu_common(hw);
135	u32 val;
136
137	if (!have_iosc_calibration)
138		return;
139
140	val = readl(cm->base + IOSC_CLK_CALI_REG);
141	writel(val & ~(IOSC_CLK_CALI_EN | IOSC_CLK_CALI_SRC_SEL),
142	       cm->base + IOSC_CLK_CALI_REG);
143}
144
145static unsigned long ccu_iosc_32k_recalc_rate(struct clk_hw *hw,
146					      unsigned long parent_rate)
147{
148	struct ccu_common *cm = hw_to_ccu_common(hw);
149	u32 val;
150
151	if (have_iosc_calibration) {
152		val = readl(cm->base + IOSC_CLK_CALI_REG);
153
154		/* Assume the calibrated 32k clock is accurate. */
155		if (val & IOSC_CLK_CALI_SRC_SEL)
156			return LOSC_RATE;
157	}
158
159	val = readl(cm->base + IOSC_32K_CLK_DIV_REG) & IOSC_32K_CLK_DIV;
160
161	return parent_rate / IOSC_32K_PRE_DIV / (val + 1);
162}
163
164static unsigned long ccu_iosc_32k_recalc_accuracy(struct clk_hw *hw,
165						  unsigned long parent_accuracy)
166{
167	struct ccu_common *cm = hw_to_ccu_common(hw);
168	u32 val;
169
170	if (have_iosc_calibration) {
171		val = readl(cm->base + IOSC_CLK_CALI_REG);
172
173		/* Assume the calibrated 32k clock is accurate. */
174		if (val & IOSC_CLK_CALI_SRC_SEL)
175			return 0;
176	}
177
178	return parent_accuracy;
179}
180
181static const struct clk_ops ccu_iosc_32k_ops = {
182	.prepare		= ccu_iosc_32k_prepare,
183	.unprepare		= ccu_iosc_32k_unprepare,
184	.recalc_rate		= ccu_iosc_32k_recalc_rate,
185	.recalc_accuracy	= ccu_iosc_32k_recalc_accuracy,
186};
187
188static struct ccu_common iosc_32k_clk = {
189	.hw.init	= CLK_HW_INIT_HW("iosc-32k", &iosc_clk.hw,
190					 &ccu_iosc_32k_ops,
191					 CLK_GET_RATE_NOCACHE),
192};
193
194static const struct clk_hw *ext_osc32k[] = { NULL }; /* updated during probe */
195
196static SUNXI_CCU_GATE_HWS(ext_osc32k_gate_clk, "ext-osc32k-gate",
197			  ext_osc32k, 0x0, BIT(4), 0);
198
199static const struct clk_hw *osc32k_parents[] = {
200	&iosc_32k_clk.hw,
201	&ext_osc32k_gate_clk.common.hw
202};
203
204static struct clk_init_data osc32k_init_data = {
205	.name		= "osc32k",
206	.ops		= &ccu_mux_ops,
207	.parent_hws	= osc32k_parents,
208	.num_parents	= ARRAY_SIZE(osc32k_parents), /* updated during probe */
209};
210
211static struct ccu_mux osc32k_clk = {
212	.mux	= _SUNXI_CCU_MUX(0, 1),
213	.common	= {
214		.reg		= LOSC_CTRL_REG,
215		.features	= CCU_FEATURE_KEY_FIELD,
216		.hw.init	= &osc32k_init_data,
217	},
218};
219
220/* This falls back to the global name for fwnodes without a named reference. */
221static const struct clk_parent_data osc24M[] = {
222	{ .fw_name = "hosc", .name = "osc24M" }
223};
224
225static struct ccu_gate osc24M_32k_clk = {
226	.enable	= BIT(16),
227	.common	= {
228		.reg		= LOSC_OUT_GATING_REG,
229		.prediv		= 750,
230		.features	= CCU_FEATURE_ALL_PREDIV,
231		.hw.init	= CLK_HW_INIT_PARENTS_DATA("osc24M-32k", osc24M,
232							   &ccu_gate_ops, 0),
233	},
234};
235
236static const struct clk_hw *rtc_32k_parents[] = {
237	&osc32k_clk.common.hw,
238	&osc24M_32k_clk.common.hw
239};
240
241static struct clk_init_data rtc_32k_init_data = {
242	.name		= "rtc-32k",
243	.ops		= &ccu_mux_ops,
244	.parent_hws	= rtc_32k_parents,
245	.num_parents	= ARRAY_SIZE(rtc_32k_parents), /* updated during probe */
246	.flags		= CLK_IS_CRITICAL,
247};
248
249static struct ccu_mux rtc_32k_clk = {
250	.mux	= _SUNXI_CCU_MUX(1, 1),
251	.common	= {
252		.reg		= LOSC_CTRL_REG,
253		.features	= CCU_FEATURE_KEY_FIELD,
254		.hw.init	= &rtc_32k_init_data,
255	},
256};
257
258static struct clk_init_data osc32k_fanout_init_data = {
259	.name		= "osc32k-fanout",
260	.ops		= &ccu_mux_ops,
261	/* parents are set during probe */
262};
263
264static struct ccu_mux osc32k_fanout_clk = {
265	.enable	= BIT(0),
266	.mux	= _SUNXI_CCU_MUX(1, 2),
267	.common	= {
268		.reg		= LOSC_OUT_GATING_REG,
269		.hw.init	= &osc32k_fanout_init_data,
270	},
271};
272
273static struct ccu_common *sun6i_rtc_ccu_clks[] = {
274	&iosc_clk,
275	&iosc_32k_clk,
276	&ext_osc32k_gate_clk.common,
277	&osc32k_clk.common,
278	&osc24M_32k_clk.common,
279	&rtc_32k_clk.common,
280	&osc32k_fanout_clk.common,
281};
282
283static struct clk_hw_onecell_data sun6i_rtc_ccu_hw_clks = {
284	.num = CLK_NUMBER,
285	.hws = {
286		[CLK_OSC32K]		= &osc32k_clk.common.hw,
287		[CLK_OSC32K_FANOUT]	= &osc32k_fanout_clk.common.hw,
288		[CLK_IOSC]		= &iosc_clk.hw,
289		[CLK_IOSC_32K]		= &iosc_32k_clk.hw,
290		[CLK_EXT_OSC32K_GATE]	= &ext_osc32k_gate_clk.common.hw,
291		[CLK_OSC24M_32K]	= &osc24M_32k_clk.common.hw,
292		[CLK_RTC_32K]		= &rtc_32k_clk.common.hw,
293	},
294};
295
296static const struct sunxi_ccu_desc sun6i_rtc_ccu_desc = {
297	.ccu_clks	= sun6i_rtc_ccu_clks,
298	.num_ccu_clks	= ARRAY_SIZE(sun6i_rtc_ccu_clks),
299
300	.hw_clks	= &sun6i_rtc_ccu_hw_clks,
301};
302
303static const struct clk_parent_data sun50i_h616_osc32k_fanout_parents[] = {
304	{ .hw = &osc32k_clk.common.hw },
305	{ .fw_name = "pll-32k" },
306	{ .hw = &osc24M_32k_clk.common.hw }
307};
308
309static const struct clk_parent_data sun50i_r329_osc32k_fanout_parents[] = {
310	{ .hw = &osc32k_clk.common.hw },
311	{ .hw = &ext_osc32k_gate_clk.common.hw },
312	{ .hw = &osc24M_32k_clk.common.hw }
313};
314
315static const struct sun6i_rtc_match_data sun50i_h616_rtc_ccu_data = {
316	.have_iosc_calibration	= true,
317	.rtc_32k_single_parent	= true,
318	.osc32k_fanout_parents	= sun50i_h616_osc32k_fanout_parents,
319	.osc32k_fanout_nparents	= ARRAY_SIZE(sun50i_h616_osc32k_fanout_parents),
320};
321
322static const struct sun6i_rtc_match_data sun50i_r329_rtc_ccu_data = {
323	.have_ext_osc32k	= true,
324	.osc32k_fanout_parents	= sun50i_r329_osc32k_fanout_parents,
325	.osc32k_fanout_nparents	= ARRAY_SIZE(sun50i_r329_osc32k_fanout_parents),
326};
327
328static const struct of_device_id sun6i_rtc_ccu_match[] = {
329	{
330		.compatible	= "allwinner,sun50i-h616-rtc",
331		.data		= &sun50i_h616_rtc_ccu_data,
332	},
333	{
334		.compatible	= "allwinner,sun50i-r329-rtc",
335		.data		= &sun50i_r329_rtc_ccu_data,
336	},
337	{},
338};
339
340int sun6i_rtc_ccu_probe(struct device *dev, void __iomem *reg)
341{
342	const struct sun6i_rtc_match_data *data;
343	struct clk *ext_osc32k_clk = NULL;
344	const struct of_device_id *match;
345
346	/* This driver is only used for newer variants of the hardware. */
347	match = of_match_device(sun6i_rtc_ccu_match, dev);
348	if (!match)
349		return 0;
350
351	data = match->data;
352	have_iosc_calibration = data->have_iosc_calibration;
353
354	if (data->have_ext_osc32k) {
355		const char *fw_name;
356
357		/* ext-osc32k was the only input clock in the old binding. */
358		fw_name = of_property_read_bool(dev->of_node, "clock-names")
359			? "ext-osc32k" : NULL;
360		ext_osc32k_clk = devm_clk_get_optional(dev, fw_name);
361		if (IS_ERR(ext_osc32k_clk))
362			return PTR_ERR(ext_osc32k_clk);
363	}
364
365	if (ext_osc32k_clk) {
366		/* Link ext-osc32k-gate to its parent. */
367		*ext_osc32k = __clk_get_hw(ext_osc32k_clk);
368	} else {
369		/* ext-osc32k-gate is an orphan, so do not register it. */
370		sun6i_rtc_ccu_hw_clks.hws[CLK_EXT_OSC32K_GATE] = NULL;
371		osc32k_init_data.num_parents = 1;
372	}
373
374	if (data->rtc_32k_single_parent)
375		rtc_32k_init_data.num_parents = 1;
376
377	osc32k_fanout_init_data.parent_data = data->osc32k_fanout_parents;
378	osc32k_fanout_init_data.num_parents = data->osc32k_fanout_nparents;
379
380	return devm_sunxi_ccu_probe(dev, reg, &sun6i_rtc_ccu_desc);
381}
382
383MODULE_IMPORT_NS(SUNXI_CCU);
384MODULE_LICENSE("GPL");
385