/* * Copyright (C) 2020 Koen Zandberg * * This file is subject to the terms and conditions of the GNU Lesser General * Public License v2.1. See the file LICENSE in the top level directory for more * details. */ /** * @ingroup cpu_gd32v * @{ * * @file * @brief GD32V Clock initialization * * @author Koen Zandberg * * Heavily based on the stm32f1 connectivity line clock initialization */ #include "kernel_defines.h" #include "irq.h" #include "cpu.h" #include "periph_conf.h" #define CLOCK_AHB_DIV 0 /* Max speed at 108 MHz */ #define CLOCK_APB1_DIV (0x04 | 0) /* Max speed at 54 MHz */ #define CLOCK_APB2_DIV (0x0 | 0) /* Max speed at 108 MHz */ #define CLOCK_AHB_DIV_CONF (CLOCK_AHB_DIV << RCU_CFG0_AHBPSC_Pos) #define CLOCK_APB1_DIV_CONF (CLOCK_APB1_DIV << RCU_CFG0_APB1PSC_Pos) #define CLOCK_APB2_DIV_CONF (CLOCK_APB2_DIV << RCU_CFG0_APB2PSC_Pos) #define PREDV0_CONF 1 /* Divide by 2 */ #ifdef CONFIG_BOARD_HAS_HXTAL #define PLL_MULT_FACTOR (CLOCK_CORECLOCK / \ (CONFIG_CLOCK_HXTAL / (PREDV0_CONF + 1)) - 1) #else #define PLL_MULT_FACTOR (CLOCK_CORECLOCK / (MHZ(8) / 2 ) - 1) #endif #define RCU_CFG0_SCS_IRC8 (0 << RCU_CFG0_SCS_Pos) #define RCU_CFG0_SCS_HXTAL (1 << RCU_CFG0_SCS_Pos) #define RCU_CFG0_SCS_PLL (2 << RCU_CFG0_SCS_Pos) #define ENABLE_DEBUG 0 #include "debug.h" void periph_clk_en(bus_t bus, uint32_t mask) { switch (bus) { case AHB: cpu_reg_enable_bits(&RCU->AHBEN, mask); break; case APB1: cpu_reg_enable_bits(&RCU->APB1EN, mask); break; case APB2: cpu_reg_enable_bits(&RCU->APB2EN, mask); break; default: DEBUG("unsupported bus %d\n", (int)bus); break; } } void periph_clk_dis(bus_t bus, uint32_t mask) { switch (bus) { case AHB: cpu_reg_disable_bits(&RCU->AHBEN, mask); break; case APB1: cpu_reg_disable_bits(&RCU->APB1EN, mask); break; case APB2: cpu_reg_disable_bits(&RCU->APB2EN, mask); break; default: DEBUG("unsupported bus %d\n", (int)bus); break; } } uint32_t periph_apb_clk(bus_t bus) { switch (bus) { case AHB: return CLOCK_AHB; case APB1: return CLOCK_APB1; case APB2: return CLOCK_APB2; } return 0; } void gd32v_enable_irc8(void) { cpu_reg_enable_bits(&RCU->CTL, RCU_CTL_IRC8MEN_Msk); while (!(RCU->CTL & RCU_CTL_IRC8MSTB_Msk)) {} } void gd32v_disable_irc8(void) { RCU->CTL &= ~(RCU_CTL_IRC8MEN_Msk); } void gd32vf103_clock_init(void) { unsigned is = irq_disable(); /* enable HSI clock for the duration of initialization */ gd32v_enable_irc8(); /* use HSI as system clock while we do any further configuration and * configure the AHB and APB clock dividers as configure by the board */ RCU->CFG0 = (RCU_CFG0_SCS_IRC8 | CLOCK_AHB_DIV_CONF | CLOCK_APB1_DIV_CONF | CLOCK_APB2_DIV_CONF); while ((RCU->CFG0 & RCU_CFG0_SCSS_Msk) != (RCU_CFG0_SCS_IRC8 << RCU_CFG0_SCSS_Pos)) {} /* disable all active clocks except IRC8 -> resets the clk configuration */ RCU->CTL &= (RCU_CTL_IRC8MCALIB_Msk | RCU_CTL_IRC8MADJ_Msk); RCU->CTL |= RCU_CTL_IRC8MEN_Msk; if (IS_ACTIVE(CONFIG_BOARD_HAS_HXTAL)) { /* if the board has an HXTAL, HXTAL is used as PLL input and PREDEV0 is set */ cpu_reg_enable_bits(&RCU->CTL, RCU_CTL_HXTALEN_Msk); while (!(RCU->CTL & RCU_CTL_HXTALSTB_Msk)) {} RCU->CFG1 = PREDV0_CONF; RCU->CFG0 |= RCU_CFG0_PLLSEL_Msk; } else { /* if the board doesn't have HXTAL, IRCM8/2 is used as PLL input */ RCU->CFG0 &= ~RCU_CFG0_PLLSEL_Msk; } RCU->CFG0 |= ((PLL_MULT_FACTOR & 0xf) << RCU_CFG0_PLLMF_3_0_Pos) | ((PLL_MULT_FACTOR & 0x10) << (RCU_CFG0_PLLMF_4_Pos - 4)); RCU->CTL |= RCU_CTL_PLLEN_Msk; /* Wait for PLL to stabilize */ while ((RCU->CTL & RCU_CTL_PLLSTB_Msk) != RCU_CTL_PLLSTB_Msk) {} /* Switch clock input */ RCU->CFG0 |= RCU_CFG0_SCS_PLL; RCU->AHBEN &= ~RCU_AHBEN_FMCSPEN_Msk; while ((RCU->CFG0 & RCU_CFG0_SCSS_Msk) != (RCU_CFG0_SCS_PLL << RCU_CFG0_SCSS_Pos)) {} if (IS_ACTIVE(CONFIG_BOARD_HAS_HXTAL)) { /* disable IRCM8 clock if HXTAL is used */ gd32v_disable_irc8(); } irq_restore(is); }