coreboot
coreboot is an Open Source project aimed at replacing the proprietary BIOS found in most computers.
pll.c
Go to the documentation of this file.
1 /* SPDX-License-Identifier: GPL-2.0-only */
2 
3 #include <console/console.h>
4 #include <device/mmio.h>
5 #include <delay.h>
6 #include <stddef.h>
7 #include <timer.h>
8 
9 #include <soc/addressmap.h>
10 #include <soc/infracfg.h>
11 #include <soc/mcucfg.h>
12 #include <soc/pll.h>
13 #include <soc/wdt.h>
14 
15 enum mux_id {
80 };
81 
82 #define MUX(_id, _reg, _mux_shift, _mux_width) \
83  [_id] = { \
84  .reg = &mtk_topckgen->_reg, \
85  .set_reg = &mtk_topckgen->_reg##_set, \
86  .clr_reg = &mtk_topckgen->_reg##_clr, \
87  .mux_shift = _mux_shift, \
88  .mux_width = _mux_width, \
89  .upd_reg = NULL, \
90  .upd_shift = 0, \
91  }
92 
93 #define MUX_UPD(_id, _reg, _mux_shift, _mux_width, _upd_reg, _upd_shift)\
94  [_id] = { \
95  .reg = &mtk_topckgen->_reg, \
96  .set_reg = &mtk_topckgen->_reg##_set, \
97  .clr_reg = &mtk_topckgen->_reg##_clr, \
98  .mux_shift = _mux_shift, \
99  .mux_width = _mux_width, \
100  .upd_reg = &mtk_topckgen->_upd_reg, \
101  .upd_shift = _upd_shift, \
102  }
103 
104 static const struct mux muxes[] = {
105  /* CLK_CFG_0 */
106  MUX_UPD(TOP_AXI_SEL, clk_cfg_0, 0, 3, clk_cfg_update, 0),
107  MUX_UPD(TOP_SPM_SEL, clk_cfg_0, 8, 2, clk_cfg_update, 1),
108  MUX_UPD(TOP_SCP_SEL, clk_cfg_0, 16, 3, clk_cfg_update, 2),
109  MUX_UPD(TOP_BUS_AXIMEM_SEL, clk_cfg_0, 24, 3, clk_cfg_update, 3),
110  /* CLK_CFG_1 */
111  MUX_UPD(TOP_DISP_SEL, clk_cfg_1, 0, 4, clk_cfg_update, 4),
112  MUX_UPD(TOP_MDP_SEL, clk_cfg_1, 8, 4, clk_cfg_update, 5),
113  MUX_UPD(TOP_IMG1_SEL, clk_cfg_1, 16, 4, clk_cfg_update, 6),
114  MUX_UPD(TOP_IMG2_SEL, clk_cfg_1, 24, 4, clk_cfg_update, 7),
115  /* CLK_CFG_2 */
116  MUX_UPD(TOP_IPE_SEL, clk_cfg_2, 0, 4, clk_cfg_update, 8),
117  MUX_UPD(TOP_DPE_SEL, clk_cfg_2, 8, 3, clk_cfg_update, 9),
118  MUX_UPD(TOP_CAM_SEL, clk_cfg_2, 16, 4, clk_cfg_update, 10),
119  MUX_UPD(TOP_CCU_SEL, clk_cfg_2, 24, 4, clk_cfg_update, 11),
120  /* CLK_CFG_3 */
121  MUX_UPD(TOP_DSP_SEL, clk_cfg_3, 0, 3, clk_cfg_update, 12),
122  /* CLK_CFG_4 */
123  MUX_UPD(TOP_DSP7_SEL, clk_cfg_4, 0, 3, clk_cfg_update, 16),
124  MUX_UPD(TOP_MFG_REF_SEL, clk_cfg_4, 16, 2, clk_cfg_update, 18),
125  MUX(TOP_MFG_PLL_SEL, clk_cfg_4, 18, 1),
126  MUX_UPD(TOP_CAMTG_SEL, clk_cfg_4, 24, 3, clk_cfg_update, 19),
127  /* CLK_CFG_5 */
128  MUX_UPD(TOP_CAMTG2_SEL, clk_cfg_5, 0, 3, clk_cfg_update, 20),
129  MUX_UPD(TOP_CAMTG3_SEL, clk_cfg_5, 8, 3, clk_cfg_update, 21),
130  MUX_UPD(TOP_CAMTG4_SEL, clk_cfg_5, 16, 3, clk_cfg_update, 22),
131  MUX_UPD(TOP_CAMTG5_SEL, clk_cfg_5, 24, 3, clk_cfg_update, 23),
132  /* CLK_CFG_6 */
133  MUX_UPD(TOP_CAMTG6_SEL, clk_cfg_6, 0, 3, clk_cfg_update, 24),
134  MUX_UPD(TOP_UART_SEL, clk_cfg_6, 8, 1, clk_cfg_update, 25),
135  MUX_UPD(TOP_SPI_SEL, clk_cfg_6, 16, 2, clk_cfg_update, 26),
136  MUX_UPD(TOP_MSDC50_0_HCLK_SEL, clk_cfg_6, 24, 2, clk_cfg_update, 27),
137  /* CLK_CFG_7 */
138  MUX_UPD(TOP_MSDC50_0_SEL, clk_cfg_7, 0, 3, clk_cfg_update, 28),
139  MUX_UPD(TOP_MSDC30_1_SEL, clk_cfg_7, 8, 3, clk_cfg_update, 29),
140  MUX_UPD(TOP_MSDC30_2_SEL, clk_cfg_7, 16, 3, clk_cfg_update, 30),
141  MUX_UPD(TOP_AUDIO_SEL, clk_cfg_7, 24, 2, clk_cfg_update1, 0),
142  /* CLK_CFG_8 */
143  MUX_UPD(TOP_AUD_INTBUS_SEL, clk_cfg_8, 0, 2, clk_cfg_update1, 1),
144  MUX_UPD(TOP_PWRAP_ULPOSC_SEL, clk_cfg_8, 8, 3, clk_cfg_update1, 2),
145  MUX_UPD(TOP_ATB_SEL, clk_cfg_8, 16, 2, clk_cfg_update1, 3),
146  MUX_UPD(TOP_PWRMCU_SEL, clk_cfg_8, 24, 3, clk_cfg_update1, 4),
147  /* CLK_CFG_9 */
148  MUX_UPD(TOP_DPI_SEL, clk_cfg_9, 0, 3, clk_cfg_update1, 5),
149  MUX_UPD(TOP_SCAM_SEL, clk_cfg_9, 8, 1, clk_cfg_update1, 6),
150  MUX_UPD(TOP_DISP_PWM_SEL, clk_cfg_9, 16, 3, clk_cfg_update1, 7),
151  MUX_UPD(TOP_USB_TOP_SEL, clk_cfg_9, 24, 2, clk_cfg_update1, 8),
152  /* CLK_CFG_10 */
153  MUX_UPD(TOP_SSUSB_XHCI_SEL, clk_cfg_10, 0, 2, clk_cfg_update1, 9),
154  MUX_UPD(TOP_I2C_SEL, clk_cfg_10, 8, 2, clk_cfg_update1, 10),
155  MUX_UPD(TOP_SENINF_SEL, clk_cfg_10, 16, 3, clk_cfg_update1, 11),
156  MUX_UPD(TOP_SENINF1_SEL, clk_cfg_10, 24, 3, clk_cfg_update1, 12),
157  /* CLK_CFG_11 */
158  MUX_UPD(TOP_SENINF2_SEL, clk_cfg_11, 0, 3, clk_cfg_update1, 13),
159  MUX_UPD(TOP_SENINF3_SEL, clk_cfg_11, 8, 3, clk_cfg_update1, 14),
160  MUX_UPD(TOP_TL_SEL, clk_cfg_11, 16, 2, clk_cfg_update1, 15),
161  MUX_UPD(TOP_DXCC_SEL, clk_cfg_11, 24, 2, clk_cfg_update1, 16),
162  /* CLK_CFG_12 */
163  MUX_UPD(TOP_AUD_ENGEN1_SEL, clk_cfg_12, 0, 2, clk_cfg_update1, 17),
164  MUX_UPD(TOP_AUD_ENGEN2_SEL, clk_cfg_12, 8, 2, clk_cfg_update1, 18),
165  MUX_UPD(TOP_AES_UFSFDE_SEL, clk_cfg_12, 16, 3, clk_cfg_update1, 19),
166  MUX_UPD(TOP_UFS_SEL, clk_cfg_12, 24, 3, clk_cfg_update1, 20),
167  /* CLK_CFG_13 */
168  MUX_UPD(TOP_AUD_1_SEL, clk_cfg_13, 0, 1, clk_cfg_update1, 21),
169  MUX_UPD(TOP_AUD_2_SEL, clk_cfg_13, 8, 1, clk_cfg_update1, 22),
170  MUX_UPD(TOP_ADSP_SEL, clk_cfg_13, 16, 3, clk_cfg_update1, 23),
171  MUX_UPD(TOP_DPMAIF_MAIN_SEL, clk_cfg_13, 24, 3, clk_cfg_update1, 24),
172  /* CLK_CFG_14 */
173  MUX_UPD(TOP_VENC_SEL, clk_cfg_14, 0, 4, clk_cfg_update1, 25),
174  MUX_UPD(TOP_VDEC_SEL, clk_cfg_14, 8, 4, clk_cfg_update1, 26),
175  MUX_UPD(TOP_CAMTM_SEL, clk_cfg_14, 16, 2, clk_cfg_update1, 27),
176  MUX_UPD(TOP_PWM_SEL, clk_cfg_14, 24, 1, clk_cfg_update1, 28),
177  /* CLK_CFG_15 */
178  MUX_UPD(TOP_AUDIO_H_SEL, clk_cfg_15, 0, 2, clk_cfg_update1, 29),
179  MUX_UPD(TOP_SPMI_MST_SEL, clk_cfg_15, 8, 3, clk_cfg_update1, 30),
180  MUX_UPD(TOP_DVFSRC_SEL, clk_cfg_15, 16, 1, clk_cfg_update2, 0),
181  MUX_UPD(TOP_AES_MSDCFDE_SEL, clk_cfg_15, 24, 3, clk_cfg_update2, 1),
182  /* CLK_CFG_16 */
183  MUX_UPD(TOP_MCUPM_SEL, clk_cfg_16, 0, 2, clk_cfg_update2, 2),
184  MUX_UPD(TOP_SFLASH_SEL, clk_cfg_16, 8, 2, clk_cfg_update2, 3),
185 };
186 
187 struct mux_sel {
188  enum mux_id id;
189  u32 sel;
190 };
191 
192 static const struct mux_sel mux_sels[] = {
193  /* CLK_CFG_0 */
194  { .id = TOP_AXI_SEL, .sel = 2 }, /* 2: mainpll_d7_d2 */
195  { .id = TOP_SPM_SEL, .sel = 2 }, /* 2: mainpll_d7_d4 */
196  { .id = TOP_SCP_SEL, .sel = 0 }, /* 0: clk26m */
197  { .id = TOP_BUS_AXIMEM_SEL, .sel = 3 }, /* 3: mainpll_d5_d2 */
198  /* CLK_CFG_1 */
199  { .id = TOP_DISP_SEL, .sel = 8 }, /* 8: mainpll_d4 */
200  { .id = TOP_MDP_SEL, .sel = 8 }, /* 8: tvdpll_ck */
201  { .id = TOP_IMG1_SEL, .sel = 1 }, /* 1: univpll_d4 */
202  { .id = TOP_IMG2_SEL, .sel = 1 }, /* 1: univpll_d4 */
203  /* CLK_CFG_2 */
204  { .id = TOP_IPE_SEL, .sel = 1 }, /* 1: mainpll_d4 */
205  { .id = TOP_DPE_SEL, .sel = 1 }, /* 1: mainpll_d4 */
206  { .id = TOP_CAM_SEL, .sel = 3 }, /* 3: univpll_d4 */
207  { .id = TOP_CCU_SEL, .sel = 8 }, /* 8: univpll_d5 */
208  /* CLK_CFG_3 */
209  { .id = TOP_DSP_SEL, .sel = 1 }, /* 1: univpll_d6_d2 */
210  /* CLK_CFG_4 */
211  { .id = TOP_DSP7_SEL, .sel = 1 }, /* 1: mainpll_d4_d2 */
212  { .id = TOP_MFG_REF_SEL, .sel = 3 }, /* 3: mainpll_d5_d2 */
213  { .id = TOP_MFG_PLL_SEL, .sel = 1 }, /* 1: mfgpll */
214  { .id = TOP_CAMTG_SEL, .sel = 1 }, /* 1: univpll_192m_d8 */
215  /* CLK_CFG_5 */
216  { .id = TOP_CAMTG2_SEL, .sel = 1 }, /* 1: univpll_192m_d8 */
217  { .id = TOP_CAMTG3_SEL, .sel = 1 }, /* 1: univpll_192m_d8 */
218  { .id = TOP_CAMTG4_SEL, .sel = 1 }, /* 1: univpll_192m_d8 */
219  { .id = TOP_CAMTG5_SEL, .sel = 1 }, /* 1: univpll_192m_d8 */
220  /* CLK_CFG_6 */
221  { .id = TOP_CAMTG6_SEL, .sel = 1 }, /* 1: univpll_192m_d8 */
222  { .id = TOP_UART_SEL, .sel = 0 }, /* 0: clk26m */
223  { .id = TOP_SPI_SEL, .sel = 1 }, /* 1: mainpll_d5_d4 */
224  { .id = TOP_MSDC50_0_HCLK_SEL, .sel = 1 }, /* 1: mainpll_d4_d2 */
225  /* CLK_CFG_7 */
226  { .id = TOP_MSDC50_0_SEL, .sel = 1 }, /* 1: msdcpll_ck */
227  { .id = TOP_MSDC30_1_SEL, .sel = 4 }, /* 4: msdcpll_d2 */
228  { .id = TOP_MSDC30_2_SEL, .sel = 4 }, /* 4: msdcpll_d2 */
229  { .id = TOP_AUDIO_SEL, .sel = 0 }, /* 0: clk26m */
230  /* CLK_CFG_8 */
231  { .id = TOP_AUD_INTBUS_SEL, .sel = 1 }, /* 1: mainpll_d4_d4 */
232  { .id = TOP_PWRAP_ULPOSC_SEL, .sel = 0 }, /* 0: osc_d10 */
233  { .id = TOP_ATB_SEL, .sel = 1 }, /* 1: mainpll_d4_d2 */
234  { .id = TOP_PWRMCU_SEL, .sel = 3 }, /* 3: mainpll_d4_d2 */
235  /* CLK_CFG_9 */
236  { .id = TOP_DPI_SEL, .sel = 1 }, /* 1: tvdpll_d2 */
237  { .id = TOP_SCAM_SEL, .sel = 1 }, /* 1: mainpll_d5_d4 */
238  { .id = TOP_DISP_PWM_SEL, .sel = 0 }, /* 0: clk26m */
239  { .id = TOP_USB_TOP_SEL, .sel = 1 }, /* 1: univpll_d5_d4 */
240  /* CLK_CFG_10 */
241  { .id = TOP_SSUSB_XHCI_SEL, .sel = 1 }, /* 1: univpll_d5_d4 */
242  { .id = TOP_I2C_SEL, .sel = 2 }, /* 2: univpll_d5_d4 */
243  { .id = TOP_SENINF_SEL, .sel = 4 }, /* 4: univpll_d7 */
244  { .id = TOP_SENINF1_SEL, .sel = 4 }, /* 4: univpll_d7 */
245  /* CLK_CFG_11 */
246  { .id = TOP_SENINF2_SEL, .sel = 4 }, /* 4: univpll_d7 */
247  { .id = TOP_SENINF3_SEL, .sel = 4 }, /* 4: univpll_d7 */
248  { .id = TOP_TL_SEL, .sel = 1 }, /* 1: univpll_192m_d2 */
249  { .id = TOP_DXCC_SEL, .sel = 1 }, /* 1: mainpll_d4_d2 */
250  /* CLK_CFG_12 */
251  { .id = TOP_AUD_ENGEN1_SEL, .sel = 2 }, /* 2: apll1_d4 */
252  { .id = TOP_AUD_ENGEN2_SEL, .sel = 2 }, /* 2: apll2_d4 */
253  { .id = TOP_AES_UFSFDE_SEL, .sel = 6 }, /* 6: univpll_d6 */
254  { .id = TOP_UFS_SEL, .sel = 6 }, /* 6: msdcpll_d2 */
255  /* CLK_CFG_13 */
256  { .id = TOP_AUD_1_SEL, .sel = 1 }, /* 1: apll1_ck */
257  { .id = TOP_AUD_2_SEL, .sel = 1 }, /* 1: apll2_ck */
258  { .id = TOP_ADSP_SEL, .sel = 7 }, /* 7: adsppll_ck */
259  { .id = TOP_DPMAIF_MAIN_SEL, .sel = 3 }, /* 3: mainpll_d4_d2 */
260  /* CLK_CFG_14 */
261  { .id = TOP_VENC_SEL, .sel = 14 }, /* 14: univpll_d5_d2 */
262  { .id = TOP_VDEC_SEL, .sel = 4 }, /* 4: mainpll_d5_d2 */
263  { .id = TOP_CAMTM_SEL, .sel = 2 }, /* 2: univpll_d6_d2 */
264  { .id = TOP_PWM_SEL, .sel = 0 }, /* 0: clk26m */
265  /* CLK_CFG_15 */
266  { .id = TOP_AUDIO_H_SEL, .sel = 3 }, /* 3: apll2_ck */
267  { .id = TOP_SPMI_MST_SEL, .sel = 0 }, /* 0: clk26m */
268  { .id = TOP_DVFSRC_SEL, .sel = 0 }, /* 0: clk26m */
269  { .id = TOP_AES_MSDCFDE_SEL, .sel = 5 }, /* 5: univpll_d6 */
270  /* CLK_CFG_16 */
271  { .id = TOP_MCUPM_SEL, .sel = 2 }, /* 2: mainpll_d6_d2 */
272  { .id = TOP_SFLASH_SEL, .sel = 1 }, /* 1: mainpll_d7_d8 */
273 };
274 
275 enum pll_id {
290 };
291 
292 const u32 pll_div_rate[] = {
293  3800UL * MHz,
294  1900 * MHz,
295  950 * MHz,
296  475 * MHz,
297  237500 * KHz,
298  0,
299 };
300 
301 static const struct pll plls[] = {
302  PLL(APMIXED_ARMPLL_LL, armpll_ll_con0, armpll_ll_con3,
303  NO_RSTB_SHIFT, 22, armpll_ll_con1, 24, armpll_ll_con1, 0,
304  pll_div_rate),
305  PLL(APMIXED_ARMPLL_BL, armpll_bl0_con0, armpll_bl_con3,
306  NO_RSTB_SHIFT, 22, armpll_bl_con1, 24, armpll_bl_con1, 0,
307  pll_div_rate),
308  PLL(APMIXED_CCIPLL, ccipll_con0, ccipll_con3,
309  NO_RSTB_SHIFT, 22, ccipll_con1, 24, ccipll_con1, 0,
310  pll_div_rate),
311  PLL(APMIXED_MAINPLL, mainpll_con0, mainpll_con3,
312  23, 22, mainpll_con1, 24, mainpll_con1, 0,
313  pll_div_rate),
314  PLL(APMIXED_UNIVPLL, univpll_con0, univpll_con3,
315  23, 22, univpll_con1, 24, univpll_con1, 0,
316  pll_div_rate),
317  PLL(APMIXED_USBPLL, usbpll_con0, usbpll_con2,
318  NO_RSTB_SHIFT, 22, usbpll_con0, 24, usbpll_con0, 0,
319  pll_div_rate),
320  PLL(APMIXED_MSDCPLL, msdcpll_con0, msdcpll_con3,
321  NO_RSTB_SHIFT, 22, msdcpll_con1, 24, msdcpll_con1, 0,
322  pll_div_rate),
323  PLL(APMIXED_MMPLL, mmpll_con0, mmpll_con3,
324  23, 22, mmpll_con1, 24, mmpll_con1, 0,
325  pll_div_rate),
326  PLL(APMIXED_ADSPPLL, adsppll_con0, adsppll_con3,
327  NO_RSTB_SHIFT, 22, adsppll_con1, 24, adsppll_con1, 0,
328  pll_div_rate),
329  PLL(APMIXED_MFGPLL, mfgpll_con0, mfgpll_con3,
330  NO_RSTB_SHIFT, 22, mfgpll_con1, 24, mfgpll_con1, 0,
331  pll_div_rate),
332  PLL(APMIXED_TVDPLL, tvdpll_con0, tvdpll_con3,
333  NO_RSTB_SHIFT, 22, tvdpll_con1, 24, tvdpll_con1, 0,
334  pll_div_rate),
335  PLL(APMIXED_APLL1, apll1_con0, apll1_con4,
336  NO_RSTB_SHIFT, 32, apll1_con1, 24, apll1_con2, 0,
337  pll_div_rate),
338  PLL(APMIXED_APLL2, apll2_con0, apll2_con4,
339  NO_RSTB_SHIFT, 32, apll2_con1, 24, apll2_con2, 0,
340  pll_div_rate),
341 };
342 
343 struct rate {
344  enum pll_id id;
345  u32 rate;
346 };
347 
348 static const struct rate rates[] = {
349  { .id = APMIXED_ARMPLL_LL, .rate = ARMPLL_LL_HZ },
350  { .id = APMIXED_ARMPLL_BL, .rate = ARMPLL_BL_HZ },
351  { .id = APMIXED_CCIPLL, .rate = CCIPLL_HZ },
352  { .id = APMIXED_MAINPLL, .rate = MAINPLL_HZ },
353  { .id = APMIXED_UNIVPLL, .rate = UNIVPLL_HZ },
354  { .id = APMIXED_USBPLL, .rate = USBPLL_HZ },
355  { .id = APMIXED_MSDCPLL, .rate = MSDCPLL_HZ },
356  { .id = APMIXED_MMPLL, .rate = MMPLL_HZ },
357  { .id = APMIXED_ADSPPLL, .rate = ADSPPLL_HZ },
358  { .id = APMIXED_MFGPLL, .rate = MFGPLL_HZ },
359  { .id = APMIXED_TVDPLL, .rate = TVDPLL_HZ },
360  { .id = APMIXED_APLL1, .rate = APLL1_HZ },
361  { .id = APMIXED_APLL2, .rate = APLL2_HZ },
362 };
363 
364 void pll_set_pcw_change(const struct pll *pll)
365 {
367 }
368 
369 void mt_pll_init(void)
370 {
371  int i;
372 
373  /* enable clock square1 low-pass filter */
374  setbits32(&mtk_apmixed->ap_pll_con0, 0x2);
375 
376  /* reduce PLL current */
377  SET32_BITFIELDS(&mtk_apmixed->ap_pllgp1_con1, PLLGP1_LVRREF, 1);
378  SET32_BITFIELDS(&mtk_apmixed->ap_pllgp2_con1, PLLGP2_LVRREF, 1);
379 
380  /* xPLL PWR ON */
381  for (i = 0; i < APMIXED_PLL_MAX; i++)
382  setbits32(plls[i].pwr_reg, PLL_PWR_ON);
383 
385 
386  /* xPLL ISO Disable */
387  for (i = 0; i < APMIXED_PLL_MAX; i++)
388  clrbits32(plls[i].pwr_reg, PLL_ISO);
389 
391 
392  /* xPLL Frequency Set */
393  for (i = 0; i < ARRAY_SIZE(rates); i++)
394  pll_set_rate(&plls[rates[i].id], rates[i].rate);
395 
396  /* AUDPLL Tuner Frequency Set */
397  write32(&mtk_apmixed->apll1_tuner_con0, read32(&mtk_apmixed->apll1_con2) + 1);
398  write32(&mtk_apmixed->apll2_tuner_con0, read32(&mtk_apmixed->apll2_con2) + 1);
399 
400  /* xPLL Frequency Enable */
401  for (i = 0; i < APMIXED_PLL_MAX; i++) {
402  if (i == APMIXED_USBPLL)
404  else
405  setbits32(plls[i].reg, PLL_EN);
406  }
407 
408  /* wait for PLL stable */
410 
411  /* xPLL DIV Enable & RSTB */
412  for (i = 0; i < APMIXED_PLL_MAX; i++) {
413  if (plls[i].rstb_shift != NO_RSTB_SHIFT) {
414  setbits32(plls[i].reg, PLL_DIV_EN);
415  setbits32(plls[i].reg, 1 << plls[i].rstb_shift);
416  }
417  }
418 
419  /* MCUCFG CLKMUX */
423 
427 
428  /* enable infrasys DCM */
430 
431  /* dcm_infracfg_ao_aximem_bus_dcm */
435  /* dcm_infracfg_ao_infra_bus_dcm */
439  /* dcm_infracfg_ao_infra_conn_bus_dcm */
446  /* dcm_infracfg_ao_infra_rx_p2p_dcm */
450  /* dcm_infracfg_ao_peri_bus_dcm */
454  /* dcm_infracfg_ao_peri_module_dcm */
458 
459  /* initialize SPM request */
460  setbits32(&mtk_topckgen->clk_scp_cfg_0, 0x3ff);
461  clrsetbits32(&mtk_topckgen->clk_scp_cfg_1, 0x100c, 0x3);
462 
463  /*
464  * TOP CLKMUX -- DO NOT CHANGE WITHOUT ADJUSTING <soc/pll.h> CONSTANTS!
465  */
466  for (i = 0; i < ARRAY_SIZE(mux_sels); i++)
467  mux_set_sel(&muxes[mux_sels[i].id], mux_sels[i].sel);
468 
469  /* enable [14] dramc_pll104m_ck */
470  setbits32(&mtk_topckgen->clk_misc_cfg_0, 1 << 14);
471 
472  /* reset CONNSYS MCU */
474  WDT_SWSYSRST_KEY, 0x88,
475  WDT_SWSYSRST_CONN_MCU, 0x1);
476 }
477 
479 {
480  /* enable [4] intermediate clock armpll_divider_pll1_ck */
481  setbits32(&mtk_topckgen->clk_misc_cfg_0, 1 << 4);
482 
483  /* switch ca55 clock source to intermediate clock */
485 
486  /* disable armpll_ll frequency output */
488 
489  /* raise armpll_ll frequency */
491 
492  /* enable armpll_ll frequency output */
495 
496  /* switch ca55 clock source back to armpll_ll */
498 
499  /* disable [4] intermediate clock armpll_divider_pll1_ck */
500  clrbits32(&mtk_topckgen->clk_misc_cfg_0, 1 << 4);
501 }
502 
504 {
505  u32 output, count, clk_dbg_cfg, clk_misc_cfg_0;
506 
507  /* backup */
508  clk_dbg_cfg = read32(&mtk_topckgen->clk_dbg_cfg);
509  clk_misc_cfg_0 = read32(&mtk_topckgen->clk_misc_cfg_0);
510 
511  /* set up frequency meter */
512  if (type == FMETER_ABIST) {
513  SET32_BITFIELDS(&mtk_topckgen->clk_dbg_cfg,
514  CLK_DBG_CFG_ABIST_CK_SEL, id,
515  CLK_DBG_CFG_CKGEN_CK_SEL, 0,
516  CLK_DBG_CFG_METER_CK_SEL, 0);
517  SET32_BITFIELDS(&mtk_topckgen->clk_misc_cfg_0,
518  CLK_MISC_CFG_0_METER_DIV, 1);
519  } else if (type == FMETER_CKGEN) {
520  SET32_BITFIELDS(&mtk_topckgen->clk_dbg_cfg,
521  CLK_DBG_CFG_ABIST_CK_SEL, 0,
522  CLK_DBG_CFG_CKGEN_CK_SEL, id,
523  CLK_DBG_CFG_METER_CK_SEL, 1);
524  SET32_BITFIELDS(&mtk_topckgen->clk_misc_cfg_0,
525  CLK_MISC_CFG_0_METER_DIV, 0);
526  } else {
527  die("unsupported fmeter type\n");
528  }
529 
530  /* enable frequency meter */
531  write32(&mtk_topckgen->clk26cali_0, 0x1000);
532 
533  /* set load count = 1024-1 */
534  SET32_BITFIELDS(&mtk_topckgen->clk26cali_1, CLK26CALI_1_LOAD_CNT, 0x3ff);
535 
536  /* trigger frequency meter */
537  SET32_BITFIELDS(&mtk_topckgen->clk26cali_0, CLK26CALI_0_TRIGGER, 1);
538 
539  /* wait frequency meter until finished */
540  if (wait_us(200, !READ32_BITFIELD(&mtk_topckgen->clk26cali_0, CLK26CALI_0_TRIGGER))) {
541  count = read32(&mtk_topckgen->clk26cali_1) & 0xffff;
542  output = (count * 26000) / 1024; /* KHz */
543  } else {
544  printk(BIOS_WARNING, "fmeter timeout\n");
545  output = 0;
546  }
547 
548  /* disable frequency meter */
549  write32(&mtk_topckgen->clk26cali_0, 0x0000);
550 
551  /* restore */
552  write32(&mtk_topckgen->clk_dbg_cfg, clk_dbg_cfg);
553  write32(&mtk_topckgen->clk_misc_cfg_0, clk_misc_cfg_0);
554 
555  if (type == FMETER_ABIST)
556  return output * 2;
557  else if (type == FMETER_CKGEN)
558  return output;
559 
560  return 0;
561 }
562 
564 {
565  /* enable [4] intermediate clock armpll_divider_pll1_ck */
566  setbits32(&mtk_topckgen->clk_misc_cfg_0, 1 << 4);
567 
568  /* switch cci clock source to intermediate clock */
570 
571  /* disable ccipll frequency output */
573 
574  /* raise ccipll frequency */
576 
577  /* enable ccipll frequency output */
580 
581  /* switch cci clock source back to ccipll */
583 
584  /* disable [4] intermediate clock armpll_divider_pll1_ck */
585  clrbits32(&mtk_topckgen->clk_misc_cfg_0, 1 << 4);
586 }
static void write32(void *addr, uint32_t val)
Definition: mmio.h:40
static uint32_t read32(const void *addr)
Definition: mmio.h:22
#define ARRAY_SIZE(a)
Definition: helpers.h:12
#define MHz
Definition: helpers.h:80
#define KHz
Definition: helpers.h:79
void mux_set_sel(const struct mux *mux, u32 sel)
Definition: pll.c:8
int pll_set_rate(const struct pll *pll, u32 rate)
Definition: pll.c:72
#define printk(level,...)
Definition: stdlib.h:16
void __noreturn die(const char *fmt,...)
Definition: die.c:17
#define setbits32(addr, set)
Definition: mmio.h:21
#define SET32_BITFIELDS(addr,...)
Definition: mmio.h:201
#define READ32_BITFIELD(addr, name)
Definition: mmio.h:207
#define clrsetbits32(addr, clear, set)
Definition: mmio.h:16
#define clrbits32(addr, clear)
Definition: mmio.h:26
#define wait_us(timeout_us, condition)
Definition: timer.h:198
unsigned int type
Definition: edid.c:57
#define BIOS_WARNING
BIOS_WARNING - Bad configuration.
Definition: loglevel.h:86
@ UNIVPLL_HZ
Definition: pll.h:196
@ MSDCPLL_HZ
Definition: pll.h:198
@ MAINPLL_HZ
Definition: pll.h:195
@ MMPLL_HZ
Definition: pll.h:197
@ APLL1_HZ
Definition: pll.h:205
@ APLL2_HZ
Definition: pll.h:206
@ TVDPLL_HZ
Definition: pll.h:200
@ PLL_ISO_DELAY
Definition: pll.h:183
@ PLL_EN_DELAY
Definition: pll.h:184
@ PLL_PWR_ON_DELAY
Definition: pll.h:182
void mt_pll_init(void)
Definition: pll.c:289
pll_id
Definition: pll.c:172
@ APMIXED_MMPLL
Definition: pll.c:177
@ APMIXED_APLL1
Definition: pll.c:183
@ APMIXED_UNIVPLL
Definition: pll.c:176
@ APMIXED_APLL2
Definition: pll.c:184
@ APMIXED_MSDCPLL
Definition: pll.c:178
@ APMIXED_MAINPLL
Definition: pll.c:175
@ APMIXED_TVDPLL
Definition: pll.c:180
const u32 pll_div_rate[]
Definition: pll.c:190
void mt_pll_raise_little_cpu_freq(u32 freq)
Definition: pll.c:420
void pll_set_pcw_change(const struct pll *pll)
Definition: pll.c:284
mux_id
Definition: pll.c:11
@ TOP_VENC_SEL
Definition: pll.c:18
@ TOP_AUD_1_SEL
Definition: pll.c:39
@ TOP_MSDC50_0_SEL
Definition: pll.c:26
@ TOP_AUD_INTBUS_SEL
Definition: pll.c:31
@ TOP_UART_SEL
Definition: pll.c:21
@ TOP_ATB_SEL
Definition: pll.c:34
@ TOP_SCAM_SEL
Definition: pll.c:43
@ TOP_MSDC30_1_SEL
Definition: pll.c:27
@ TOP_MSDC30_2_SEL
Definition: pll.c:28
@ TOP_SCP_SEL
Definition: pll.c:33
@ TOP_AUDIO_SEL
Definition: pll.c:30
@ TOP_CAMTG_SEL
Definition: pll.c:20
@ TOP_SPI_SEL
Definition: pll.c:22
@ TOP_AUD_2_SEL
Definition: pll.c:40
@ TOP_PWM_SEL
Definition: pll.c:16
@ TOP_NR_MUX
Definition: pll.c:51
@ TOP_VDEC_SEL
Definition: pll.c:17
@ TOP_AXI_SEL
Definition: pll.c:12
@ CCIPLL_HZ
Definition: pll.h:237
@ MFGPLL_HZ
Definition: pll.h:242
@ ARMPLL_LL_HZ
Definition: pll.h:235
@ APMIXED_PLL_MAX
Definition: pll.c:198
@ APMIXED_ARMPLL_LL
Definition: pll.c:186
@ APMIXED_CCIPLL
Definition: pll.c:188
@ APMIXED_MFGPLL
Definition: pll.c:193
@ TOP_AES_UFSFDE_SEL
Definition: pll.c:50
@ TOP_SENINF_SEL
Definition: pll.c:46
@ TOP_AUD_ENGEN2_SEL
Definition: pll.c:49
@ TOP_PWRMCU_SEL
Definition: pll.c:37
@ TOP_CAMTG2_SEL
Definition: pll.c:23
@ TOP_DSP_SEL
Definition: pll.c:16
@ TOP_UFS_SEL
Definition: pll.c:51
@ TOP_I2C_SEL
Definition: pll.c:44
@ TOP_MSDC50_0_HCLK_SEL
Definition: pll.c:28
@ TOP_SSUSB_XHCI_SEL
Definition: pll.c:42
@ TOP_USB_TOP_SEL
Definition: pll.c:41
@ TOP_CAMTG4_SEL
Definition: pll.c:25
@ TOP_DISP_PWM_SEL
Definition: pll.c:40
@ TOP_CAMTG3_SEL
Definition: pll.c:24
@ TOP_AUD_ENGEN1_SEL
Definition: pll.c:48
@ TOP_SPM_SEL
Definition: pll.c:43
@ TOP_PWRAP_ULPOSC_SEL
Definition: pll.c:35
@ TOP_DXCC_SEL
Definition: pll.c:47
@ TOP_CAM_SEL
Definition: pll.c:15
static struct mt8186_mcucfg_regs *const mtk_mcucfg
Definition: mcucfg.h:945
@ ARMPLL_BL_HZ
Definition: pll.h:477
@ ADSPPLL_HZ
Definition: pll.h:485
@ PLL_DIV_EN
Definition: pll.h:457
@ MCU_MUX_MASK
Definition: pll.h:469
@ MCU_DIV_MASK
Definition: pll.h:466
@ MCU_DIV_1
Definition: pll.h:467
@ MCU_MUX_SRC_PLL
Definition: pll.h:470
@ APMIXED_ARMPLL_BL
Definition: pll.c:274
@ APMIXED_ADSPPLL
Definition: pll.c:282
u32 mt_fmeter_get_freq_khz(enum fmeter_type type, u32 id)
Definition: pll.c:519
void mt_pll_raise_cci_freq(u32 freq)
Definition: pll.c:500
@ TOP_DVFSRC_SEL
Definition: pll.c:68
@ TOP_SENINF1_SEL
Definition: pll.c:50
@ TOP_CAMTM_SEL
Definition: pll.c:55
@ TOP_SENINF3_SEL
Definition: pll.c:52
@ TOP_DISP_SEL
Definition: pll.c:62
@ TOP_IPE_SEL
Definition: pll.c:59
@ TOP_IMG1_SEL
Definition: pll.c:58
@ TOP_AES_MSDCFDE_SEL
Definition: pll.c:53
@ TOP_DPI_SEL
Definition: pll.c:80
@ TOP_CAMTG6_SEL
Definition: pll.c:29
@ TOP_SPMI_MST_SEL
Definition: pll.c:70
@ TOP_MDP_SEL
Definition: pll.c:63
@ TOP_SENINF2_SEL
Definition: pll.c:51
@ TOP_CAMTG5_SEL
Definition: pll.c:28
@ TOP_AUDIO_H_SEL
Definition: pll.c:64
static struct mt8192_infracfg_regs *const mt8192_infracfg
Definition: infracfg.h:416
@ USBPLL_EN
Definition: pll.h:247
@ MCU_MUX_SRC_DIV_PLL1
Definition: pll.h:258
@ INFRACFG_AO_AXIMEM_BUS_DCM_REG0_MASK
Definition: pll.h:321
@ INFRACFG_AO_PERI_BUS_DCM_REG0_ON
Definition: pll.h:356
@ INFRACFG_AO_INFRA_CONN_BUS_DCM_REG1_MASK
Definition: pll.h:343
@ INFRACFG_AO_PERI_BUS_DCM_REG0_MASK
Definition: pll.h:347
@ INFRACFG_AO_INFRA_RX_P2P_DCM_REG0_ON
Definition: pll.h:346
@ INFRACFG_AO_INFRA_CONN_BUS_DCM_REG0_MASK
Definition: pll.h:341
@ INFRACFG_AO_INFRA_CONN_BUS_DCM_REG0_ON
Definition: pll.h:342
@ INFRACFG_AO_INFRA_BUS_DCM_REG0_MASK
Definition: pll.h:323
@ INFRACFG_AO_INFRA_RX_P2P_DCM_REG0_MASK
Definition: pll.h:345
@ INFRACFG_AO_PERI_MODULE_DCM_REG0_MASK
Definition: pll.h:365
@ INFRACFG_AO_AXIMEM_BUS_DCM_REG0_ON
Definition: pll.h:322
@ INFRACFG_AO_INFRA_BUS_DCM_REG0_ON
Definition: pll.h:332
@ INFRACFG_AO_PERI_MODULE_DCM_REG0_ON
Definition: pll.h:366
@ INFRACFG_AO_INFRA_CONN_BUS_DCM_REG1_ON
Definition: pll.h:344
@ USBPLL_HZ
Definition: pll.h:278
static const struct mux muxes[]
Definition: pll.c:104
@ APMIXED_USBPLL
Definition: pll.c:281
static const struct rate rates[]
Definition: pll.c:348
#define MUX_UPD(_id, _reg, _mux_shift, _mux_width, _upd_reg, _upd_shift)
Definition: pll.c:93
#define MUX(_id, _reg, _mux_shift, _mux_width)
Definition: pll.c:82
static const struct mux_sel mux_sels[]
Definition: pll.c:192
static const struct pll plls[]
Definition: pll.c:301
@ TOP_MFG_PLL_SEL
Definition: pll.c:31
@ TOP_MFG_REF_SEL
Definition: pll.c:30
@ TOP_DPE_SEL
Definition: pll.c:25
@ TOP_MCUPM_SEL
Definition: pll.c:77
@ TOP_DPMAIF_MAIN_SEL
Definition: pll.c:68
@ TOP_SFLASH_SEL
Definition: pll.c:78
@ TOP_DSP7_SEL
Definition: pll.c:29
@ TOP_ADSP_SEL
Definition: pll.c:67
@ TOP_TL_SEL
Definition: pll.c:59
@ TOP_BUS_AXIMEM_SEL
Definition: pll.c:19
@ TOP_CCU_SEL
Definition: pll.c:27
@ TOP_IMG2_SEL
Definition: pll.c:23
#define PLL_PCW_CHG
Definition: pll_common.h:19
#define NO_RSTB_SHIFT
Definition: pll_common.h:18
#define PLL_PWR_ON
Definition: pll_common.h:14
#define PLL_EN
Definition: pll_common.h:15
#define mtk_topckgen
Definition: pll_common.h:11
#define PLL(_id, _reg, _pwr_reg, _rstb, _pcwbits, _div_reg, _div_shift, _pcw_reg, _pcw_shift, _div_rate)
Definition: pll_common.h:44
#define PLL_ISO
Definition: pll_common.h:16
#define mtk_apmixed
Definition: pll_common.h:12
fmeter_type
Definition: pll_common.h:76
@ FMETER_CKGEN
Definition: pll_common.h:78
@ FMETER_ABIST
Definition: pll_common.h:77
uint32_t u32
Definition: stdint.h:51
Definition: dw_i2c.c:39
u32 infra_aximem_idle_bit_en_0
Definition: infracfg.h:254
u32 wdt_swsysrst
Definition: wdt.h:16
Definition: pll.c:115
u32 sel
Definition: pll.c:117
enum mux_id id
Definition: pll.c:116
Definition: pll_common.h:22
Definition: pll_common.h:32
void * div_reg
Definition: pll_common.h:35
Definition: pll.c:262
u32 rate
Definition: pll.c:264
enum pll_id id
Definition: pll.c:263
void udelay(uint32_t us)
Definition: udelay.c:15
#define count
static struct mtk_wdt_regs *const mtk_wdt
Definition: wdt.h:41