1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/cpu/cc26x2_cc13x2/osc.c
Jean Pierre Dudey a66c693ad5
cc26x2_cc13x2: add oscillator switching functions
Signed-off-by: Jean Pierre Dudey <jeandudey@hotmail.com>
2020-05-02 13:25:41 -05:00

83 lines
2.1 KiB
C

/*
* Copyright (C) 2020 Locha Inc
*
* 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_cc26x2_cc13x2
* @{
*
* @file
* @brief CC26x2/CC13x2 Oscillator functions
*
* @author Jean Pierre Dudey <jeandudey@hotmail.com>
* @}
*/
#include <assert.h>
#include "cpu.h"
static inline bool _hf_source_ready(void)
{
if (DDI_0_OSC->STAT0 & DDI_0_OSC_STAT0_PENDINGSCLKHFSWITCHING) {
return true;
}
return false;
}
static inline uint32_t _hf_source_get(void)
{
return (DDI_0_OSC->STAT0 & DDI_0_OSC_STAT0_SCLK_HF_SRC_m) >>
DDI_0_OSC_STAT0_SCLK_HF_SRC_s;
}
static inline void _hf_source_set(uint32_t osc)
{
uint32_t mask = DDI_0_OSC_CTL0_SCLK_HF_SRC_SEL_m;
uint32_t ctl = osc << DDI_0_OSC_CTL0_SCLK_HF_SRC_SEL_s;
/* Use a 16-bit masked write, target bits are on the lower 16-bit
* half */
DDI_0_OSC_M16->CTL0.LOW = (mask << 16) | ctl;
}
void osc_hf_source_switch(uint32_t osc)
{
if (_hf_source_get() == osc) {
return;
}
/* Request oscillator change */
_hf_source_set(osc);
/* Wait for the oscillator to be ready */
while (_hf_source_ready()) {}
/* If target clock source is RCOSC, change clock source for DCDC to RCOSC */
if (osc == OSC_RCOSC_HF) {
/* Force DCDC to use RCOSC before switching SCLK_HF to RCOSC */
uint32_t mask = DDI_0_OSC_CTL0_CLK_DCDC_SRC_SEL_m;
uint32_t data = DDI_0_OSC_CTL0_CLK_DCDC_SRC_SEL_m >> 16;
DDI_0_OSC_M16->CTL0.HIGH = mask | data;
/* Dummy read to ensure that the write has propagated */
DDI_0_OSC->CTL0;
}
/* Switch the HF clock source */
rom_hapi_hf_source_safe_switch();
/* If target clock source is XOSC, change clock source for DCDC to "auto" */
if (osc == OSC_XOSC_HF) {
/* Set DCDC clock source back to "auto" after SCLK_HF was switched to
* XOSC */
uint32_t mask = DDI_0_OSC_CTL0_CLK_DCDC_SRC_SEL_m;
uint32_t data = 0;
DDI_0_OSC_M16->CTL0.HIGH = mask | data;
}
}