/* * Copyright (c) 2016, ARM Limited and Contributors. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright notice, this * list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * Neither the name of ARM nor the names of its contributors may be used * to endorse or promote products derived from this software without specific * prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include "sunxi_private.h" static void mmio_clrsetbits32(uintptr_t addr, uint32_t mask, uint32_t bits) { uint32_t regval = mmio_read_32(addr); regval &= ~mask; regval |= bits; mmio_write_32(addr, regval); } static void mmio_setbits32(uintptr_t addr, uint32_t bits) { uint32_t regval = mmio_read_32(addr); regval |= bits; mmio_write_32(addr, regval); } /* TODO (prt): we should have a timeout and return an error/success... */ static int pll_wait_until_stable(uintptr_t addr) { while ((mmio_read_32(addr) & PLL_STABLE_BIT) != PLL_STABLE_BIT) { /* spin */ } return 0; } static void get_pll1_factors(unsigned int *freq_mhz, uint8_t *n, uint8_t *k, uint8_t *m) { /* * We set M=1 (i.e. we only allow parent_rate steps). * The reason for that is, that we've observed stability issues * if M gets higher. Of course it makes the calculation faster as well. */ const uint8_t N_max = 32; /* 5 bits */ const unsigned int step_mhz = 24; /* MHz */ uint8_t N, K; /* multiplicators (e.g. N = n+1) */ /* We try to keep K as low as possible. */ if (*freq_mhz < step_mhz*(N_max*1-1)) K = 1; else if (*freq_mhz < step_mhz*(N_max*2-1)) K = 2; else if (*freq_mhz < step_mhz*(N_max*3-1)) K = 3; else K = 4; N = *freq_mhz / (step_mhz * K); *freq_mhz = (step_mhz * N * K); *n = N-1; *k = K-1; *m = 0; } void sun50i_set_cpu_pll(unsigned int freq_mhz) { uint8_t n, k, m; uint32_t reg; /* Set up dividers (suitable for the target clock frequency) and switch CPUX (and thus AXI & APB) to the LOSC24 clock */ mmio_write_32(CCMU_CPUX_AXI_CFG_REG, ( CPUX_SRCSEL_OSC24M | APB_CLKDIV(4) | AXI_CLKDIV(3) )); udelay(20); /* Calculate the factors for the target frequency. */ get_pll1_factors(&freq_mhz, &n, &k, &m); /* Build up the register value without the enable bit. */ reg = (n<<8) | (k<<4) | m; /* Set to the target frequency, but don't enable yet. */ mmio_write_32(CCMU_PLL_CPUX_CTRL_REG, reg); /* Enable PLL_CPUX again */ mmio_setbits32(CCMU_PLL_CPUX_CTRL_REG, PLL_ENABLE_BIT); /* Wait until the PLL_CPUX becomes stable */ pll_wait_until_stable(CCMU_PLL_CPUX_CTRL_REG); /* Wait another 20us, because Allwinner does so... */ udelay(20); /* Switch AXI clock back to PLL_CPUX, dividers have already been set up */ mmio_clrsetbits32(CCMU_CPUX_AXI_CFG_REG, CPUX_SRCSEL_MASK, CPUX_SRCSEL_PLLCPUX); /* Wait 1000us, because Allwiner does so... */ udelay(1000); } unsigned int sun50i_get_cpu_pll(void) { uint8_t k, m, n, p; uint32_t reg; /* Read out the PLL CPUX control register */ reg = mmio_read_32(CCMU_PLL_CPUX_CTRL_REG); /* Parse the PLL factors */ p = (reg>>16) & 0x3; n = (reg>>8) & 0xff; k = (reg>>4) & 0x3; m = reg & 0x3; /* Transform p to a divider */ if (p == 0) p = 1; else if (p == 1) p = 2; else p = 4; /* Return the PLL frequency in MHz. */ return 24*(n+1)*(k+1)/((m+1)*p); } int sunxi_setup_clocks(uint16_t socid) { uint32_t reg; /* Avoid reprogramming PERIPH0 if not necessary */ reg = mmio_read_32(CCMU_PLL_PERIPH0_CTRL_REG); if ((reg & 0x0fffffff) != 0x41811) /* is not at 600 MHz? */ mmio_write_32(CCMU_PLL_PERIPH0_CTRL_REG, 0x80041811); /* Set CPU speed to 816MHz */ sun50i_set_cpu_pll(816); /* AHB1 = PERIPH0 / (3 * 1) = 200MHz, APB1 = AHB1 / 2 */ mmio_write_32(CCMU_AHB1_APB1_CFG_REG, 0x00003180); mmio_write_32(CCMU_APB2_CFG_GREG, 0x01000000); /* APB2 => 24 MHz */ mmio_write_32(CCMU_AHB2_CFG_GREG, 0x00000001); /* AHB2 => 300 MHz */ return 0; }