2020-04-30 18:30:31 +02:00
|
|
|
/*
|
|
|
|
* 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 "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 */
|
2020-10-12 21:41:16 +02:00
|
|
|
while (!_hf_source_ready()) {}
|
2020-04-30 18:30:31 +02:00
|
|
|
|
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
}
|