mirror of
https://github.com/cesanta/mongoose.git
synced 2025-01-11 16:00:11 +08:00
40 lines
2.1 KiB
C
40 lines
2.1 KiB
C
// Copyright (c) 2018-2021 Cesanta Software Limited
|
|
// All rights reserved
|
|
// TRM: https://www.st.com/resource/en/reference_manual/cd00171190.pdf
|
|
// clang-format off
|
|
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
|
|
static inline void spin(uint32_t count) { while (count--) asm("nop"); }
|
|
static inline void setreg(volatile uint32_t *r, uint32_t clear_mask, uint32_t set_mask) { *r &= ~clear_mask; *r |= set_mask; }
|
|
|
|
#define BIT(x) ((uint32_t) 1 << (x))
|
|
#define PIN(bank, num) ((((bank) - 'A') << 8) | (num))
|
|
|
|
// RCC registers, TRM section 7.3, memory map section 3.3
|
|
struct rcc { volatile uint32_t CR, CFGR, CIR, APB2RSTR, APB1RSTR, AHBENR, APB2ENR, APB1ENR, BDCR, CSR; };
|
|
#define RCC ((struct rcc *) 0x40021000)
|
|
|
|
static inline void init_ram(void) {
|
|
extern uint32_t _bss_start, _bss_end;
|
|
extern uint32_t _data_start, _data_end, _data_flash_start;
|
|
memset(&_bss_start, 0, ((char *) &_bss_end - (char *) &_bss_start));
|
|
memcpy(&_data_start, &_data_flash_start, ((char *) &_data_end - (char *) &_data_start));
|
|
}
|
|
|
|
// GPIO registers, TRM section 9.2, memory map section 3.3, regs 9.1.0
|
|
enum { GPIO_OUT_PP, GPIO_OUT_OD, GPIO_AF_PP, GPIO_AF_OD, GPIO_ANA = 0, GPIO_FLOATING, GPIO_PULLED }; // 9.1.0
|
|
enum { GPIO_IN, GPIO_OUT_10MHZ, GPIO_OUT_2MHZ, GPIO_OUT_50MHZ }; // 9.1.0
|
|
enum { OUTPUT = (GPIO_OUT_PP << 2) | GPIO_OUT_2MHZ, INPUT = (GPIO_FLOATING << 2) | GPIO_IN, INPUT_PULLUP = (GPIO_PULLED << 2) | GPIO_IN};
|
|
struct gpio { volatile uint32_t CRL, CRH, IDR, ODR, BSRR, BRR, LCKR; };
|
|
static inline struct gpio *gpio_bank(uint16_t pin) { return (struct gpio *) (0x40010800 + 0x400 * (pin >> 8)); }
|
|
static inline void gpio_on(uint16_t pin) { gpio_bank(pin)->ODR |= BIT(pin & 255); }
|
|
static inline void gpio_off(uint16_t pin) { gpio_bank(pin)->ODR &= ~BIT(pin & 255); }
|
|
static inline void gpio_toggle(uint16_t pin) { gpio_bank(pin)->ODR ^= BIT(pin & 255); }
|
|
static inline void gpio_init(uint16_t pin, uint32_t mode) {
|
|
uint8_t n = pin & 255, shift = (n < 8 ? n : (n - 8)) * 4;
|
|
volatile uint32_t *reg = n < 8 ? &gpio_bank(pin)->CRL : &gpio_bank(pin)->CRH;
|
|
setreg(reg, 15 << shift, mode << shift);
|
|
}
|
|
// clang-format on
|