mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 01:52:45 +01:00
361 lines
14 KiB
C
361 lines
14 KiB
C
|
/*
|
||
|
* Copyright (C) 2014 INRIA
|
||
|
* 2023 Otto-von-Guericke-Universität Magdeburg
|
||
|
*
|
||
|
* 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_msp430fxyz
|
||
|
* @{
|
||
|
* @file
|
||
|
* @brief MSP430Fxzy clock initialization
|
||
|
*
|
||
|
* @author Oliver Hahm <oliver.hahm@inria.fr>
|
||
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
||
|
* @author Marian Buschsieweke <marian.buschsieweke@ovgu.de>
|
||
|
*
|
||
|
* @}
|
||
|
*/
|
||
|
|
||
|
#include <assert.h>
|
||
|
#include <stdbool.h>
|
||
|
#include <stdint.h>
|
||
|
|
||
|
#include "debug.h"
|
||
|
#include "macros/math.h"
|
||
|
#include "macros/units.h"
|
||
|
#include "periph_conf.h"
|
||
|
#include "periph_cpu.h"
|
||
|
|
||
|
#define RSEL_MASK (RSEL0 | RSEL1 | RSEL2)
|
||
|
|
||
|
uint32_t msp430_fxyz_dco_freq;
|
||
|
|
||
|
static inline bool is_dco_in_use(const msp430_fxyz_clock_params_t *params)
|
||
|
{
|
||
|
return ((params->main_clock_source == MAIN_CLOCK_SOURCE_DCOCLK) ||
|
||
|
(params->submain_clock_source == SUBMAIN_CLOCK_SOURCE_DCOCLK));
|
||
|
}
|
||
|
|
||
|
/* As low speed crystal only a standard 32.768 kHz watch crystal is allowed */
|
||
|
static inline bool is_valid_low_freq(uint32_t freq) {
|
||
|
return (freq == 32768);
|
||
|
}
|
||
|
|
||
|
/* As high speed crystal anything between 450 kHz and 8 MHz is allowed */
|
||
|
static inline bool is_valid_high_freq(uint32_t freq) {
|
||
|
return ((freq >= KHZ(450)) && (freq <= MHZ(8)));
|
||
|
}
|
||
|
|
||
|
/* C doesn't have constexpr() or an equivalent feature to perform complex
|
||
|
* compile time tests. To still do so, we rely on constant folding and dead
|
||
|
* code elimination:
|
||
|
* If a test doesn't pass, we call an undefined function with a meaningful
|
||
|
* name as "error message". If that call is optimized out (because the
|
||
|
* configuration test passes), the linking step passes. If the config is
|
||
|
* invalid, the function call will not be optimized out and the linker will not
|
||
|
* be able to resolve all symbols, resulting in invalid configs being caught
|
||
|
* at compile time. */
|
||
|
static void check_config(void)
|
||
|
{
|
||
|
/* LFXT1 can either be a low frequency 32.768 kHz watch crystal or a
|
||
|
* high frequency crustal between 450 kHz and 8 MHz. We cannot function
|
||
|
* without this crystal */
|
||
|
if (!is_valid_low_freq(clock_params.lfxt1_frequency)
|
||
|
&& !is_valid_high_freq(clock_params.lfxt1_frequency)) {
|
||
|
extern void lfxt1_frequency_invalid(void);
|
||
|
lfxt1_frequency_invalid();
|
||
|
}
|
||
|
|
||
|
/* XT2 is not required and may be configured as 0 Hz to indicate its
|
||
|
* absence. If it is present, we require 450 kHz <= XT <= 8 MHz */
|
||
|
if ((clock_params.xt2_frequency != 0) &&
|
||
|
!is_valid_high_freq(clock_params.xt2_frequency)) {
|
||
|
extern void xt2_frequency_invalid(void);
|
||
|
xt2_frequency_invalid();
|
||
|
}
|
||
|
|
||
|
/* XT2 can only source the main system clock if it is present */
|
||
|
if ((clock_params.main_clock_source == MAIN_CLOCK_SOURCE_XT2CLK)
|
||
|
&& (clock_params.xt2_frequency == 0)) {
|
||
|
extern void xt2_as_main_clock_source_selected_but_xt2_is_not_present(void);
|
||
|
xt2_as_main_clock_source_selected_but_xt2_is_not_present();
|
||
|
}
|
||
|
|
||
|
/* If the DCO is disabled (0 Hz), we cannot use it as clock source */
|
||
|
if (is_dco_in_use(&clock_params)
|
||
|
&& (clock_params.target_dco_frequency == 0)) {
|
||
|
extern void dco_configured_as_clock_source_but_is_disabled(void);
|
||
|
dco_configured_as_clock_source_but_is_disabled();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void busy_wait(uint16_t loops)
|
||
|
{
|
||
|
while (loops) {
|
||
|
/* This empty inline assembly should be enough to convince the
|
||
|
* compiler that the loop cannot be optimized out. Tested with
|
||
|
* GCC 12.2 and clang 16.0.0 successfully. */
|
||
|
__asm__ __volatile__ (
|
||
|
""
|
||
|
: /* no outputs */
|
||
|
: /* no inputs */
|
||
|
: /* no clobbers */
|
||
|
);
|
||
|
loops--;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void wait_for_crystals_to_stabilize(void)
|
||
|
{
|
||
|
/* The MSP430x1xx MCU family have an oscillator fault detector that sets
|
||
|
* the OFIFG flag when a fault is detected, which can trigger an IRQ
|
||
|
* and/or be polled in software. We go for polling here.
|
||
|
*
|
||
|
* There are some foot guns to take into account:
|
||
|
*
|
||
|
* 1. The LFXT1 clock is only monitored in high speed mode. So for the
|
||
|
* typical setup of using a 32.768 kHz watch crystal one can only wait
|
||
|
* for some preconfigured static value and hope this was long enough
|
||
|
* to stabilize. (But that seems to work fine in practise.)
|
||
|
* 2. If no high speed crystal is in use (e.g. LFXT1 is used with a
|
||
|
* watch crystal and XT2 is not connected), the fault flag will
|
||
|
* stick and cannot be cleared. Waiting for the flag to clear would
|
||
|
* result in getting stuck during boot.
|
||
|
*
|
||
|
* Because of 1, we go for best effort by just waiting UINT16_MAX busy
|
||
|
* loops for the watch crystal to stabilize. This is common practise and
|
||
|
* seems to work just fine.
|
||
|
* Because of 2 we don't wait for OFIFG if no high speed crystal is in use.
|
||
|
*
|
||
|
* We wait for watch crystal first, as chances are good that by then the
|
||
|
* high speed crystal (if any) is stable or about to become stable as well,
|
||
|
* so that the second wait is fast.
|
||
|
*/
|
||
|
|
||
|
if (is_valid_low_freq(clock_params.lfxt1_frequency)) {
|
||
|
/* wait best effort for watch crystal to stabilize */
|
||
|
busy_wait(UINT16_MAX);
|
||
|
}
|
||
|
|
||
|
if (is_valid_high_freq(clock_params.lfxt1_frequency)
|
||
|
|| is_valid_high_freq(clock_params.xt2_frequency)) {
|
||
|
/* at least one high frequency crystal present, wait for that to
|
||
|
* stabilize */
|
||
|
do {
|
||
|
/* Clear oscillator fault flag */
|
||
|
IFG1 &= ~(OFIFG);
|
||
|
/* Now wait for at least 50 μs so that the oscillator fault flag is
|
||
|
* reliably set again if the oscillators are not stable yet.
|
||
|
* The datasheet indicates that counting down form UINT8_MAX when
|
||
|
* running from DCO with default settings (as during boot) is taking
|
||
|
* at least 50 μs */
|
||
|
busy_wait(UINT8_MAX);
|
||
|
/* Finally, check if the oscillator fault flag was reset by hardware */
|
||
|
} while (IFG1 & OFIFG);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void calibrate_dco(void)
|
||
|
{
|
||
|
/* We assume that the LFXT1 frequency is stable. Now we compute what the
|
||
|
* ratio of the (unstable) DCO frequency to the stable LFXT1 frequency
|
||
|
* (available as auxiliary clock with a clock divider of 8) and use a timer
|
||
|
* to capture the actual clock ratio */
|
||
|
const uint16_t clock_ratio = DIV_ROUND(clock_params.target_dco_frequency,
|
||
|
(clock_params.lfxt1_frequency / 8));
|
||
|
|
||
|
/* The CCI2B input on capture/compare unit 2 of timer A has the auxiliary
|
||
|
* clock as internal input. Hence, for every tick of the auxiliary clock
|
||
|
* we will get one capture of the continuously running timer. The diff
|
||
|
* of two subsequent captures matches then the actual clock ratio. */
|
||
|
TACCTL2 = CCIS_1 /* CCI2B as capture input */
|
||
|
| CM_1 /* capture on rising edge */
|
||
|
| CAP; /* capture mode instead of compare mode */
|
||
|
/* configure timer A to run from submain clock, which now is sourced
|
||
|
* by DCO without divider*/
|
||
|
TACTL = TASSEL_2 /* submain clock (DCO) as source */
|
||
|
| TACLR /* clear timer */
|
||
|
| MC_2; /* continuous mode, count up to UINT16_MAX */
|
||
|
|
||
|
/* First, binary search for the correct RSELx settings (which have a
|
||
|
* higher impact on the actual frequency than the DCOCTL settings).
|
||
|
* We're starting with a highest bit. */
|
||
|
|
||
|
uint8_t bcsctl1 = BCSCTL1 & ~(RSEL_MASK);
|
||
|
uint8_t rselx = 0;
|
||
|
|
||
|
for (uint8_t iter = 0x04; iter != 0; iter >>= 1) {
|
||
|
BCSCTL1 = bcsctl1 | rselx | iter;
|
||
|
|
||
|
/* busy wait for timer to capture */
|
||
|
while (!(TACCTL2 & CCIFG)) { }
|
||
|
/* clear capture flag */
|
||
|
TACCTL2 &= ~CCIFG;
|
||
|
uint16_t start = TACCR2;
|
||
|
|
||
|
/* busy wait for timer to capture */
|
||
|
while (!(TACCTL2 & CCIFG)) { }
|
||
|
/* clear capture flag */
|
||
|
TACCTL2 &= ~CCIFG;
|
||
|
uint16_t stop = TACCR2;
|
||
|
|
||
|
uint16_t actual_clock_ratio = stop - start;
|
||
|
if (actual_clock_ratio <= clock_ratio) {
|
||
|
/* DCO too slow, or just fast enough --> keep the bit */
|
||
|
rselx |= iter;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Now do the same dance again, but for DCOCTL */
|
||
|
uint8_t dcoctl = 0;
|
||
|
uint16_t actual_clock_ratio;
|
||
|
|
||
|
for (uint8_t iter = 0x80; iter != 0; iter >>= 1) {
|
||
|
DCOCTL = dcoctl | iter;
|
||
|
|
||
|
/* busy wait for timer to capture */
|
||
|
while (!(TACCTL2 & CCIFG)) { }
|
||
|
/* clear capture flag */
|
||
|
TACCTL2 &= ~CCIFG;
|
||
|
uint16_t start = TACCR2;
|
||
|
|
||
|
/* busy wait for timer to capture */
|
||
|
while (!(TACCTL2 & CCIFG)) { }
|
||
|
/* clear capture flag */
|
||
|
TACCTL2 &= ~CCIFG;
|
||
|
uint16_t stop = TACCR2;
|
||
|
|
||
|
actual_clock_ratio = stop - start;
|
||
|
if (actual_clock_ratio <= clock_ratio) {
|
||
|
/* DCO too slow, or just fast enough --> keep the bit */
|
||
|
dcoctl |= iter;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* stopping timer used fore calibration */
|
||
|
TACCTL2 = 0;
|
||
|
TACTL = 0;
|
||
|
|
||
|
/* calculate actual DCO frequency */
|
||
|
msp430_fxyz_dco_freq = (clock_params.lfxt1_frequency * actual_clock_ratio) >> 3;
|
||
|
}
|
||
|
|
||
|
void default_clock_init(void)
|
||
|
{
|
||
|
/* This function contains only dead code and the call should be fully
|
||
|
* optimized out by the compiler. In fact, we rely on this: If it is
|
||
|
* not fully optimized out, we will get linker errors. */
|
||
|
check_config();
|
||
|
|
||
|
#ifdef __MSP430_HAS_WDT__
|
||
|
/* Stopping watchdog timer, clock initialization may take a while. */
|
||
|
WDTCTL = WDTPW | WDTHOLD;
|
||
|
#endif
|
||
|
|
||
|
/* set LFXT1 and XT2 hardware settings, keep RSEL to reset value (only
|
||
|
* RSEL 2 bit set) */
|
||
|
uint8_t bcsctl1 = RSEL2;
|
||
|
if (clock_params.xt2_frequency == 0) {
|
||
|
bcsctl1 |= XT2OFF;
|
||
|
}
|
||
|
if (clock_params.lfxt1_frequency > 32768) {
|
||
|
bcsctl1 |= XTS;
|
||
|
}
|
||
|
|
||
|
/* We want to calibrate DCO against the auxiliary clock domain, which is
|
||
|
* hardwired to use LFXT1 as clock source. Because LFXT1 is always using
|
||
|
* a crystal (and mostly even an accurate watch crystal), we can assume
|
||
|
* that the auxiliary clock to be accurate (enough). The algorithm we
|
||
|
* use for clock calibration assumes that multiple DCO ticks match one
|
||
|
* tick on the auxiliary clock. We use the highest possible clock divider
|
||
|
* for now, to increase the range this works. */
|
||
|
if (clock_params.target_dco_frequency != 0) {
|
||
|
bcsctl1 |= AUXILIARY_CLOCK_DIVIDE_BY_8;
|
||
|
}
|
||
|
|
||
|
BCSCTL1 = bcsctl1;
|
||
|
|
||
|
uint8_t bcsctl2 = 0;
|
||
|
|
||
|
if (clock_params.has_r_osc) {
|
||
|
bcsctl2 = DCOR;
|
||
|
}
|
||
|
|
||
|
BCSCTL2 = bcsctl2;
|
||
|
|
||
|
/* In case we run from a crystal, we need that to stabilize. If we run from
|
||
|
* DCO instead, we also need the crystals to become stable so that we can
|
||
|
* calibrate the DCO against it.
|
||
|
* ==> Wait for crystal to be stable regardless of config */
|
||
|
wait_for_crystals_to_stabilize();
|
||
|
|
||
|
/* If a target DCO frequency was configured, we calibrate the DCO to run at
|
||
|
* that - even if the DCO is not used to source either main or submain
|
||
|
* clock. The application may switch the clock source later on to DCO, e.g.
|
||
|
* to run by default at a low clock to reduce power consumption and go to
|
||
|
* "turbo mode" for event processing */
|
||
|
if (clock_params.target_dco_frequency != 0) {
|
||
|
calibrate_dco();
|
||
|
}
|
||
|
|
||
|
/* Crystals are now stable and the DCO has been calibrated, if the
|
||
|
* board config asked for a calibrated DCO clock. We can now safely
|
||
|
* apply the clock sources and also apply the clock dividers */
|
||
|
|
||
|
/* BCSCTL1 contains now freshly calibrated RSELx parameters in addition
|
||
|
* to the auxiliary clock divider we used for calibration. We read the
|
||
|
* settings back and clear the clock divider to apply the one provided
|
||
|
* by the board settings. */
|
||
|
bcsctl1 = BCSCTL1 & ~(AUXILIARY_CLOCK_DIVIDE_BY_8);
|
||
|
BCSCTL1 = bcsctl1 | clock_params.auxiliary_clock_divier;
|
||
|
|
||
|
/* apply main and submain clock source and clock dividers */
|
||
|
BCSCTL2 = bcsctl2
|
||
|
| clock_params.main_clock_source
|
||
|
| clock_params.submain_clock_source
|
||
|
| clock_params.main_clock_divier
|
||
|
| clock_params.submain_clock_divier;
|
||
|
|
||
|
/* if DCO is not used at all, disable it to preserve power */
|
||
|
if (clock_params.target_dco_frequency == 0) {
|
||
|
/* Setting bit SCG0 in the status register (r2) disables the DCO.
|
||
|
* We do so in assembly, as r2 is not memory mapped. */
|
||
|
__asm__ __volatile__ (
|
||
|
"bis %[scg0], r2" "\n\t"
|
||
|
: /* no outputs */
|
||
|
: /* inputs: */
|
||
|
[scg0] "i"(SCG0) /* bitmask to set SCGO0 as immediate */
|
||
|
: /* no clobbers */
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
__attribute__((weak, alias("default_clock_init"))) void clock_init(void);
|
||
|
|
||
|
uint32_t msp430_fxyz_submain_clock_freq(void) {
|
||
|
uint16_t shift = (clock_params.submain_clock_divier >> 1) & 0x3;
|
||
|
switch (clock_params.submain_clock_source) {
|
||
|
case SUBMAIN_CLOCK_SOURCE_LFXT1CLK:
|
||
|
/* the submain clock cannot explicitly select LFXT1 or XT2, but really
|
||
|
* is "crystal" or DCO. If the selection is "crystal", it will default
|
||
|
* to XT2 as clock source (if enabled) and falls back to LFXT1. */
|
||
|
if (clock_params.xt2_frequency) {
|
||
|
return clock_params.xt2_frequency >> shift;
|
||
|
}
|
||
|
return clock_params.lfxt1_frequency >> shift;
|
||
|
default:
|
||
|
case SUBMAIN_CLOCK_SOURCE_DCOCLK:
|
||
|
return msp430_fxyz_dco_freq >> shift;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
uint32_t msp430_fxyz_auxiliary_clock_freq(void)
|
||
|
{
|
||
|
uint16_t shift = (clock_params.auxiliary_clock_divier >> 4) & 0x3;
|
||
|
return clock_params.lfxt1_frequency >> shift;
|
||
|
}
|