xref: /openbmc/linux/drivers/clk/at91/at91sam9n12.c (revision 2612e3bbc0386368a850140a6c9b990cd496a5ec)
1143e04daSAlexandre Belloni // SPDX-License-Identifier: GPL-2.0
2143e04daSAlexandre Belloni #include <linux/clk-provider.h>
3143e04daSAlexandre Belloni #include <linux/mfd/syscon.h>
4143e04daSAlexandre Belloni #include <linux/slab.h>
5143e04daSAlexandre Belloni 
6143e04daSAlexandre Belloni #include <dt-bindings/clock/at91.h>
7143e04daSAlexandre Belloni 
8143e04daSAlexandre Belloni #include "pmc.h"
9143e04daSAlexandre Belloni 
107a110b91SClaudiu Beznea static DEFINE_SPINLOCK(at91sam9n12_mck_lock);
117a110b91SClaudiu Beznea 
12143e04daSAlexandre Belloni static const struct clk_master_characteristics mck_characteristics = {
13143e04daSAlexandre Belloni 	.output = { .min = 0, .max = 133333333 },
14143e04daSAlexandre Belloni 	.divisors = { 1, 2, 4, 3 },
15143e04daSAlexandre Belloni 	.have_div3_pres = 1,
16143e04daSAlexandre Belloni };
17143e04daSAlexandre Belloni 
18143e04daSAlexandre Belloni static u8 plla_out[] = { 0, 1, 2, 3, 0, 1, 2, 3 };
19143e04daSAlexandre Belloni 
20143e04daSAlexandre Belloni static u16 plla_icpll[] = { 0, 0, 0, 0, 1, 1, 1, 1 };
21143e04daSAlexandre Belloni 
22143e04daSAlexandre Belloni static const struct clk_range plla_outputs[] = {
23143e04daSAlexandre Belloni 	{ .min = 745000000, .max = 800000000 },
24143e04daSAlexandre Belloni 	{ .min = 695000000, .max = 750000000 },
25143e04daSAlexandre Belloni 	{ .min = 645000000, .max = 700000000 },
26143e04daSAlexandre Belloni 	{ .min = 595000000, .max = 650000000 },
27143e04daSAlexandre Belloni 	{ .min = 545000000, .max = 600000000 },
28143e04daSAlexandre Belloni 	{ .min = 495000000, .max = 555000000 },
29143e04daSAlexandre Belloni 	{ .min = 445000000, .max = 500000000 },
30143e04daSAlexandre Belloni 	{ .min = 400000000, .max = 450000000 },
31143e04daSAlexandre Belloni };
32143e04daSAlexandre Belloni 
33143e04daSAlexandre Belloni static const struct clk_pll_characteristics plla_characteristics = {
34143e04daSAlexandre Belloni 	.input = { .min = 2000000, .max = 32000000 },
35143e04daSAlexandre Belloni 	.num_output = ARRAY_SIZE(plla_outputs),
36143e04daSAlexandre Belloni 	.output = plla_outputs,
37143e04daSAlexandre Belloni 	.icpll = plla_icpll,
38143e04daSAlexandre Belloni 	.out = plla_out,
39143e04daSAlexandre Belloni };
40143e04daSAlexandre Belloni 
41143e04daSAlexandre Belloni static u8 pllb_out[] = { 0 };
42143e04daSAlexandre Belloni 
43143e04daSAlexandre Belloni static const struct clk_range pllb_outputs[] = {
44143e04daSAlexandre Belloni 	{ .min = 30000000, .max = 100000000 },
45143e04daSAlexandre Belloni };
46143e04daSAlexandre Belloni 
47143e04daSAlexandre Belloni static const struct clk_pll_characteristics pllb_characteristics = {
48143e04daSAlexandre Belloni 	.input = { .min = 2000000, .max = 32000000 },
49143e04daSAlexandre Belloni 	.num_output = ARRAY_SIZE(pllb_outputs),
50143e04daSAlexandre Belloni 	.output = pllb_outputs,
51143e04daSAlexandre Belloni 	.out = pllb_out,
52143e04daSAlexandre Belloni };
53143e04daSAlexandre Belloni 
54143e04daSAlexandre Belloni static const struct {
55143e04daSAlexandre Belloni 	char *n;
56143e04daSAlexandre Belloni 	char *p;
5768b3b6f1SClaudiu Beznea 	unsigned long flags;
58143e04daSAlexandre Belloni 	u8 id;
59143e04daSAlexandre Belloni } at91sam9n12_systemck[] = {
6068b3b6f1SClaudiu Beznea 	/*
6168b3b6f1SClaudiu Beznea 	 * ddrck feeds DDR controller and is enabled by bootloader thus we need
6268b3b6f1SClaudiu Beznea 	 * to keep it enabled in case there is no Linux consumer for it.
6368b3b6f1SClaudiu Beznea 	 */
6468b3b6f1SClaudiu Beznea 	{ .n = "ddrck", .p = "masterck_div", .id = 2, .flags = CLK_IS_CRITICAL },
657a110b91SClaudiu Beznea 	{ .n = "lcdck", .p = "masterck_div", .id = 3 },
66143e04daSAlexandre Belloni 	{ .n = "uhpck", .p = "usbck",        .id = 6 },
67143e04daSAlexandre Belloni 	{ .n = "udpck", .p = "usbck",        .id = 7 },
68143e04daSAlexandre Belloni 	{ .n = "pck0",  .p = "prog0",        .id = 8 },
69143e04daSAlexandre Belloni 	{ .n = "pck1",  .p = "prog1",        .id = 9 },
70143e04daSAlexandre Belloni };
71143e04daSAlexandre Belloni 
72143e04daSAlexandre Belloni static const struct clk_pcr_layout at91sam9n12_pcr_layout = {
73143e04daSAlexandre Belloni 	.offset = 0x10c,
74143e04daSAlexandre Belloni 	.cmd = BIT(12),
75143e04daSAlexandre Belloni 	.pid_mask = GENMASK(5, 0),
76143e04daSAlexandre Belloni 	.div_mask = GENMASK(17, 16),
77143e04daSAlexandre Belloni };
78143e04daSAlexandre Belloni 
79143e04daSAlexandre Belloni struct pck {
80143e04daSAlexandre Belloni 	char *n;
81143e04daSAlexandre Belloni 	u8 id;
82143e04daSAlexandre Belloni };
83143e04daSAlexandre Belloni 
84143e04daSAlexandre Belloni static const struct pck at91sam9n12_periphck[] = {
85143e04daSAlexandre Belloni 	{ .n = "pioAB_clk",  .id = 2, },
86143e04daSAlexandre Belloni 	{ .n = "pioCD_clk",  .id = 3, },
87143e04daSAlexandre Belloni 	{ .n = "fuse_clk",   .id = 4, },
88143e04daSAlexandre Belloni 	{ .n = "usart0_clk", .id = 5, },
89143e04daSAlexandre Belloni 	{ .n = "usart1_clk", .id = 6, },
90143e04daSAlexandre Belloni 	{ .n = "usart2_clk", .id = 7, },
91143e04daSAlexandre Belloni 	{ .n = "usart3_clk", .id = 8, },
92143e04daSAlexandre Belloni 	{ .n = "twi0_clk",   .id = 9, },
93143e04daSAlexandre Belloni 	{ .n = "twi1_clk",   .id = 10, },
94143e04daSAlexandre Belloni 	{ .n = "mci0_clk",   .id = 12, },
95143e04daSAlexandre Belloni 	{ .n = "spi0_clk",   .id = 13, },
96143e04daSAlexandre Belloni 	{ .n = "spi1_clk",   .id = 14, },
97143e04daSAlexandre Belloni 	{ .n = "uart0_clk",  .id = 15, },
98143e04daSAlexandre Belloni 	{ .n = "uart1_clk",  .id = 16, },
99143e04daSAlexandre Belloni 	{ .n = "tcb_clk",    .id = 17, },
100143e04daSAlexandre Belloni 	{ .n = "pwm_clk",    .id = 18, },
101143e04daSAlexandre Belloni 	{ .n = "adc_clk",    .id = 19, },
102143e04daSAlexandre Belloni 	{ .n = "dma0_clk",   .id = 20, },
103143e04daSAlexandre Belloni 	{ .n = "uhphs_clk",  .id = 22, },
104143e04daSAlexandre Belloni 	{ .n = "udphs_clk",  .id = 23, },
105143e04daSAlexandre Belloni 	{ .n = "lcdc_clk",   .id = 25, },
106143e04daSAlexandre Belloni 	{ .n = "sha_clk",    .id = 27, },
107143e04daSAlexandre Belloni 	{ .n = "ssc0_clk",   .id = 28, },
108143e04daSAlexandre Belloni 	{ .n = "aes_clk",    .id = 29, },
109143e04daSAlexandre Belloni 	{ .n = "trng_clk",   .id = 30, },
110143e04daSAlexandre Belloni };
111143e04daSAlexandre Belloni 
at91sam9n12_pmc_setup(struct device_node * np)112143e04daSAlexandre Belloni static void __init at91sam9n12_pmc_setup(struct device_node *np)
113143e04daSAlexandre Belloni {
114143e04daSAlexandre Belloni 	struct clk_range range = CLK_RANGE(0, 0);
115143e04daSAlexandre Belloni 	const char *slck_name, *mainxtal_name;
116143e04daSAlexandre Belloni 	struct pmc_data *at91sam9n12_pmc;
117143e04daSAlexandre Belloni 	const char *parent_names[6];
118143e04daSAlexandre Belloni 	struct regmap *regmap;
119143e04daSAlexandre Belloni 	struct clk_hw *hw;
120143e04daSAlexandre Belloni 	int i;
121143e04daSAlexandre Belloni 	bool bypass;
122143e04daSAlexandre Belloni 
123143e04daSAlexandre Belloni 	i = of_property_match_string(np, "clock-names", "slow_clk");
124143e04daSAlexandre Belloni 	if (i < 0)
125143e04daSAlexandre Belloni 		return;
126143e04daSAlexandre Belloni 
127143e04daSAlexandre Belloni 	slck_name = of_clk_get_parent_name(np, i);
128143e04daSAlexandre Belloni 
129143e04daSAlexandre Belloni 	i = of_property_match_string(np, "clock-names", "main_xtal");
130143e04daSAlexandre Belloni 	if (i < 0)
131143e04daSAlexandre Belloni 		return;
132143e04daSAlexandre Belloni 	mainxtal_name = of_clk_get_parent_name(np, i);
133143e04daSAlexandre Belloni 
134153bc1c6SAhmad Fatoum 	regmap = device_node_to_regmap(np);
135143e04daSAlexandre Belloni 	if (IS_ERR(regmap))
136143e04daSAlexandre Belloni 		return;
137143e04daSAlexandre Belloni 
13803a1ee1dSMichał Mirosław 	at91sam9n12_pmc = pmc_data_allocate(PMC_PLLBCK + 1,
13999767cd4SMichał Mirosław 					   nck(at91sam9n12_systemck), 31, 0, 2);
140143e04daSAlexandre Belloni 	if (!at91sam9n12_pmc)
141143e04daSAlexandre Belloni 		return;
142143e04daSAlexandre Belloni 
143143e04daSAlexandre Belloni 	hw = at91_clk_register_main_rc_osc(regmap, "main_rc_osc", 12000000,
144143e04daSAlexandre Belloni 					   50000000);
145143e04daSAlexandre Belloni 	if (IS_ERR(hw))
146143e04daSAlexandre Belloni 		goto err_free;
147143e04daSAlexandre Belloni 
148143e04daSAlexandre Belloni 	bypass = of_property_read_bool(np, "atmel,osc-bypass");
149143e04daSAlexandre Belloni 
150b5105e37SClaudiu Beznea 	hw = at91_clk_register_main_osc(regmap, "main_osc", mainxtal_name, NULL,
151143e04daSAlexandre Belloni 					bypass);
152143e04daSAlexandre Belloni 	if (IS_ERR(hw))
153143e04daSAlexandre Belloni 		goto err_free;
154143e04daSAlexandre Belloni 
155143e04daSAlexandre Belloni 	parent_names[0] = "main_rc_osc";
156143e04daSAlexandre Belloni 	parent_names[1] = "main_osc";
157b5105e37SClaudiu Beznea 	hw = at91_clk_register_sam9x5_main(regmap, "mainck", parent_names, NULL, 2);
158143e04daSAlexandre Belloni 	if (IS_ERR(hw))
159143e04daSAlexandre Belloni 		goto err_free;
160143e04daSAlexandre Belloni 
161143e04daSAlexandre Belloni 	at91sam9n12_pmc->chws[PMC_MAIN] = hw;
162143e04daSAlexandre Belloni 
163143e04daSAlexandre Belloni 	hw = at91_clk_register_pll(regmap, "pllack", "mainck", 0,
164143e04daSAlexandre Belloni 				   &at91rm9200_pll_layout, &plla_characteristics);
165143e04daSAlexandre Belloni 	if (IS_ERR(hw))
166143e04daSAlexandre Belloni 		goto err_free;
167143e04daSAlexandre Belloni 
168143e04daSAlexandre Belloni 	hw = at91_clk_register_plldiv(regmap, "plladivck", "pllack");
169143e04daSAlexandre Belloni 	if (IS_ERR(hw))
170143e04daSAlexandre Belloni 		goto err_free;
171143e04daSAlexandre Belloni 
17203a1ee1dSMichał Mirosław 	at91sam9n12_pmc->chws[PMC_PLLACK] = hw;
17303a1ee1dSMichał Mirosław 
174143e04daSAlexandre Belloni 	hw = at91_clk_register_pll(regmap, "pllbck", "mainck", 1,
175143e04daSAlexandre Belloni 				   &at91rm9200_pll_layout, &pllb_characteristics);
176143e04daSAlexandre Belloni 	if (IS_ERR(hw))
177143e04daSAlexandre Belloni 		goto err_free;
178143e04daSAlexandre Belloni 
17903a1ee1dSMichał Mirosław 	at91sam9n12_pmc->chws[PMC_PLLBCK] = hw;
18003a1ee1dSMichał Mirosław 
181143e04daSAlexandre Belloni 	parent_names[0] = slck_name;
182143e04daSAlexandre Belloni 	parent_names[1] = "mainck";
183143e04daSAlexandre Belloni 	parent_names[2] = "plladivck";
184143e04daSAlexandre Belloni 	parent_names[3] = "pllbck";
1857a110b91SClaudiu Beznea 	hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
186171e502cSClaudiu Beznea 					   parent_names, NULL,
187143e04daSAlexandre Belloni 					   &at91sam9x5_master_layout,
1887a110b91SClaudiu Beznea 					   &mck_characteristics,
1898e842f02SClaudiu Beznea 					   &at91sam9n12_mck_lock);
1907a110b91SClaudiu Beznea 	if (IS_ERR(hw))
1917a110b91SClaudiu Beznea 		goto err_free;
1927a110b91SClaudiu Beznea 
1937a110b91SClaudiu Beznea 	hw = at91_clk_register_master_div(regmap, "masterck_div",
194171e502cSClaudiu Beznea 					  "masterck_pres", NULL,
1957a110b91SClaudiu Beznea 					  &at91sam9x5_master_layout,
1967a110b91SClaudiu Beznea 					  &mck_characteristics,
1977a110b91SClaudiu Beznea 					  &at91sam9n12_mck_lock,
1987029db09SClaudiu Beznea 					  CLK_SET_RATE_GATE, 0);
199143e04daSAlexandre Belloni 	if (IS_ERR(hw))
200143e04daSAlexandre Belloni 		goto err_free;
201143e04daSAlexandre Belloni 
202143e04daSAlexandre Belloni 	at91sam9n12_pmc->chws[PMC_MCK] = hw;
203143e04daSAlexandre Belloni 
204143e04daSAlexandre Belloni 	hw = at91sam9n12_clk_register_usb(regmap, "usbck", "pllbck");
205143e04daSAlexandre Belloni 	if (IS_ERR(hw))
206143e04daSAlexandre Belloni 		goto err_free;
207143e04daSAlexandre Belloni 
208143e04daSAlexandre Belloni 	parent_names[0] = slck_name;
209143e04daSAlexandre Belloni 	parent_names[1] = "mainck";
210143e04daSAlexandre Belloni 	parent_names[2] = "plladivck";
211143e04daSAlexandre Belloni 	parent_names[3] = "pllbck";
2127a110b91SClaudiu Beznea 	parent_names[4] = "masterck_div";
213143e04daSAlexandre Belloni 	for (i = 0; i < 2; i++) {
214143e04daSAlexandre Belloni 		char name[6];
215143e04daSAlexandre Belloni 
216143e04daSAlexandre Belloni 		snprintf(name, sizeof(name), "prog%d", i);
217143e04daSAlexandre Belloni 
218143e04daSAlexandre Belloni 		hw = at91_clk_register_programmable(regmap, name,
2191a2669dfSClaudiu Beznea 						    parent_names, NULL, 5, i,
220c57aaaa2SClaudiu Beznea 						    &at91sam9x5_programmable_layout,
221c57aaaa2SClaudiu Beznea 						    NULL);
222143e04daSAlexandre Belloni 		if (IS_ERR(hw))
223143e04daSAlexandre Belloni 			goto err_free;
22499767cd4SMichał Mirosław 
22599767cd4SMichał Mirosław 		at91sam9n12_pmc->pchws[i] = hw;
226143e04daSAlexandre Belloni 	}
227143e04daSAlexandre Belloni 
228143e04daSAlexandre Belloni 	for (i = 0; i < ARRAY_SIZE(at91sam9n12_systemck); i++) {
229143e04daSAlexandre Belloni 		hw = at91_clk_register_system(regmap, at91sam9n12_systemck[i].n,
230*1a537f62SClaudiu Beznea 					      at91sam9n12_systemck[i].p, NULL,
23168b3b6f1SClaudiu Beznea 					      at91sam9n12_systemck[i].id,
23268b3b6f1SClaudiu Beznea 					      at91sam9n12_systemck[i].flags);
233143e04daSAlexandre Belloni 		if (IS_ERR(hw))
234143e04daSAlexandre Belloni 			goto err_free;
235143e04daSAlexandre Belloni 
236143e04daSAlexandre Belloni 		at91sam9n12_pmc->shws[at91sam9n12_systemck[i].id] = hw;
237143e04daSAlexandre Belloni 	}
238143e04daSAlexandre Belloni 
239143e04daSAlexandre Belloni 	for (i = 0; i < ARRAY_SIZE(at91sam9n12_periphck); i++) {
240143e04daSAlexandre Belloni 		hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
241143e04daSAlexandre Belloni 							 &at91sam9n12_pcr_layout,
242143e04daSAlexandre Belloni 							 at91sam9n12_periphck[i].n,
243c2f2ca0bSClaudiu Beznea 							 "masterck_div", NULL,
244143e04daSAlexandre Belloni 							 at91sam9n12_periphck[i].id,
24568b3b6f1SClaudiu Beznea 							 &range, INT_MIN, 0);
246143e04daSAlexandre Belloni 		if (IS_ERR(hw))
247143e04daSAlexandre Belloni 			goto err_free;
248143e04daSAlexandre Belloni 
249143e04daSAlexandre Belloni 		at91sam9n12_pmc->phws[at91sam9n12_periphck[i].id] = hw;
250143e04daSAlexandre Belloni 	}
251143e04daSAlexandre Belloni 
252143e04daSAlexandre Belloni 	of_clk_add_hw_provider(np, of_clk_hw_pmc_get, at91sam9n12_pmc);
253143e04daSAlexandre Belloni 
254143e04daSAlexandre Belloni 	return;
255143e04daSAlexandre Belloni 
256143e04daSAlexandre Belloni err_free:
2577425f246SMichał Mirosław 	kfree(at91sam9n12_pmc);
258143e04daSAlexandre Belloni }
259143e04daSAlexandre Belloni /*
260143e04daSAlexandre Belloni  * The TCB is used as the clocksource so its clock is needed early. This means
261143e04daSAlexandre Belloni  * this can't be a platform driver.
262143e04daSAlexandre Belloni  */
263428d97e1STudor Ambarus CLK_OF_DECLARE(at91sam9n12_pmc, "atmel,at91sam9n12-pmc", at91sam9n12_pmc_setup);
264