2015-01-12 14:42:48 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014-2015 Eistec AB
|
|
|
|
*
|
|
|
|
* 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 board_mulle
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Board specific implementations for the Mulle board
|
|
|
|
*
|
2015-09-20 13:47:39 +02:00
|
|
|
* @author Joakim Nohlgård <joakim.nohlgard@eistec.se>
|
2015-01-12 14:42:48 +01:00
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stddef.h> /* for NULL */
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "board.h"
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "mcg.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "periph/uart.h"
|
2015-07-22 14:13:30 +02:00
|
|
|
#include "periph/rtt.h"
|
2015-01-12 14:42:48 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Initialize the boards on-board LEDs
|
|
|
|
*
|
|
|
|
* The LEDs are initialized here in order to be able to use them in the early
|
|
|
|
* boot for diagnostics.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
static inline void leds_init(void);
|
|
|
|
|
|
|
|
/** @brief Initialize the GPIO pins controlling the power switches. */
|
|
|
|
static inline void power_pins_init(void);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Set clock prescalers to safe values
|
|
|
|
*
|
|
|
|
* This should be done before switching to FLL/PLL as clock source to ensure
|
|
|
|
* that all clocks remain within the specified limits.
|
|
|
|
*/
|
|
|
|
static inline void set_safe_clock_dividers(void);
|
|
|
|
|
|
|
|
/** @brief Set the FLL source clock to RTC32k */
|
|
|
|
static inline void set_fll_source(void);
|
|
|
|
|
|
|
|
|
|
|
|
void board_init(void)
|
|
|
|
{
|
|
|
|
/* initialize the boards LEDs, this is done first for debugging purposes */
|
|
|
|
leds_init();
|
|
|
|
|
2015-07-22 14:13:30 +02:00
|
|
|
/* Initialize power control pins */
|
|
|
|
power_pins_init();
|
|
|
|
|
|
|
|
/* Turn on Vperiph for peripherals */
|
|
|
|
gpio_set(MULLE_POWER_VPERIPH);
|
|
|
|
|
|
|
|
/* Turn on AVDD for reading voltages */
|
|
|
|
gpio_set(MULLE_POWER_AVDD);
|
|
|
|
|
2015-01-12 14:42:48 +01:00
|
|
|
LED_RED_ON;
|
|
|
|
|
|
|
|
/* Initialize RTC oscillator as early as possible since we are using it as a
|
|
|
|
* base clock for the FLL.
|
|
|
|
* It takes a while to stabilize the oscillator, therefore we do this as
|
|
|
|
* soon as possible during boot in order to let it stabilize while other
|
|
|
|
* stuff is initializing. */
|
|
|
|
/* If the clock is not stable then the UART will have the wrong baud rate
|
|
|
|
* for debug prints as well */
|
2015-07-22 14:13:30 +02:00
|
|
|
rtt_init();
|
2015-01-12 14:42:48 +01:00
|
|
|
|
|
|
|
/* Set up clocks */
|
|
|
|
set_safe_clock_dividers();
|
|
|
|
|
|
|
|
set_fll_source();
|
|
|
|
|
|
|
|
kinetis_mcg_set_mode(KINETIS_MCG_FEE);
|
|
|
|
|
|
|
|
/* At this point we need to wait for 1 ms until the clock is stable.
|
|
|
|
* Since the clock is not yet stable we can only guess how long we must
|
|
|
|
* wait. I have tried to make this as short as possible but still being able
|
|
|
|
* to read the initialization messages written on the UART.
|
|
|
|
* (If the clock is not stable all UART output is garbled until it has
|
|
|
|
* stabilized) */
|
|
|
|
for (int i = 0; i < 100000; ++i) {
|
|
|
|
asm volatile("nop\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Update SystemCoreClock global var */
|
|
|
|
SystemCoreClockUpdate();
|
|
|
|
|
|
|
|
/* initialize the CPU */
|
|
|
|
cpu_init();
|
|
|
|
}
|
|
|
|
|
2015-07-22 14:13:30 +02:00
|
|
|
/**
|
|
|
|
* @brief Initialize the boards on-board LEDs
|
|
|
|
*
|
|
|
|
* The LEDs are initialized here in order to be able to use them in the early
|
|
|
|
* boot for diagnostics.
|
|
|
|
*
|
|
|
|
*/
|
2015-01-12 14:42:48 +01:00
|
|
|
static inline void leds_init(void)
|
|
|
|
{
|
|
|
|
/* The pin configuration can be found in board.h and periph_conf.h */
|
2015-06-03 18:27:05 +02:00
|
|
|
gpio_init(LED_RED_GPIO, GPIO_DIR_OUT, GPIO_NOPULL);
|
|
|
|
gpio_init(LED_YELLOW_GPIO, GPIO_DIR_OUT, GPIO_NOPULL);
|
|
|
|
gpio_init(LED_GREEN_GPIO, GPIO_DIR_OUT, GPIO_NOPULL);
|
2015-01-12 14:42:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void power_pins_init(void)
|
|
|
|
{
|
2015-06-03 18:27:05 +02:00
|
|
|
gpio_init(MULLE_POWER_AVDD, GPIO_DIR_OUT, GPIO_NOPULL);
|
|
|
|
gpio_init(MULLE_POWER_VPERIPH, GPIO_DIR_OUT, GPIO_NOPULL);
|
|
|
|
gpio_init(MULLE_POWER_VSEC, GPIO_DIR_OUT, GPIO_NOPULL);
|
2015-01-12 14:42:48 +01:00
|
|
|
gpio_clear(MULLE_POWER_AVDD);
|
|
|
|
gpio_clear(MULLE_POWER_VPERIPH);
|
|
|
|
gpio_clear(MULLE_POWER_VSEC);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void set_safe_clock_dividers(void)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* We want to achieve the following clocks:
|
|
|
|
* Core/system: <100MHz
|
|
|
|
* Bus: <50MHz
|
|
|
|
* FlexBus: <50MHz
|
|
|
|
* Flash: <25MHz
|
|
|
|
*
|
|
|
|
* using dividers 1-2-2-4 will obey the above limits when using a 96MHz FLL source.
|
|
|
|
*/
|
|
|
|
SIM->CLKDIV1 = (
|
|
|
|
SIM_CLKDIV1_OUTDIV1(CONFIG_CLOCK_K60_SYS_DIV) | /* Core/System clock divider */
|
|
|
|
SIM_CLKDIV1_OUTDIV2(CONFIG_CLOCK_K60_BUS_DIV) | /* Bus clock divider */
|
|
|
|
SIM_CLKDIV1_OUTDIV3(CONFIG_CLOCK_K60_FB_DIV) | /* FlexBus divider, not used in Mulle */
|
|
|
|
SIM_CLKDIV1_OUTDIV4(CONFIG_CLOCK_K60_FLASH_DIV)); /* Flash clock divider */
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void set_fll_source(void)
|
|
|
|
{
|
|
|
|
/* Select FLL as source (as opposed to PLL) */
|
|
|
|
SIM->SOPT2 &= ~(SIM_SOPT2_PLLFLLSEL_MASK);
|
|
|
|
/* Use external 32kHz RTC clock as source for OSC32K */
|
|
|
|
#if K60_CPU_REV == 1
|
|
|
|
SIM->SOPT1 |= SIM_SOPT1_OSC32KSEL_MASK;
|
|
|
|
#elif K60_CPU_REV == 2
|
|
|
|
SIM->SOPT1 = (SIM->SOPT1 & ~(SIM_SOPT1_OSC32KSEL_MASK)) | SIM_SOPT1_OSC32KSEL(2);
|
|
|
|
#else
|
|
|
|
#error Unknown K60 CPU revision
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Select RTC 32kHz clock as reference clock for the FLL */
|
|
|
|
#if K60_CPU_REV == 1
|
|
|
|
/* Rev 1 parts */
|
|
|
|
SIM->SOPT2 |= SIM_SOPT2_MCGCLKSEL_MASK;
|
|
|
|
#elif K60_CPU_REV == 2
|
|
|
|
/* Rev 2 parts */
|
|
|
|
MCG->C7 = (MCG_C7_OSCSEL_MASK);
|
|
|
|
#else
|
|
|
|
#error Unknown K60 CPU revision
|
|
|
|
#endif
|
|
|
|
}
|