mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2024-12-29 04:50:03 +01:00
469 lines
19 KiB
C
469 lines
19 KiB
C
|
/******************************************************************************
|
||
|
* Filename: osc.c
|
||
|
* Revised: 2019-02-14 09:35:31 +0100 (Thu, 14 Feb 2019)
|
||
|
* Revision: 54539
|
||
|
*
|
||
|
* Description: Driver for setting up the system Oscillators
|
||
|
*
|
||
|
* Copyright (c) 2015 - 2017, Texas Instruments Incorporated
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
*
|
||
|
* 1) Redistributions of source code must retain the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer.
|
||
|
*
|
||
|
* 2) Redistributions in binary form must reproduce the above copyright notice,
|
||
|
* this list of conditions and the following disclaimer in the documentation
|
||
|
* and/or other materials provided with the distribution.
|
||
|
*
|
||
|
* 3) Neither the name of the ORGANIZATION nor the names of its contributors may
|
||
|
* be used to endorse or promote products derived from this software without
|
||
|
* specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
||
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
||
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
||
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
||
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
||
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
||
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
||
|
* POSSIBILITY OF SUCH DAMAGE.
|
||
|
*
|
||
|
******************************************************************************/
|
||
|
|
||
|
#include "../inc/hw_types.h"
|
||
|
#include "../inc/hw_ccfg.h"
|
||
|
#include "../inc/hw_fcfg1.h"
|
||
|
#include "aon_batmon.h"
|
||
|
#include "aon_rtc.h"
|
||
|
#include "osc.h"
|
||
|
#include "setup_rom.h"
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Handle support for DriverLib in ROM:
|
||
|
// This section will undo prototype renaming made in the header file
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
#if !defined(DOXYGEN)
|
||
|
#undef OSCClockSourceSet
|
||
|
#define OSCClockSourceSet NOROM_OSCClockSourceSet
|
||
|
#undef OSCClockSourceGet
|
||
|
#define OSCClockSourceGet NOROM_OSCClockSourceGet
|
||
|
#undef OSCHF_GetStartupTime
|
||
|
#define OSCHF_GetStartupTime NOROM_OSCHF_GetStartupTime
|
||
|
#undef OSCHF_TurnOnXosc
|
||
|
#define OSCHF_TurnOnXosc NOROM_OSCHF_TurnOnXosc
|
||
|
#undef OSCHF_AttemptToSwitchToXosc
|
||
|
#define OSCHF_AttemptToSwitchToXosc NOROM_OSCHF_AttemptToSwitchToXosc
|
||
|
#undef OSCHF_SwitchToRcOscTurnOffXosc
|
||
|
#define OSCHF_SwitchToRcOscTurnOffXosc NOROM_OSCHF_SwitchToRcOscTurnOffXosc
|
||
|
#undef OSCHF_DebugGetCrystalAmplitude
|
||
|
#define OSCHF_DebugGetCrystalAmplitude NOROM_OSCHF_DebugGetCrystalAmplitude
|
||
|
#undef OSCHF_DebugGetExpectedAverageCrystalAmplitude
|
||
|
#define OSCHF_DebugGetExpectedAverageCrystalAmplitude NOROM_OSCHF_DebugGetExpectedAverageCrystalAmplitude
|
||
|
#undef OSC_HPOSCRelativeFrequencyOffsetGet
|
||
|
#define OSC_HPOSCRelativeFrequencyOffsetGet NOROM_OSC_HPOSCRelativeFrequencyOffsetGet
|
||
|
#undef OSC_AdjustXoscHfCapArray
|
||
|
#define OSC_AdjustXoscHfCapArray NOROM_OSC_AdjustXoscHfCapArray
|
||
|
#undef OSC_HPOSCRelativeFrequencyOffsetToRFCoreFormatConvert
|
||
|
#define OSC_HPOSCRelativeFrequencyOffsetToRFCoreFormatConvert NOROM_OSC_HPOSCRelativeFrequencyOffsetToRFCoreFormatConvert
|
||
|
#endif
|
||
|
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// OSCHF switch time calculator defines and globals
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
|
||
|
#define RTC_CV_TO_MS(x) (( 1000 * ( x )) >> 16 )
|
||
|
#define RTC_CV_TO_US(x) (( 1000000 * ( x )) >> 16 )
|
||
|
|
||
|
typedef struct {
|
||
|
uint32_t previousStartupTimeInUs ;
|
||
|
uint32_t timeXoscOff_CV ;
|
||
|
uint32_t timeXoscOn_CV ;
|
||
|
uint32_t timeXoscStable_CV ;
|
||
|
int32_t tempXoscOff ;
|
||
|
} OscHfGlobals_t;
|
||
|
|
||
|
static OscHfGlobals_t oscHfGlobals;
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Configure the oscillator input to the a source clock.
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
void
|
||
|
OSCClockSourceSet(uint32_t ui32SrcClk, uint32_t ui32Osc)
|
||
|
{
|
||
|
// Check the arguments.
|
||
|
ASSERT((ui32SrcClk & OSC_SRC_CLK_LF) ||
|
||
|
(ui32SrcClk & OSC_SRC_CLK_MF) ||
|
||
|
(ui32SrcClk & OSC_SRC_CLK_HF));
|
||
|
ASSERT((ui32Osc == OSC_RCOSC_HF) ||
|
||
|
(ui32Osc == OSC_RCOSC_LF) ||
|
||
|
(ui32Osc == OSC_XOSC_HF) ||
|
||
|
(ui32Osc == OSC_XOSC_LF));
|
||
|
|
||
|
// Request the high frequency source clock (using 24 MHz XTAL)
|
||
|
if(ui32SrcClk & OSC_SRC_CLK_HF)
|
||
|
{
|
||
|
// Enable the HF XTAL as HF clock source
|
||
|
DDI16BitfieldWrite(AUX_DDI0_OSC_BASE, DDI_0_OSC_O_CTL0,
|
||
|
DDI_0_OSC_CTL0_SCLK_HF_SRC_SEL_M,
|
||
|
DDI_0_OSC_CTL0_SCLK_HF_SRC_SEL_S,
|
||
|
ui32Osc);
|
||
|
}
|
||
|
|
||
|
// Configure the medium frequency source clock
|
||
|
if(ui32SrcClk & OSC_SRC_CLK_MF)
|
||
|
{
|
||
|
DDI16BitfieldWrite(AUX_DDI0_OSC_BASE, DDI_0_OSC_O_CTL0,
|
||
|
DDI_0_OSC_CTL0_SCLK_MF_SRC_SEL_M,
|
||
|
DDI_0_OSC_CTL0_SCLK_MF_SRC_SEL_S,
|
||
|
ui32Osc);
|
||
|
}
|
||
|
|
||
|
// Configure the low frequency source clock.
|
||
|
if(ui32SrcClk & OSC_SRC_CLK_LF)
|
||
|
{
|
||
|
// Change the clock source.
|
||
|
DDI16BitfieldWrite(AUX_DDI0_OSC_BASE, DDI_0_OSC_O_CTL0,
|
||
|
DDI_0_OSC_CTL0_SCLK_LF_SRC_SEL_M,
|
||
|
DDI_0_OSC_CTL0_SCLK_LF_SRC_SEL_S,
|
||
|
ui32Osc);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Get the source clock settings
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
uint32_t
|
||
|
OSCClockSourceGet(uint32_t ui32SrcClk)
|
||
|
{
|
||
|
uint32_t ui32ClockSource;
|
||
|
|
||
|
// Check the arguments.
|
||
|
ASSERT((ui32SrcClk & OSC_SRC_CLK_LF) ||
|
||
|
(ui32SrcClk & OSC_SRC_CLK_HF));
|
||
|
|
||
|
// Return the source for the selected clock.
|
||
|
if(ui32SrcClk == OSC_SRC_CLK_LF)
|
||
|
{
|
||
|
ui32ClockSource = DDI16BitfieldRead(AUX_DDI0_OSC_BASE, DDI_0_OSC_O_STAT0,
|
||
|
DDI_0_OSC_STAT0_SCLK_LF_SRC_M,
|
||
|
DDI_0_OSC_STAT0_SCLK_LF_SRC_S);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ui32ClockSource = DDI16BitfieldRead(AUX_DDI0_OSC_BASE, DDI_0_OSC_O_STAT0,
|
||
|
DDI_0_OSC_STAT0_SCLK_HF_SRC_M,
|
||
|
DDI_0_OSC_STAT0_SCLK_HF_SRC_S);
|
||
|
}
|
||
|
return (ui32ClockSource);
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Returns maximum startup time (in microseconds) of XOSC_HF
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
uint32_t
|
||
|
OSCHF_GetStartupTime( uint32_t timeUntilWakeupInMs )
|
||
|
{
|
||
|
uint32_t deltaTimeSinceXoscOnInMs ;
|
||
|
int32_t deltaTempSinceXoscOn ;
|
||
|
uint32_t newStartupTimeInUs ;
|
||
|
|
||
|
deltaTimeSinceXoscOnInMs = RTC_CV_TO_MS( AONRTCCurrentCompareValueGet() - oscHfGlobals.timeXoscOn_CV );
|
||
|
deltaTempSinceXoscOn = AONBatMonTemperatureGetDegC() - oscHfGlobals.tempXoscOff;
|
||
|
|
||
|
if ( deltaTempSinceXoscOn < 0 ) {
|
||
|
deltaTempSinceXoscOn = -deltaTempSinceXoscOn;
|
||
|
}
|
||
|
|
||
|
if ( (( timeUntilWakeupInMs + deltaTimeSinceXoscOnInMs ) > 3000 ) ||
|
||
|
( deltaTempSinceXoscOn > 5 ) ||
|
||
|
( oscHfGlobals.timeXoscStable_CV < oscHfGlobals.timeXoscOn_CV ) ||
|
||
|
( oscHfGlobals.previousStartupTimeInUs == 0 ) )
|
||
|
{
|
||
|
newStartupTimeInUs = 2000;
|
||
|
if (( HWREG( CCFG_BASE + CCFG_O_SIZE_AND_DIS_FLAGS ) & CCFG_SIZE_AND_DIS_FLAGS_DIS_XOSC_OVR_M ) == 0 ) {
|
||
|
newStartupTimeInUs = (( HWREG( CCFG_BASE + CCFG_O_MODE_CONF_1 ) &
|
||
|
CCFG_MODE_CONF_1_XOSC_MAX_START_M ) >>
|
||
|
CCFG_MODE_CONF_1_XOSC_MAX_START_S ) * 125;
|
||
|
// Note: CCFG startup time is "in units of 100us" adding 25% margin results in *125
|
||
|
}
|
||
|
} else {
|
||
|
newStartupTimeInUs = RTC_CV_TO_US( oscHfGlobals.timeXoscStable_CV - oscHfGlobals.timeXoscOn_CV );
|
||
|
newStartupTimeInUs += ( newStartupTimeInUs >> 2 ); // Add 25 percent margin
|
||
|
if ( newStartupTimeInUs < oscHfGlobals.previousStartupTimeInUs ) {
|
||
|
newStartupTimeInUs = oscHfGlobals.previousStartupTimeInUs;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( newStartupTimeInUs < 200 ) {
|
||
|
newStartupTimeInUs = 200;
|
||
|
}
|
||
|
if ( newStartupTimeInUs > 4000 ) {
|
||
|
newStartupTimeInUs = 4000;
|
||
|
}
|
||
|
return ( newStartupTimeInUs );
|
||
|
}
|
||
|
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Turns on XOSC_HF (but without switching to XOSC_HF)
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
void
|
||
|
OSCHF_TurnOnXosc( void )
|
||
|
{
|
||
|
#if ( defined( ROM_OSCClockSourceSet ))
|
||
|
ROM_OSCClockSourceSet( OSC_SRC_CLK_HF | OSC_SRC_CLK_MF, OSC_XOSC_HF );
|
||
|
#else
|
||
|
OSCClockSourceSet( OSC_SRC_CLK_HF | OSC_SRC_CLK_MF, OSC_XOSC_HF );
|
||
|
#endif
|
||
|
oscHfGlobals.timeXoscOn_CV = AONRTCCurrentCompareValueGet();
|
||
|
}
|
||
|
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Switch to XOSC_HF if XOSC_HF is ready.
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
bool
|
||
|
OSCHF_AttemptToSwitchToXosc( void )
|
||
|
{
|
||
|
uint32_t startupTimeInUs;
|
||
|
uint32_t prevLimmit25InUs;
|
||
|
|
||
|
#if ( defined( ROM_OSCClockSourceGet ))
|
||
|
if ( ROM_OSCClockSourceGet( OSC_SRC_CLK_HF ) == OSC_XOSC_HF )
|
||
|
#else
|
||
|
if ( OSCClockSourceGet( OSC_SRC_CLK_HF ) == OSC_XOSC_HF )
|
||
|
#endif
|
||
|
{
|
||
|
// Already on XOSC - nothing to do
|
||
|
return ( 1 );
|
||
|
}
|
||
|
if ( OSCHfSourceReady()) {
|
||
|
OSCHfSourceSwitch();
|
||
|
|
||
|
// Store startup time, but limit to 25 percent reduction each time.
|
||
|
oscHfGlobals.timeXoscStable_CV = AONRTCCurrentCompareValueGet();
|
||
|
startupTimeInUs = RTC_CV_TO_US( oscHfGlobals.timeXoscStable_CV - oscHfGlobals.timeXoscOn_CV );
|
||
|
prevLimmit25InUs = oscHfGlobals.previousStartupTimeInUs;
|
||
|
prevLimmit25InUs -= ( prevLimmit25InUs >> 2 ); // 25 percent margin
|
||
|
oscHfGlobals.previousStartupTimeInUs = startupTimeInUs;
|
||
|
if ( prevLimmit25InUs > startupTimeInUs ) {
|
||
|
oscHfGlobals.previousStartupTimeInUs = prevLimmit25InUs;
|
||
|
}
|
||
|
return ( 1 );
|
||
|
}
|
||
|
return ( 0 );
|
||
|
}
|
||
|
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Switch to RCOSC_HF and turn off XOSC_HF
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
void
|
||
|
OSCHF_SwitchToRcOscTurnOffXosc( void )
|
||
|
{
|
||
|
// Set SCLK_HF and SCLK_MF to RCOSC_HF without checking
|
||
|
// Doing this anyway to keep HF and MF in sync
|
||
|
#if ( defined( ROM_OSCClockSourceSet ))
|
||
|
ROM_OSCClockSourceSet( OSC_SRC_CLK_HF | OSC_SRC_CLK_MF, OSC_RCOSC_HF );
|
||
|
#else
|
||
|
OSCClockSourceSet( OSC_SRC_CLK_HF | OSC_SRC_CLK_MF, OSC_RCOSC_HF );
|
||
|
#endif
|
||
|
|
||
|
// Do the switching if not already running on RCOSC_HF
|
||
|
#if ( defined( ROM_OSCClockSourceGet ))
|
||
|
if ( ROM_OSCClockSourceGet( OSC_SRC_CLK_HF ) != OSC_RCOSC_HF )
|
||
|
#else
|
||
|
if ( OSCClockSourceGet( OSC_SRC_CLK_HF ) != OSC_RCOSC_HF )
|
||
|
#endif
|
||
|
{
|
||
|
OSCHfSourceSwitch();
|
||
|
}
|
||
|
|
||
|
oscHfGlobals.timeXoscOff_CV = AONRTCCurrentCompareValueGet();
|
||
|
oscHfGlobals.tempXoscOff = AONBatMonTemperatureGetDegC();
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Adjust the XOSC HF cap array relative to the factory setting
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
void
|
||
|
OSC_AdjustXoscHfCapArray( int32_t capArrDelta )
|
||
|
{
|
||
|
// read the MODE_CONF register in CCFG
|
||
|
uint32_t ccfg_ModeConfReg = HWREG( CCFG_BASE + CCFG_O_MODE_CONF );
|
||
|
// Clear CAP_MODE and the CAPARRAY_DELATA field
|
||
|
ccfg_ModeConfReg &= ~( CCFG_MODE_CONF_XOSC_CAPARRAY_DELTA_M | CCFG_MODE_CONF_XOSC_CAP_MOD_M );
|
||
|
// Insert new delta value
|
||
|
ccfg_ModeConfReg |= ((((uint32_t)capArrDelta) << CCFG_MODE_CONF_XOSC_CAPARRAY_DELTA_S ) & CCFG_MODE_CONF_XOSC_CAPARRAY_DELTA_M );
|
||
|
// Update the HW register with the new delta value
|
||
|
DDI32RegWrite(AUX_DDI0_OSC_BASE, DDI_0_OSC_O_ANABYPASSVAL1, SetupGetTrimForAnabypassValue1( ccfg_ModeConfReg ));
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Calculate the temperature dependent relative frequency offset of HPOSC
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
int32_t
|
||
|
OSC_HPOSCRelativeFrequencyOffsetGet( int32_t tempDegC )
|
||
|
{
|
||
|
// Estimate HPOSC frequency, using temperature and curve fitting parameters
|
||
|
|
||
|
uint32_t fitParams = HWREG( FCFG1_BASE + FCFG1_O_FREQ_OFFSET );
|
||
|
// Extract the P0,P1,P2 params, and sign extend them via shifting up/down
|
||
|
int32_t paramP0 = (((int32_t)( fitParams << ( 32 - FCFG1_FREQ_OFFSET_HPOSC_COMP_P0_W - FCFG1_FREQ_OFFSET_HPOSC_COMP_P0_S )))
|
||
|
>> ( 32 - FCFG1_FREQ_OFFSET_HPOSC_COMP_P0_W ));
|
||
|
int32_t paramP1 = (((int32_t)( fitParams << ( 32 - FCFG1_FREQ_OFFSET_HPOSC_COMP_P1_W - FCFG1_FREQ_OFFSET_HPOSC_COMP_P1_S )))
|
||
|
>> ( 32 - FCFG1_FREQ_OFFSET_HPOSC_COMP_P1_W ));
|
||
|
int32_t paramP2 = (((int32_t)( fitParams << ( 32 - FCFG1_FREQ_OFFSET_HPOSC_COMP_P2_W - FCFG1_FREQ_OFFSET_HPOSC_COMP_P2_S )))
|
||
|
>> ( 32 - FCFG1_FREQ_OFFSET_HPOSC_COMP_P2_W ));
|
||
|
|
||
|
uint32_t fitParP3 = HWREG( FCFG1_BASE + FCFG1_O_MISC_CONF_2 );
|
||
|
// Extract the P3 param, and sign extend via shifting up/down
|
||
|
int32_t paramP3 = (((int32_t)( fitParP3 << ( 32 - FCFG1_MISC_CONF_2_HPOSC_COMP_P3_W - FCFG1_MISC_CONF_2_HPOSC_COMP_P3_S )))
|
||
|
>> ( 32 - FCFG1_MISC_CONF_2_HPOSC_COMP_P3_W ));
|
||
|
|
||
|
// Now we can find the HPOSC freq offset, given as a signed variable d, expressed by:
|
||
|
//
|
||
|
// F_HPOSC = F_nom * (1 + d/(2^22)) , where: F_HPOSC = HPOSC frequency
|
||
|
// F_nom = nominal clock source frequency (e.g. 48.000 MHz)
|
||
|
// d = describes relative freq offset
|
||
|
|
||
|
// We can estimate the d variable, using temperature compensation parameters:
|
||
|
//
|
||
|
// d = P0 + P1*(t - T0) + P2*(t - T0)^2 + P3*(t - T0)^3, where: P0,P1,P2,P3 are curve fitting parameters from FCFG1
|
||
|
// t = current temperature (from temp sensor) in deg C
|
||
|
// T0 = 27 deg C (fixed temperature constant)
|
||
|
int32_t tempDelta = (tempDegC - 27);
|
||
|
int32_t tempDeltaX2 = tempDelta * tempDelta;
|
||
|
int32_t d = paramP0 + ((tempDelta*paramP1)>>3) + ((tempDeltaX2*paramP2)>>10) + ((tempDeltaX2*tempDelta*paramP3)>>18);
|
||
|
|
||
|
return ( d );
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Converts the relative frequency offset of HPOSC to the RF Core parameter format.
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
int16_t
|
||
|
OSC_HPOSCRelativeFrequencyOffsetToRFCoreFormatConvert( int32_t HPOSC_RelFreqOffset )
|
||
|
{
|
||
|
// The input argument, hereby referred to simply as "d", describes the frequency offset
|
||
|
// of the HPOSC relative to the nominal frequency in this way:
|
||
|
//
|
||
|
// F_HPOSC = F_nom * (1 + d/(2^22))
|
||
|
//
|
||
|
// But for use by the radio, to compensate the frequency error, we need to find the
|
||
|
// frequency offset "rfcFreqOffset" defined in the following format:
|
||
|
//
|
||
|
// F_nom = F_HPOSC * (1 + rfCoreFreqOffset/(2^22))
|
||
|
//
|
||
|
// To derive "rfCoreFreqOffset" from "d" we combine the two above equations and get:
|
||
|
//
|
||
|
// (1 + rfCoreFreqOffset/(2^22)) = (1 + d/(2^22))^-1
|
||
|
//
|
||
|
// Which can be rewritten into:
|
||
|
//
|
||
|
// rfCoreFreqOffset = -d*(2^22) / ((2^22) + d)
|
||
|
//
|
||
|
// = -d * [ 1 / (1 + d/(2^22)) ]
|
||
|
//
|
||
|
// To avoid doing a 64-bit division due to the (1 + d/(2^22))^-1 expression,
|
||
|
// we can use Taylor series (Maclaurin series) to approximate it:
|
||
|
//
|
||
|
// 1 / (1 - x) ~= 1 + x + x^2 + x^3 + x^4 + ... etc (Maclaurin series)
|
||
|
//
|
||
|
// In our case, we have x = - d/(2^22), and we only include up to the first
|
||
|
// order term of the series, as the second order term ((d^2)/(2^44)) is very small:
|
||
|
//
|
||
|
// freqError ~= -d + d^2/(2^22) (+ small approximation error)
|
||
|
//
|
||
|
// The approximation error is negligible for our use.
|
||
|
|
||
|
int32_t rfCoreFreqOffset = -HPOSC_RelFreqOffset + (( HPOSC_RelFreqOffset * HPOSC_RelFreqOffset ) >> 22 );
|
||
|
|
||
|
return ( rfCoreFreqOffset );
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Get crystal amplitude (assuming crystal is running).
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
uint32_t
|
||
|
OSCHF_DebugGetCrystalAmplitude( void )
|
||
|
{
|
||
|
uint32_t oscCfgRegCopy ;
|
||
|
uint32_t startTime ;
|
||
|
uint32_t deltaTime ;
|
||
|
uint32_t ampValue ;
|
||
|
|
||
|
// The specified method is as follows:
|
||
|
// 1. Set minimum interval between oscillator amplitude calibrations.
|
||
|
// (Done by setting PER_M=0 and PER_E=1)
|
||
|
// 2. Wait approximately 4 milliseconds in order to measure over a
|
||
|
// moderately large number of calibrations.
|
||
|
// 3. Read out the crystal amplitude value from the peek detector.
|
||
|
// 4. Restore original oscillator amplitude calibrations interval.
|
||
|
// 5. Return crystal amplitude value converted to millivolt.
|
||
|
oscCfgRegCopy = HWREG( AON_WUC_BASE + AON_WUC_O_OSCCFG );
|
||
|
HWREG( AON_WUC_BASE + AON_WUC_O_OSCCFG ) = ( 1 << AON_WUC_OSCCFG_PER_E_S );
|
||
|
startTime = AONRTCCurrentCompareValueGet();
|
||
|
do {
|
||
|
deltaTime = AONRTCCurrentCompareValueGet() - startTime;
|
||
|
} while ( deltaTime < ((uint32_t)( 0.004 * FACTOR_SEC_TO_COMP_VAL_FORMAT )));
|
||
|
ampValue = ( HWREG( AUX_DDI0_OSC_BASE + DDI_0_OSC_O_STAT1 ) &
|
||
|
DDI_0_OSC_STAT1_HPM_UPDATE_AMP_M ) >>
|
||
|
DDI_0_OSC_STAT1_HPM_UPDATE_AMP_S ;
|
||
|
HWREG( AON_WUC_BASE + AON_WUC_O_OSCCFG ) = oscCfgRegCopy;
|
||
|
|
||
|
return ( ampValue * 15 );
|
||
|
}
|
||
|
|
||
|
//*****************************************************************************
|
||
|
//
|
||
|
// Get the expected average crystal amplitude.
|
||
|
//
|
||
|
//*****************************************************************************
|
||
|
uint32_t
|
||
|
OSCHF_DebugGetExpectedAverageCrystalAmplitude( void )
|
||
|
{
|
||
|
uint32_t ampCompTh1 ;
|
||
|
uint32_t highThreshold ;
|
||
|
uint32_t lowThreshold ;
|
||
|
|
||
|
ampCompTh1 = HWREG( AUX_DDI0_OSC_BASE + DDI_0_OSC_O_AMPCOMPTH1 );
|
||
|
highThreshold = ( ampCompTh1 & DDI_0_OSC_AMPCOMPTH1_HPMRAMP3_HTH_M ) >>
|
||
|
DDI_0_OSC_AMPCOMPTH1_HPMRAMP3_HTH_S ;
|
||
|
lowThreshold = ( ampCompTh1 & DDI_0_OSC_AMPCOMPTH1_HPMRAMP3_LTH_M ) >>
|
||
|
DDI_0_OSC_AMPCOMPTH1_HPMRAMP3_LTH_S ;
|
||
|
|
||
|
return ((( highThreshold + lowThreshold ) * 15 ) >> 1 );
|
||
|
}
|