| /* SPDX-License-Identifier: (GPL-2.0+ OR MIT) */ |
| /* |
| * arch/arm/cpu/armv8/axg/clock.c |
| * |
| * Copyright (C) 2020 Amlogic, Inc. All rights reserved. |
| * |
| */ |
| |
| #include <config.h> |
| #include <common.h> |
| #include <asm/arch/io.h> |
| #include <asm/arch/secure_apb.h> |
| #include <asm/arch/clock.h> |
| |
| /* !!must use nonzero value as default value. Otherwise it linked to bss segment, |
| but this segment not cleared to zero while running "board_init_f" */ |
| #define CLK_UNKNOWN (0xffffffff) |
| |
| __u32 get_clk81(void) |
| { |
| static __u32 clk81_freq=CLK_UNKNOWN; |
| if (clk81_freq == CLK_UNKNOWN) |
| { |
| clk81_freq=(clk_util_clk_msr(CLK81)*1000000); |
| } |
| return clk81_freq; |
| } |
| |
| struct __clk_rate{ |
| unsigned clksrc; |
| __u32 (*get_rate)(void); |
| }; |
| |
| struct __clk_rate clkrate[]={ |
| { |
| .clksrc=CLK81, |
| .get_rate=get_clk81, |
| }, |
| }; |
| |
| int clk_get_rate(unsigned clksrc) |
| { |
| int i; |
| for (i = 0; i < sizeof(clkrate)/sizeof(clkrate[0]); i++) |
| { |
| if (clksrc == clkrate[i].clksrc) |
| return clkrate[i].get_rate(); |
| } |
| return -1; |
| } |
| |
| const char* clk_table[] = { |
| [109] = "audio_locker_in ", |
| [108] = "audio_locker_out ", |
| [107] = "pcie_refclk_p ", |
| [106] = "pcie_refclk_n ", |
| [105] = "audio_mclk_a ", |
| [104] = "audio_mclk_b ", |
| [103] = "audio_mclk_c ", |
| [102] = "audio_mclk_d ", |
| [101] = "audio_mclk_e ", |
| [100] = "audio_mclk_f ", |
| [99] = "audio_sclk_a ", |
| [98] = "audio_sclk_b ", |
| [97] = "audio_sclk_c ", |
| [96] = "audio_sclk_d ", |
| [95] = "audio_sclk_e ", |
| [94] = "audio_sclk_f ", |
| [93] = "audio_lrclk_a ", |
| [92] = "audio_lrclk_b ", |
| [91] = "audio_lrclk_c ", |
| [90] = "audio_lrclk_d ", |
| [89] = "audio_lrclk_e ", |
| [88] = "audio_lrclk_f ", |
| [87] = "audio_spdifint_clk ", |
| [86] = "audio_spdifout_clk ", |
| [85] = "audio_pdm_sysclk ", |
| [84] = "audio_resample_clk ", |
| [83] = "0 ", |
| [82] = "Cts_ge2d_clk ", |
| [81] = "Cts_vapbclk ", |
| [80] = "Rng_ring_osc_clk[3]", |
| [79] = "Rng_ring_osc_clk[2]", |
| [78] = "Rng_ring_osc_clk[1]", |
| [77] = "Rng_ring_osc_clk[0]", |
| [76] = "tdmin_lb_sclk ", |
| [75] = "tdmin_lb_lrclk ", |
| [74] = "wifi_beacon ", |
| [73] = "cts_pwm_C_clk ", |
| [72] = "cts_pwm_D_clk ", |
| [71] = "audio_slv_sclk_a ", |
| [70] = "audio_slv_sclk_b ", |
| [69] = "audio_slv_sclk_c ", |
| [68] = "audio_slv_lrclk_a ", |
| [67] = "audio_slv_lrclk_b ", |
| [66] = "audio_slv_lrclk_c ", |
| [65] = "0 ", |
| [64] = "0 ", |
| [63] = "0 ", |
| [62] = "0 ", |
| [61] = "gpio_clk_msr ", |
| [60] = "0 ", |
| [59] = "0 ", |
| [58] = "0 ", |
| [57] = "0 ", |
| [56] = "0 ", |
| [55] = "0 ", |
| [54] = "0 ", |
| [53] = "0 ", |
| [52] = "sd_emmc_clk_B ", |
| [51] = "sd_emmc_clk_C ", |
| [50] = "mp3_clk_out ", |
| [49] = "mp2_clk_out ", |
| [48] = "mp1_clk_out ", |
| [47] = "ddr_dpll_pt_clk ", |
| [46] = "cts_vpu_clk ", |
| [45] = "cts_pwm_A_clk ", |
| [44] = "cts_pwm_B_clk ", |
| [43] = "fclk_div5 ", |
| [42] = "mp0_clk_out ", |
| [41] = "mod_eth_rx_clk_rmii", |
| [40] = "mod_eth_tx_clk ", |
| [39] = "0 ", |
| [38] = "0 ", |
| [37] = "0 ", |
| [36] = "0 ", |
| [35] = "0 ", |
| [34] = "0 ", |
| [33] = "0 ", |
| [32] = "0 ", |
| [31] = "MPLL_CLK_TEST_OUT ", |
| [30] = "0 ", |
| [29] = "0 ", |
| [28] = "Cts_sar_adc_clk ", |
| [27] = "0 ", |
| [26] = "0 ", |
| [25] = "0 ", |
| [24] = "0 ", |
| [23] = "mmc_clk ", |
| [22] = "0 ", |
| [21] = "0 ", |
| [20] = "rtc_osc_clk_out ", |
| [19] = "0 ", |
| [18] = "sys_cpu_clk_div16 ", |
| [17] = "sys_pll_div16 ", |
| [16] = "0 ", |
| [15] = "0 ", |
| [14] = "0 ", |
| [13] = "0 ", |
| [12] = "0 ", |
| [11] = "0 ", |
| [10] = "0 ", |
| [9] = "0 ", |
| [8] = "0 ", |
| [7] = "clk81 ", |
| [6] = "0 ", |
| [5] = "gp1_pll_clk ", |
| [4] = "gp0_pll_clk ", |
| [3] = "A53_ring_osc_clk ", |
| [2] = "am_ring_osc_clk_out_ee[2]", |
| [1] = "am_ring_osc_clk_out_ee[1]", |
| [0] = "am_ring_osc_clk_out_ee[0]", |
| }; |
| |
| unsigned long clk_util_clk_msr(unsigned long clk_mux) |
| { |
| unsigned int regval = 0; |
| |
| WRITE_CBUS_REG(MSR_CLK_REG0, 0); |
| /* Set the measurement gate to 64uS */ |
| CLEAR_CBUS_REG_MASK(MSR_CLK_REG0, 0xffff); |
| /* 64uS is enough for measure the frequence? */ |
| SET_CBUS_REG_MASK(MSR_CLK_REG0, (64 - 1)); |
| /* Disable continuous measurement */ |
| /* Disable interrupts */ |
| CLEAR_CBUS_REG_MASK(MSR_CLK_REG0, ((1 << 18) | (1 << 17))); |
| CLEAR_CBUS_REG_MASK(MSR_CLK_REG0, (0x7f << 20)); |
| SET_CBUS_REG_MASK(MSR_CLK_REG0, (clk_mux << 20) | /* Select MUX */ |
| (1 << 19) | /* enable the clock */ |
| (1 << 16)); /* enable measuring */ |
| /* Wait for the measurement to be done */ |
| regval = READ_CBUS_REG(MSR_CLK_REG0); |
| do { |
| regval = READ_CBUS_REG(MSR_CLK_REG0); |
| } while (regval & (1 << 31)); |
| |
| /* Disable measuring */ |
| CLEAR_CBUS_REG_MASK(MSR_CLK_REG0, (1 << 16)); |
| regval = (READ_CBUS_REG(MSR_CLK_REG2) + 31) & 0x000FFFFF; |
| |
| return (regval >> 6); |
| } |
| |
| int clk_msr(int index) |
| { |
| unsigned int index_total = sizeof(clk_table) / sizeof(char *); |
| |
| int i; |
| if (index == 0xff) { |
| for (i = 0; i < index_total; i++) |
| printf("[%4d][%4ld MHz] %s\n", i, clk_util_clk_msr(i), clk_table[i]); |
| } |
| else { |
| if (index >= index_total) { |
| printf("clk msr legal range: [0-%d]\n", index_total-1); |
| return -1; |
| } |
| printf("[%4d][%4ld MHz] %s\n", index, clk_util_clk_msr(index), clk_table[index]); |
| } |
| |
| return 0; |
| } |
| |
| #if 0 |
| int axg_pcie_set(void) |
| { |
| int delay = 24000000; |
| unsigned int reg; |
| |
| #define BIT(nr) (1UL << (nr)) |
| #define MESON_PLL_RESET BIT(29) |
| #define MESON_PLL_LOCK BIT(31) |
| |
| #define AXG_MIPI_CNTL0_ENABLE BIT(29) |
| #define AXG_MIPI_CNTL0_BANDGAP BIT(26) |
| #define AXG_PCIE_PLL_CNTL 0x40010242 |
| #define AXG_PCIE_PLL_CNTL1 0xc084b2ab |
| #define AXG_PCIE_PLL_CNTL2 0xb75020be |
| #define AXG_PCIE_PLL_CNTL3 0x0a5aaa88 |
| #define AXG_PCIE_PLL_CNTL4 0xc000004d |
| #define AXG_PCIE_PLL_CNTL5 0x00078000 |
| #define AXG_PCIE_PLL_CNTL6 0x003303de |
| |
| /* setting AXG PCIE PLL 100M */ |
| reg = readl(HHI_MIPI_CNTL0); |
| reg |= (AXG_MIPI_CNTL0_ENABLE | AXG_MIPI_CNTL0_BANDGAP); |
| writel(reg, HHI_MIPI_CNTL0); |
| writel(AXG_PCIE_PLL_CNTL1, HHI_PCIE_PLL_CNTL1); |
| writel(AXG_PCIE_PLL_CNTL2, HHI_PCIE_PLL_CNTL2); |
| writel(AXG_PCIE_PLL_CNTL3, HHI_PCIE_PLL_CNTL3); |
| writel(AXG_PCIE_PLL_CNTL4, HHI_PCIE_PLL_CNTL4); |
| writel(AXG_PCIE_PLL_CNTL5, HHI_PCIE_PLL_CNTL5); |
| writel(AXG_PCIE_PLL_CNTL6, HHI_PCIE_PLL_CNTL6); |
| |
| /* PLL reset */ |
| writel(AXG_PCIE_PLL_CNTL | MESON_PLL_RESET, HHI_PCIE_PLL_CNTL); |
| udelay(10); |
| writel(readl(HHI_PCIE_PLL_CNTL) &(~ MESON_PLL_RESET), HHI_PCIE_PLL_CNTL); |
| |
| while (delay > 0) { |
| /* check lock bit */ |
| if (readl(HHI_PCIE_PLL_CNTL) & MESON_PLL_LOCK) |
| return 0; |
| delay--; |
| } |
| |
| return -1; |
| } |
| |
| void axg_pll_set(void) |
| { |
| int ret; |
| int count = 3; |
| |
| ret = axg_pcie_set(); |
| while (count--) { |
| if (!ret) { |
| printf("%s: pcie pll lock to 100M \n", |
| __func__); |
| break; |
| }else { |
| printf("%s: pcie pll did not lock, trying to set to 100M again\n", |
| __func__); |
| ret = axg_pcie_set(); |
| } |
| } |
| return; |
| } |
| #endif |
| |
| #ifdef CONFIG_AML_SPICC |
| /* generic clock control for spicc0. |
| * if deleted, you have to add it into all axg board files as necessary. |
| */ |
| int spicc0_clk_enable(bool enable) |
| { |
| u32 regv; |
| |
| regv = readl(P_HHI_GCLK_MPEG0); |
| if (enable) |
| regv |= 1 << 8; |
| else |
| regv &= ~ (1 << 8); |
| writel(regv, P_HHI_GCLK_MPEG0); |
| |
| return 0; |
| } |
| #endif /* CONFIG_AML_SPICC */ |