2019-09-16 18:45:36 +03:00
|
|
|
// Code to setup clocks and gpio on stm32f0
|
|
|
|
//
|
|
|
|
// Copyright (C) 2019 Kevin O'Connor <kevin@koconnor.net>
|
|
|
|
//
|
|
|
|
// This file may be distributed under the terms of the GNU GPLv3 license.
|
|
|
|
|
2019-11-25 22:00:30 +03:00
|
|
|
#include "autoconf.h" // CONFIG_CLOCK_REF_FREQ
|
2019-10-24 18:35:18 +03:00
|
|
|
#include "board/armcm_boot.h" // armcm_main
|
2019-10-27 00:11:04 +03:00
|
|
|
#include "board/irq.h" // irq_disable
|
2019-09-16 18:45:36 +03:00
|
|
|
#include "command.h" // DECL_CONSTANT_STR
|
|
|
|
#include "internal.h" // enable_pclock
|
2019-10-24 18:35:18 +03:00
|
|
|
#include "sched.h" // sched_main
|
2019-09-16 18:45:36 +03:00
|
|
|
|
|
|
|
#define FREQ_PERIPH 48000000
|
|
|
|
|
|
|
|
// Enable a peripheral clock
|
|
|
|
void
|
|
|
|
enable_pclock(uint32_t periph_base)
|
|
|
|
{
|
|
|
|
if (periph_base < SYSCFG_BASE) {
|
|
|
|
uint32_t pos = (periph_base - APBPERIPH_BASE) / 0x400;
|
|
|
|
RCC->APB1ENR |= 1 << pos;
|
|
|
|
RCC->APB1ENR;
|
|
|
|
} else if (periph_base < AHBPERIPH_BASE) {
|
|
|
|
uint32_t pos = (periph_base - SYSCFG_BASE) / 0x400;
|
|
|
|
RCC->APB2ENR |= 1 << pos;
|
|
|
|
RCC->APB2ENR;
|
|
|
|
} else {
|
|
|
|
uint32_t pos = (periph_base - AHB2PERIPH_BASE) / 0x400;
|
|
|
|
RCC->AHBENR |= 1 << (pos + 17);
|
|
|
|
RCC->AHBENR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if a peripheral clock has been enabled
|
|
|
|
int
|
|
|
|
is_enabled_pclock(uint32_t periph_base)
|
|
|
|
{
|
|
|
|
if (periph_base < SYSCFG_BASE) {
|
|
|
|
uint32_t pos = (periph_base - APBPERIPH_BASE) / 0x400;
|
|
|
|
return RCC->APB1ENR & (1 << pos);
|
|
|
|
} else if (periph_base < AHBPERIPH_BASE) {
|
|
|
|
uint32_t pos = (periph_base - SYSCFG_BASE) / 0x400;
|
|
|
|
return RCC->APB2ENR & (1 << pos);
|
|
|
|
} else {
|
|
|
|
uint32_t pos = (periph_base - AHB2PERIPH_BASE) / 0x400;
|
|
|
|
return RCC->AHBENR & (1 << (pos + 17));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the frequency of the given peripheral clock
|
|
|
|
uint32_t
|
|
|
|
get_pclock_frequency(uint32_t periph_base)
|
|
|
|
{
|
|
|
|
return FREQ_PERIPH;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Enable a GPIO peripheral clock
|
|
|
|
void
|
|
|
|
gpio_clock_enable(GPIO_TypeDef *regs)
|
|
|
|
{
|
|
|
|
uint32_t rcc_pos = ((uint32_t)regs - AHB2PERIPH_BASE) / 0x400;
|
|
|
|
RCC->AHBENR |= 1 << (rcc_pos + 17);
|
|
|
|
RCC->AHBENR;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the mode and extended function of a pin
|
|
|
|
void
|
|
|
|
gpio_peripheral(uint32_t gpio, uint32_t mode, int pullup)
|
|
|
|
{
|
|
|
|
GPIO_TypeDef *regs = digital_regs[GPIO2PORT(gpio)];
|
|
|
|
|
|
|
|
// Enable GPIO clock
|
|
|
|
gpio_clock_enable(regs);
|
|
|
|
|
|
|
|
// Configure GPIO
|
|
|
|
uint32_t mode_bits = mode & 0xf, func = (mode >> 4) & 0xf, od = mode >> 8;
|
|
|
|
uint32_t pup = pullup ? (pullup > 0 ? 1 : 2) : 0;
|
|
|
|
uint32_t pos = gpio % 16, af_reg = pos / 8;
|
|
|
|
uint32_t af_shift = (pos % 8) * 4, af_msk = 0x0f << af_shift;
|
|
|
|
uint32_t m_shift = pos * 2, m_msk = 0x03 << m_shift;
|
|
|
|
|
|
|
|
regs->AFR[af_reg] = (regs->AFR[af_reg] & ~af_msk) | (func << af_shift);
|
|
|
|
regs->MODER = (regs->MODER & ~m_msk) | (mode_bits << m_shift);
|
|
|
|
regs->PUPDR = (regs->PUPDR & ~m_msk) | (pup << m_shift);
|
|
|
|
regs->OTYPER = (regs->OTYPER & ~(1 << pos)) | (od << pos);
|
|
|
|
regs->OSPEEDR = (regs->OSPEEDR & ~m_msk) | (0x02 << m_shift);
|
|
|
|
}
|
|
|
|
|
2019-10-27 00:11:04 +03:00
|
|
|
#define USB_BOOT_FLAG_ADDR (CONFIG_RAM_START + CONFIG_RAM_SIZE - 1024)
|
|
|
|
#define USB_BOOT_FLAG 0x55534220424f4f54 // "USB BOOT"
|
|
|
|
|
2019-09-16 18:45:36 +03:00
|
|
|
// Handle USB reboot requests
|
|
|
|
void
|
|
|
|
usb_request_bootloader(void)
|
|
|
|
{
|
2019-10-27 00:11:04 +03:00
|
|
|
irq_disable();
|
|
|
|
*(uint64_t*)USB_BOOT_FLAG_ADDR = USB_BOOT_FLAG;
|
|
|
|
NVIC_SystemReset();
|
2019-09-16 18:45:36 +03:00
|
|
|
}
|
|
|
|
|
2019-12-10 05:03:05 +03:00
|
|
|
// Copy vector table and remap ram so new vector table is used
|
|
|
|
static void
|
|
|
|
enable_ram_vectortable(void)
|
|
|
|
{
|
|
|
|
// Symbols created by armcm_link.lds.S linker script
|
|
|
|
extern uint32_t _ram_vectortable_start, _ram_vectortable_end;
|
|
|
|
extern uint32_t _text_vectortable_start;
|
|
|
|
|
|
|
|
uint32_t count = (&_ram_vectortable_end - &_ram_vectortable_start) * 4;
|
|
|
|
__builtin_memcpy(&_ram_vectortable_start, &_text_vectortable_start, count);
|
|
|
|
barrier();
|
|
|
|
|
|
|
|
enable_pclock(SYSCFG_BASE);
|
|
|
|
SYSCFG->CFGR1 |= 3 << SYSCFG_CFGR1_MEM_MODE_Pos;
|
|
|
|
}
|
|
|
|
|
2019-11-25 22:00:30 +03:00
|
|
|
#if !CONFIG_STM32_CLOCK_REF_INTERNAL
|
2019-09-16 18:45:36 +03:00
|
|
|
DECL_CONSTANT_STR("RESERVE_PINS_crystal", "PF0,PF1");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Configure and enable the PLL as clock source
|
|
|
|
static void
|
|
|
|
pll_setup(void)
|
|
|
|
{
|
|
|
|
uint32_t cfgr;
|
2019-11-25 22:00:30 +03:00
|
|
|
if (!CONFIG_STM32_CLOCK_REF_INTERNAL) {
|
|
|
|
// Configure 48Mhz PLL from external crystal (HSE)
|
|
|
|
uint32_t div = CONFIG_CLOCK_FREQ / CONFIG_CLOCK_REF_FREQ;
|
2021-01-18 20:47:59 +03:00
|
|
|
RCC->CR = ((RCC->CR & ~RCC_CR_HSITRIM) | RCC_CR_HSEON
|
|
|
|
| (CONFIG_STM32F0_TRIM << RCC_CR_HSITRIM_Pos));
|
2019-11-25 22:00:30 +03:00
|
|
|
cfgr = RCC_CFGR_PLLSRC_HSE_PREDIV | ((div - 2) << RCC_CFGR_PLLMUL_Pos);
|
2019-09-16 18:45:36 +03:00
|
|
|
} else {
|
|
|
|
// Configure 48Mhz PLL from internal 8Mhz oscillator (HSI)
|
2019-11-25 22:00:30 +03:00
|
|
|
uint32_t div2 = (CONFIG_CLOCK_FREQ / 8000000) * 2;
|
|
|
|
cfgr = RCC_CFGR_PLLSRC_HSI_DIV2 | ((div2 - 2) << RCC_CFGR_PLLMUL_Pos);
|
2019-09-16 18:45:36 +03:00
|
|
|
}
|
|
|
|
RCC->CFGR = cfgr;
|
|
|
|
RCC->CR |= RCC_CR_PLLON;
|
|
|
|
|
|
|
|
// Wait for PLL lock
|
|
|
|
while (!(RCC->CR & RCC_CR_PLLRDY))
|
|
|
|
;
|
|
|
|
|
|
|
|
// Switch system clock to PLL
|
|
|
|
RCC->CFGR = cfgr | RCC_CFGR_SW_PLL;
|
|
|
|
while ((RCC->CFGR & RCC_CFGR_SWS_Msk) != RCC_CFGR_SWS_PLL)
|
|
|
|
;
|
|
|
|
|
2019-11-14 06:20:29 +03:00
|
|
|
// Setup CFGR3 register
|
|
|
|
uint32_t cfgr3 = RCC_CFGR3_I2C1SW;
|
2019-09-16 18:45:36 +03:00
|
|
|
if (CONFIG_USBSERIAL)
|
2019-11-14 06:20:29 +03:00
|
|
|
// Select PLL as source for USB clock
|
|
|
|
cfgr3 |= RCC_CFGR3_USBSW;
|
|
|
|
RCC->CFGR3 = cfgr3;
|
2019-09-16 18:45:36 +03:00
|
|
|
}
|
|
|
|
|
2019-09-17 04:39:29 +03:00
|
|
|
// Configure and enable internal 48Mhz clock on the stm32f042
|
|
|
|
static void
|
|
|
|
hsi48_setup(void)
|
|
|
|
{
|
|
|
|
#if CONFIG_MACH_STM32F042
|
|
|
|
// Enable HSI48
|
|
|
|
RCC->CR2 |= RCC_CR2_HSI48ON;
|
|
|
|
while (!(RCC->CR2 & RCC_CR2_HSI48RDY))
|
|
|
|
;
|
|
|
|
|
|
|
|
// Switch system clock to HSI48
|
|
|
|
RCC->CFGR = RCC_CFGR_SW_HSI48;
|
|
|
|
while ((RCC->CFGR & RCC_CFGR_SWS_Msk) != RCC_CFGR_SWS_HSI48)
|
|
|
|
;
|
|
|
|
|
|
|
|
// Enable USB clock recovery
|
|
|
|
if (CONFIG_USBSERIAL) {
|
|
|
|
enable_pclock(CRS_BASE);
|
|
|
|
CRS->CR |= CRS_CR_AUTOTRIMEN | CRS_CR_CEN;
|
|
|
|
}
|
2019-11-14 06:20:29 +03:00
|
|
|
|
|
|
|
// Setup I2C1 clock
|
|
|
|
RCC->CFGR3 = RCC_CFGR3_I2C1SW;
|
2019-09-17 04:39:29 +03:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2019-11-05 18:08:32 +03:00
|
|
|
// Enable high speed internal 14Mhz clock for ADC
|
|
|
|
static void
|
|
|
|
hsi14_setup(void)
|
|
|
|
{
|
|
|
|
// Enable HSI14 for ADC
|
|
|
|
RCC->CR2 = RCC_CR2_HSI14ON;
|
|
|
|
while (!(RCC->CR2 & RCC_CR2_HSI14RDY))
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
2019-10-24 18:35:18 +03:00
|
|
|
// Main entry point - called from armcm_boot.c:ResetHandler()
|
2019-09-16 18:45:36 +03:00
|
|
|
void
|
2019-10-24 18:35:18 +03:00
|
|
|
armcm_main(void)
|
2019-09-16 18:45:36 +03:00
|
|
|
{
|
2019-10-27 00:11:04 +03:00
|
|
|
if (CONFIG_USBSERIAL && CONFIG_MACH_STM32F042
|
|
|
|
&& *(uint64_t*)USB_BOOT_FLAG_ADDR == USB_BOOT_FLAG) {
|
|
|
|
*(uint64_t*)USB_BOOT_FLAG_ADDR = 0;
|
|
|
|
uint32_t *sysbase = (uint32_t*)0x1fffc400;
|
|
|
|
asm volatile("mov sp, %0\n bx %1"
|
|
|
|
: : "r"(sysbase[0]), "r"(sysbase[1]));
|
|
|
|
}
|
|
|
|
|
2019-10-24 18:35:18 +03:00
|
|
|
SystemInit();
|
|
|
|
|
2019-12-10 05:03:05 +03:00
|
|
|
if (CONFIG_ARMCM_RAM_VECTORTABLE)
|
|
|
|
enable_ram_vectortable();
|
|
|
|
|
2019-09-16 18:45:36 +03:00
|
|
|
// Set flash latency
|
|
|
|
FLASH->ACR = (1 << FLASH_ACR_LATENCY_Pos) | FLASH_ACR_PRFTBE;
|
|
|
|
|
|
|
|
// Configure main clock
|
2020-12-08 17:00:47 +03:00
|
|
|
if (CONFIG_MACH_STM32F042 && CONFIG_STM32_CLOCK_REF_INTERNAL
|
|
|
|
&& CONFIG_USBSERIAL)
|
2019-09-17 04:39:29 +03:00
|
|
|
hsi48_setup();
|
|
|
|
else
|
|
|
|
pll_setup();
|
2019-10-16 00:45:48 +03:00
|
|
|
|
2019-11-05 18:08:32 +03:00
|
|
|
// Turn on hsi14 oscillator for ADC
|
|
|
|
hsi14_setup();
|
|
|
|
|
2020-06-25 01:59:38 +03:00
|
|
|
// Support pin remapping USB/CAN pins on low pinout stm32f042
|
2019-10-16 00:45:48 +03:00
|
|
|
#ifdef SYSCFG_CFGR1_PA11_PA12_RMP
|
2019-12-28 14:01:58 +03:00
|
|
|
if (CONFIG_STM32F042_PIN_SWAP) {
|
2019-10-16 00:45:48 +03:00
|
|
|
enable_pclock(SYSCFG_BASE);
|
|
|
|
SYSCFG->CFGR1 |= SYSCFG_CFGR1_PA11_PA12_RMP;
|
|
|
|
}
|
|
|
|
#endif
|
2019-10-24 18:35:18 +03:00
|
|
|
|
|
|
|
sched_main();
|
2019-09-16 18:45:36 +03:00
|
|
|
}
|