Skip to content

dragon-dreamer/mcutl

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Modern C++17 template header-only library for microcontrollers (MCUTL)

What's this?

This is a modern C++17 header-only template library for microcontrollers. The basic idea is to provide access to microcontroller features and peripherals via a uniform interface and precompute as much as possible during compilation. In other words, it is a thin template-based highly optimized HAL (hardware abstraction layer). Bonus: it supports unit-testing!

As this is my hobby project, it currently supports some features of Cortex-M3 STM32F101, STM32F102, STM32F103 and partially STM32F104 and STM32F105 controllers only, but I may add more in the future. The library does not guarantee stability of its interface, as this is a work in progress. However, there are already some remarkable pieces of code, so you may want to use them or learn them anyway.

Full documentation

Can be found here.

MCUTL features compared to any other MCU library

MCUTL Any other MCU library
Header-only Requires compilation
Compiles and includes to binary only the code which is required to do what you ask Compiles full-blown bodies of functions with branches for every occasion, which leads to larger and slower code
Performs compile-time checks so you aren't able to compile ridiculous code (like settings incorrect clock prescalers) You can easily pass some invalid arguments to functions, most of them are not checked
Precomputes everything that's possible, minimizing code size and increasing its speed Precomputes a few lines of code using macros, but most of calculations are run-time
Unit-testable on the host PC, provides mocks for memory reads and writes and MCU-specific instructions. You can test at low level (memory reads/writes) both the library and the code which utilizes it Well, no, you have to mock everything yourself
Heavily relies on compiler optimizations, debug builds may be huge Doesn't depend on compiler optimizations that much
Requires C++17 Nope

The MCUTL library is a thin layer over hardware with testing support. The library does not account for any RTOS and does not lock any mutexes or resources. You have to do this yourself when writing some shared memory or registers.

Compiler support

Currently supported and tested compilers are G++ 8.4 and newer and Clang 7.0 and newer.

Sweet example

Suppose you'd like to configure clocks of your STM32F103xxxx microcontroller. With MCUTL you can do this:

// Use external 16 MHz crystal, set core frequency to 72 MHz,
// set some requirements for SPI1 frequency, and also request valid
// USB peripheral frequency (which should be 48 MHz for STM32F103).
using clock_config = mcutl::clock::config<
	mcutl::clock::external_high_speed_crystal<16_MHz>,
	mcutl::clock::core<mcutl::clock::required_frequency<72_MHz>>,
	mcutl::clock::spi1<mcutl::clock::min_frequency<100_KHz>, mcutl::clock::max_frequency<200_KHz>>,
	mcutl::clock::provide_usb_frequency,
	mcutl::clock::base_configuration_is_currently_present
>;

mcutl::clock::configure_clocks<clock_config>();

This will:

  • Check all the prerequisites and display a readable compile-time error if it's not possible to configure MCU clocks in a way requested. For example, if it's not possible to configure the core to run with 72 MHz frequency with SPI1 from 100 to 200 KHz frequency and USB peripheral enabled at the same time, you'll know this when compiling.
  • Compile the code to perform actions exactly required to achieve the configuration requested. No excessive checks, no code that's not needed. You don't need to generate the configuration code using STM32CubeMX, you just get what you see.
  • Bonus: if you later wish to change your 16 MHz crystal to a 12 MHz one, you just change a single character in your configuration and recompile the code!

This code takes 124 bytes after compilation with -Os switch or 144 bytes with -O3. It operates directly on registers required and doesn't do anything that can be done at compile time.

This is roughly the equivalent C code you could write to achive the same thing (and this is the actual code which is generated for the C++ code above):

auto cfgr = RCC->CFGR;
RCC->CR |= RCC_CR_HSEON;
while (!(RCC->CR & RCC_CR_HSERDY)) {}

cfgr &= ~(RCC_CFGR_PPRE1 | RCC_CFGR_PPRE2 | RCC_CFGR_ADCPRE | RCC_CFGR_HPRE | RCC_CFGR_SW
	| RCC_CFGR_PLLMULL | RCC_CFGR_PLLXTPRE_HSE_DIV2 | RCC_CFGR_PLLSRC | RCC_CFGR_USBPRE);
cfgr |= RCC_CFGR_PPRE1_DIV2 | RCC_CFGR_PPRE2_DIV2 | RCC_CFGR_ADCPRE_DIV4 | RCC_CFGR_PLLMULL9
	| RCC_CFGR_PLLXTPRE_HSE_DIV2 | RCC_CFGR_PLLSRC;
RCC->CFGR = cfgr;

RCC->CR |= RCC_CR_PLLON;
while (!(RCC->CR & RCC_CR_PLLRDY)) {}

auto flash_acr = FLASH->ACR;
flash_acr &= ~FLASH_ACR_LATENCY;
flash_acr |= FLASH_ACR_LATENCY_1;
FLASH->ACR = flash_acr;

cfgr = RCC->CFGR;
cfgr &= ~RCC_CFGR_SW;
cfgr |= RCC_CFGR_SW_PLL;
RCC->CFGR = cfgr;
while ((RCC->CFGR & RCC_CFGR_SWS) == RCC_CFGR_SWS_HSI) {}

RCC->CR &= ~RCC_CR_HSION;

MCU support

MCU \\ Feature Peripheral configClock configGPIOInterruptsExternal interrupts Hardware CRCBackup domainLow-power modesDMASYSTICKReal-time clockADCTimersSPI
STM32F101, STM32F102, STM32F103 Full (enable, disable, reset, clear reset) Yes, except Clock Security System (CSS), clock-out capability, Watchdog clock Yes, except locking and remapping Yes, except faults, vector table relocation or editing Yes Yes Yes Yes Yes Yes Yes Partial Partial Partial
STM32F105, STM32F107 No

Releases

No releases published

Packages

No packages published