mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
83 lines
2.1 KiB
C
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;
|
||
|
}
|
||
|
}
|