You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

180 lines
6.6 KiB
C

/*****************************************************************************
* BSP for NUCLEO-C031C6 with uC/OS-II RTOS
*****************************************************************************/
#include "uc_ao.h" /* uC/AO API */
#include "bsp.h" /* Board Support Package */
#include <stdbool.h> /* needed by the TI drivers */
#include "stm32c0xx.h" /* CMSIS-compliant header file for the MCU used */
/* add other drivers if necessary... */
// LED marked "LD4" (PA.5) on the NUCLEO-C031C6 board
#define LD4_PIN 5U
// external LED to be inserted between GND (short leg) and
// D12 (longer leg) on the CN9 connector
#define LD5_PIN 6U
// Button B1 (PC.13) on the NUCLEO-C031C6 board
#define B1_PIN 13U
/* uCOS-II application hooks ===============================================*/
void App_TimeTickHook(void) {
/* state of the button debouncing, see below */
static struct ButtonsDebouncing {
uint32_t depressed;
uint32_t previous;
} buttons = { 0U, 0U };
uint32_t current;
uint32_t tmp;
TimeEvent_tick(); /* process all uC/AO time events */
/* Perform the debouncing of buttons. The algorithm for debouncing
* adapted from the book "Embedded Systems Dictionary" by Jack Ganssle
* and Michael Barr, page 71.
*/
current = ~GPIOC->IDR; // read Port C with state of Button B1
tmp = buttons.depressed; /* save the debounced depressed buttons */
buttons.depressed |= (buttons.previous & current); /* set depressed */
buttons.depressed &= (buttons.previous | current); /* clear released */
buttons.previous = current; /* update the history */
tmp ^= buttons.depressed; /* changed debounced depressed */
if ((tmp & (1U << B1_PIN)) != 0U) { // debounced B1 state changed?
if ((current & (1U << B1_PIN)) != 0U) { // is B1 depressed?
/* post the "button-pressed" event */
static Event const buttonPressedEvt = {BUTTON_PRESSED_SIG};
Active_post(AO_BlinkyButton, &buttonPressedEvt);
}
else { /* the button is released */
/* post the "button-released" event */
static Event const buttonReleasedEvt = {BUTTON_RELEASED_SIG};
Active_post(AO_BlinkyButton, &buttonReleasedEvt);
}
}
}
/*..........................................................................*/
void App_TaskIdleHook(void) {
#ifdef NDEBUG
/* Put the CPU and peripherals to the low-power mode.
* you might need to customize the clock management for your application,
* see the datasheet for your particular Cortex-M3 MCU.
*/
__WFI(); /* Wait-For-Interrupt */
#endif
}
/*..........................................................................*/
void App_TaskCreateHook (OS_TCB *ptcb) { (void)ptcb; }
void App_TaskDelHook (OS_TCB *ptcb) { (void)ptcb; }
void App_TaskReturnHook (OS_TCB *ptcb) { (void)ptcb; }
void App_TaskStatHook (void) {}
void App_TaskSwHook (void) {}
void App_TCBInitHook (OS_TCB *ptcb) { (void)ptcb; }
/* BSP functions ===========================================================*/
void BSP_init(void) {
/* NOTE: SystemInit() has been already called from the startup code
* but SystemCoreClock needs to be updated
*/
SystemCoreClockUpdate();
// enable GPIOA clock port for the LEDs
RCC->IOPENR |= (1U << 0U);
// NUCLEO-C031C6 board has LED LD4 on GPIOA pin LD4_PIN
// and external LED LD5 on GPIO LD5_PIN
// set the LED pins as push-pull output, no pull-up, pull-down
GPIOA->MODER &= ~((3U << 2U*LD4_PIN) | (3U << 2U*LD5_PIN));
GPIOA->MODER |= ((1U << 2U*LD4_PIN) | (1U << 2U*LD5_PIN));
GPIOA->OTYPER &= ~((1U << LD4_PIN) | (1U << LD5_PIN));
GPIOA->OSPEEDR &= ~((3U << 2U*LD4_PIN) | (3U << 2U*LD5_PIN));
GPIOA->OSPEEDR |= ((1U << 2U*LD4_PIN) | (1U << 2U*LD5_PIN));
GPIOA->PUPDR &= ~((3U << 2U*LD4_PIN) | (3U << 2U*LD5_PIN));
// enable GPIOC clock port for the Button B1
RCC->IOPENR |= (1U << 2U);
// configure Button B1 (PC.13) pins as input, no pull-up, pull-down
GPIOC->MODER &= ~(3U << 2*B1_PIN);
GPIOC->OSPEEDR &= ~(3U << 2*B1_PIN);
GPIOC->OSPEEDR |= (1U << 2*B1_PIN);
GPIOC->PUPDR &= ~(3U << 2*B1_PIN);
// configure Button B1 interrupt as falling edge
EXTI->EMR1 &= ~(1U << B1_PIN);
EXTI->IMR1 |= (1U << B1_PIN);
EXTI->RTSR1 &= ~(1U << B1_PIN);
EXTI->FTSR1 |= (1U << B1_PIN);
EXTI->EXTICR[3] &= ~(7U << 8); // EXTI port C line 13
EXTI->EXTICR[3] |= (2U << 8); // EXTI port C line 13
}
/*..........................................................................*/
void BSP_start(void) {
/* NOTE: SystemInit() has been already called from the startup code
* but SystemCoreClock needs to be updated
*/
SystemCoreClockUpdate();
/* set up the SysTick timer to fire at BSP_TICKS_PER_SEC rate
* NOTE: do NOT call OS_CPU_SysTickInit() from uC/OS-II
*/
SysTick_Config(SystemCoreClock / OS_TICKS_PER_SEC);
/* set priorities of ALL ISRs used in the system, see NOTE1 */
NVIC_SetPriority(SysTick_IRQn, CPU_CFG_KA_IPL_BOUNDARY + 1U);
/* ... */
/* enable IRQs in the NVIC... */
/* ... */
}
/*..........................................................................*/
void BSP_ledRedOn(void) {
/* no red LED on board */
}
/*..........................................................................*/
void BSP_ledRedOff(void) {
/* no red LED on board */
}
/*..........................................................................*/
void BSP_ledBlueOn(void) {
/* no blue LED on board */
}
/*..........................................................................*/
void BSP_ledBlueOff(void) {
/* no blue LED on board */
}
/*..........................................................................*/
void BSP_ledGreenOn(void) {
GPIOA->BSRR = (1U << LD4_PIN); /* turn LED on */
}
/*..........................................................................*/
void BSP_ledGreenOff(void) {
GPIOA->BSRR = (1U << (LD4_PIN + 16U)); /* turn LED off */
}
//............................................................................
Q_NORETURN Q_onAssert(char const * const module, int const id) {
(void)module; // unused parameter
(void)id; // unused parameter
#ifndef NDEBUG
GPIOA->BSRR = (1U << LD4_PIN); // turn LED on
// for debugging, hang on in an endless loop...
for (;;) {
}
#endif
NVIC_SystemReset();
}
// error-handling function called by exception handlers in the startup code
Q_NORETURN assert_failed(char const * const module, int const id);
Q_NORETURN assert_failed(char const * const module, int const id) {
Q_onAssert(module, id);
}