/* * Copyright (C) 2019 ML!PA Consulting GmbH * * 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_samd5x * @{ * * @file cpu.c * @brief Implementation of the CPU initialization for Microchip SAMD5x/SAME5x MCUs * * @author Benjamin Valentin * @} */ #include "cpu.h" #include "periph_conf.h" #include "periph/init.h" #include "stdio_base.h" #if CLOCK_CORECLOCK == 0 #error Please select CLOCK_CORECLOCK #endif /* use DFLL for low frequency operation */ #if CLOCK_CORECLOCK > SAM0_DFLL_FREQ_HZ #define USE_DPLL 1 #else #define USE_DPLL 0 #if (SAM0_DFLL_FREQ_HZ % CLOCK_CORECLOCK) #error For frequencies < 48 MHz, CLOCK_CORECLOCK must be a divider of 48 MHz #endif #endif /* If the CPU clock is lower than the minimal DPLL Freq set fDPLL = 2 * CLOCK_CORECLOCK */ #if USE_DPLL && (CLOCK_CORECLOCK < SAM0_DPLL_FREQ_MIN_HZ) #define DPLL_DIV 2 #else #define DPLL_DIV 1 #endif static void xosc32k_init(void) { OSC32KCTRL->XOSC32K.reg = OSC32KCTRL_XOSC32K_ENABLE | OSC32KCTRL_XOSC32K_EN1K | OSC32KCTRL_XOSC32K_EN32K | OSC32KCTRL_XOSC32K_RUNSTDBY | OSC32KCTRL_XOSC32K_XTALEN | OSC32KCTRL_XOSC32K_STARTUP(7); while (!OSC32KCTRL->STATUS.bit.XOSC32KRDY) {} } static void dfll_init(void) { uint32_t reg = OSCCTRL_DFLLCTRLB_QLDIS #ifdef OSCCTRL_DFLLCTRLB_WAITLOCK | OSCCTRL_DFLLCTRLB_WAITLOCK #endif ; /* workaround for Errata 2.8.3 DFLLVAL.FINE Value When DFLL48M Re-enabled */ OSCCTRL->DFLLMUL.reg = 0; /* Write new DFLLMULL configuration */ OSCCTRL->DFLLCTRLB.reg = 0; /* Select Open loop configuration */ OSCCTRL->DFLLCTRLA.bit.ENABLE = 1; /* Enable DFLL */ OSCCTRL->DFLLVAL.reg = OSCCTRL->DFLLVAL.reg; /* Reload DFLLVAL register */ OSCCTRL->DFLLCTRLB.reg = reg; /* Write final DFLL configuration */ OSCCTRL->DFLLCTRLA.reg = OSCCTRL_DFLLCTRLA_ENABLE; while (!OSCCTRL->STATUS.bit.DFLLRDY) {} } #if USE_DPLL static void fdpll0_init(uint32_t f_cpu) { /* We source the DPLL from 32kHz GCLK1 */ const uint32_t LDR = ((f_cpu << 5) / 32768); /* disable the DPLL before changing the configuration */ OSCCTRL->Dpll[0].DPLLCTRLA.bit.ENABLE = 0; while (OSCCTRL->Dpll[0].DPLLSYNCBUSY.reg) {} /* set DPLL clock source */ GCLK->PCHCTRL[OSCCTRL_GCLK_ID_FDPLL0].reg = GCLK_PCHCTRL_GEN(1) | GCLK_PCHCTRL_CHEN; while (!(GCLK->PCHCTRL[OSCCTRL_GCLK_ID_FDPLL0].reg & GCLK_PCHCTRL_CHEN)) {} OSCCTRL->Dpll[0].DPLLRATIO.reg = OSCCTRL_DPLLRATIO_LDRFRAC(LDR & 0x1F) | OSCCTRL_DPLLRATIO_LDR((LDR >> 5) - 1); /* Without LBYPASS, startup takes very long, see errata section 2.13. */ OSCCTRL->Dpll[0].DPLLCTRLB.reg = OSCCTRL_DPLLCTRLB_REFCLK_GCLK | OSCCTRL_DPLLCTRLB_WUF | OSCCTRL_DPLLCTRLB_LBYPASS; OSCCTRL->Dpll[0].DPLLCTRLA.reg = OSCCTRL_DPLLCTRLA_ENABLE; while (OSCCTRL->Dpll[0].DPLLSYNCBUSY.reg) {} while (!(OSCCTRL->Dpll[0].DPLLSTATUS.bit.CLKRDY && OSCCTRL->Dpll[0].DPLLSTATUS.bit.LOCK)) {} } #endif static void gclk_connect(uint8_t id, uint8_t src, uint32_t flags) { GCLK->GENCTRL[id].reg = GCLK_GENCTRL_SRC(src) | GCLK_GENCTRL_GENEN | flags | GCLK_GENCTRL_IDC; while (GCLK->SYNCBUSY.reg & GCLK_SYNCBUSY_GENCTRL(id)) {} } void sam0_gclk_enable(uint8_t id) { /* clocks 0 & 1 are always running */ switch (id) { case SAM0_GCLK_8MHZ: /* 8 MHz clock used by xtimer */ #if USE_DPLL gclk_connect(SAM0_GCLK_8MHZ, GCLK_SOURCE_DPLL0, GCLK_GENCTRL_DIV(DPLL_DIV * CLOCK_CORECLOCK / 8000000)); #else gclk_connect(SAM0_GCLK_8MHZ, GCLK_SOURCE_DFLL, GCLK_GENCTRL_DIV(SAM0_DFLL_FREQ_HZ / 8000000)); #endif break; case SAM0_GCLK_48MHZ: gclk_connect(SAM0_GCLK_48MHZ, GCLK_SOURCE_DFLL, 0); break; } } uint32_t sam0_gclk_freq(uint8_t id) { switch (id) { case SAM0_GCLK_MAIN: return CLOCK_CORECLOCK; case SAM0_GCLK_32KHZ: return 32768; case SAM0_GCLK_8MHZ: return 8000000; case SAM0_GCLK_48MHZ: return SAM0_DFLL_FREQ_HZ; default: return 0; } } void cpu_pm_cb_enter(int deep) { (void) deep; /* will be called before entering sleep */ } void cpu_pm_cb_leave(int deep) { /* will be called after wake-up */ if (deep) { /* DFLL needs to be re-initialized to work around errata 2.8.3 */ dfll_init(); } } /** * @brief Initialize the CPU, set IRQ priorities, clocks */ void cpu_init(void) { /* Disable the RTC module to prevent synchronization issues during CPU init if the RTC was running from a previous boot (e.g wakeup from backup) */ if (RTC->MODE2.CTRLA.bit.ENABLE) { while (RTC->MODE2.SYNCBUSY.reg) {} RTC->MODE2.CTRLA.bit.ENABLE = 0; while (RTC->MODE2.SYNCBUSY.reg) {} } /* initialize the Cortex-M core */ cortexm_init(); /* turn on only needed APB peripherals */ MCLK->APBAMASK.reg = MCLK_APBAMASK_MCLK | MCLK_APBAMASK_OSCCTRL | MCLK_APBAMASK_OSC32KCTRL | MCLK_APBAMASK_GCLK | MCLK_APBAMASK_SUPC | MCLK_APBAMASK_PAC #ifdef MODULE_PERIPH_PM | MCLK_APBAMASK_PM #endif #ifdef MODULE_PERIPH_GPIO_IRQ | MCLK_APBAMASK_EIC #endif ; MCLK->APBBMASK.reg = 0 #ifdef MODULE_PERIPH_FLASHPAGE | MCLK_APBBMASK_NVMCTRL #endif #ifdef MODULE_PERIPH_GPIO | MCLK_APBBMASK_PORT #endif ; MCLK->APBCMASK.reg = 0; MCLK->APBDMASK.reg = 0; /* enable the Cortex M Cache Controller */ CMCC->CTRL.bit.CEN = 1; xosc32k_init(); gclk_connect(SAM0_GCLK_32KHZ, GCLK_SOURCE_XOSC32K, 0); /* make sure main clock is not sourced from DPLL */ dfll_init(); gclk_connect(SAM0_GCLK_MAIN, GCLK_SOURCE_DFLL, 0); #if USE_DPLL fdpll0_init(CLOCK_CORECLOCK * DPLL_DIV); /* source main clock from DPLL */ gclk_connect(SAM0_GCLK_MAIN, GCLK_SOURCE_DPLL0, GCLK_GENCTRL_DIV(DPLL_DIV)); #else gclk_connect(SAM0_GCLK_MAIN, GCLK_SOURCE_DFLL, GCLK_GENCTRL_DIV(SAM0_DFLL_FREQ_HZ / CLOCK_CORECLOCK)); #endif /* initialize stdio prior to periph_init() to allow use of DEBUG() there */ stdio_init(); /* trigger static peripheral initialization */ periph_init(); /* set ONDEMAND bit after all clocks have been configured */ /* This is to avoid setting the source for the main clock to ONDEMAND before using it. */ OSCCTRL->Dpll[0].DPLLCTRLA.reg |= OSCCTRL_DPLLCTRLA_ONDEMAND; }