mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-15 18:32:45 +01:00
892370121d
cpu/stm32/qdec: test if callback pointer is set Callback pointer is not tested and could result in a hard fault if the pointer is NULL. Thus only activate interrupt if a callback provided. Signed-off-by: Gilles DOFFE <g.doffe@gmail.com>
230 lines
4.7 KiB
C
230 lines
4.7 KiB
C
/*
|
|
* Copyright (C) 2017 Gilles DOFFE <gdoffe@gmail.com>
|
|
*
|
|
* 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 cpu_cortexm_common
|
|
* @ingroup drivers_periph_qdec
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level QDEC driver implementation
|
|
*
|
|
* @author Gilles DOFFE <gilles.doffe@gmail.com>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <errno.h>
|
|
|
|
#include "cpu.h"
|
|
#include "assert.h"
|
|
#include "periph/qdec.h"
|
|
#include "periph/gpio.h"
|
|
|
|
#ifdef QDEC_NUMOF
|
|
|
|
/**
|
|
* @brief Interrupt context for each configured qdec
|
|
*/
|
|
static qdec_isr_ctx_t isr_ctx[QDEC_NUMOF];
|
|
|
|
/**
|
|
* @brief Read the current value of the given qdec device. Internal use.
|
|
*
|
|
* @param[in] dev the qdec to read the current value from
|
|
* @param[in] dev perform a reset of qdec counter if not 0
|
|
*
|
|
* @return the qdecs current value
|
|
*/
|
|
static int32_t _qdec_read(qdec_t qdec, uint8_t reset);
|
|
|
|
static inline TIM_TypeDef *dev(qdec_t qdec)
|
|
{
|
|
return qdec_config[qdec].dev;
|
|
}
|
|
|
|
int32_t qdec_init(qdec_t qdec, qdec_mode_t mode, qdec_cb_t cb, void *arg)
|
|
{
|
|
/* Control variables */
|
|
uint8_t i = 0;
|
|
|
|
/* Verify parameters */
|
|
assert((qdec < QDEC_NUMOF));
|
|
|
|
/* Power on the used timer */
|
|
periph_clk_en(qdec_config[qdec].bus, qdec_config[qdec].rcc_mask);
|
|
|
|
/* Reset configuration and CC channels */
|
|
dev(qdec)->CR1 = 0;
|
|
dev(qdec)->CR2 = 0;
|
|
dev(qdec)->SMCR = 0;
|
|
dev(qdec)->CCER = 0;
|
|
for (i = 0; i < QDEC_CHAN; i++) {
|
|
TIM_CHAN(qdec, i) = 0;
|
|
}
|
|
|
|
/* Count on A (TI1) signal edges, B (TI2) signal edges or both,
|
|
* default to EINVAL (Invalid argument).
|
|
*/
|
|
switch (mode) {
|
|
/* X2 mode */
|
|
case QDEC_X2:
|
|
dev(qdec)->SMCR |= (0x02 << TIM_SMCR_SMS_Pos);
|
|
break;
|
|
/* X4 mode */
|
|
case QDEC_X4:
|
|
dev(qdec)->SMCR |= (0x03 << TIM_SMCR_SMS_Pos);
|
|
break;
|
|
/* X1 mode does not exist on STM32 as STM32 always counts
|
|
* on both rising and falling edges from encoder
|
|
*/
|
|
case QDEC_X1:
|
|
default:
|
|
errno = EINVAL;
|
|
goto err_invalid_mode;
|
|
}
|
|
|
|
/* Reset configuration and CC channels */
|
|
for (i = 0; i < QDEC_CHAN; i++) {
|
|
TIM_CHAN(qdec, i) = 0;
|
|
}
|
|
|
|
/* Configure the used pins */
|
|
i = 0;
|
|
while ((i < QDEC_CHAN) && (qdec_config[qdec].chan[i].pin != GPIO_UNDEF)) {
|
|
gpio_init(qdec_config[qdec].chan[i].pin, GPIO_IN);
|
|
#ifndef CPU_FAM_STM32F1
|
|
gpio_init_af(qdec_config[qdec].chan[i].pin, qdec_config[qdec].af);
|
|
#endif
|
|
i++;
|
|
}
|
|
|
|
/* Set counting max value */
|
|
dev(qdec)->ARR = qdec_config[qdec].max;
|
|
|
|
/* Set TIMx_CNT value to half of of TIMx_ARR to permit countdown */
|
|
dev(qdec)->CNT = dev(qdec)->ARR / 2;
|
|
|
|
/* Remember the interrupt context */
|
|
isr_ctx[qdec].cb = cb;
|
|
isr_ctx[qdec].arg = arg;
|
|
|
|
/* Enable the qdec's interrupt only if there is a callback provided */
|
|
if (cb) {
|
|
NVIC_EnableIRQ(qdec_config[qdec].irqn);
|
|
dev(qdec)->DIER |= TIM_DIER_UIE;
|
|
} else {
|
|
dev(qdec)->DIER &= ~TIM_DIER_UIE;
|
|
}
|
|
|
|
/* Reset counter and start qdec */
|
|
qdec_start(qdec);
|
|
|
|
return 0;
|
|
|
|
/* Error management */
|
|
err_invalid_mode:
|
|
return errno;
|
|
}
|
|
|
|
inline int32_t qdec_read(qdec_t qdec)
|
|
{
|
|
return _qdec_read(qdec, false);
|
|
}
|
|
|
|
inline int32_t qdec_read_and_reset(qdec_t qdec)
|
|
{
|
|
return _qdec_read(qdec, true);
|
|
}
|
|
|
|
static int32_t _qdec_read(qdec_t qdec, uint8_t reset)
|
|
{
|
|
int32_t count = 0;
|
|
uint32_t irq_save = 0;
|
|
|
|
/* Protect critical section */
|
|
irq_save = irq_disable();
|
|
|
|
/* Get counter value */
|
|
count = dev(qdec)->CNT;
|
|
|
|
/* Reset counter if asked */
|
|
if (reset)
|
|
{
|
|
dev(qdec)->CNT = dev(qdec)->ARR / 2;
|
|
}
|
|
|
|
/* Restore IRQ */
|
|
irq_restore(irq_save);
|
|
|
|
/* Subtract offset before return */
|
|
count -= dev(qdec)->ARR / 2;
|
|
|
|
/* Return count minus offset */
|
|
return count;
|
|
}
|
|
|
|
void qdec_start(qdec_t qdec)
|
|
{
|
|
dev(qdec)->CR1 |= TIM_CR1_CEN;
|
|
}
|
|
|
|
void qdec_stop(qdec_t qdec)
|
|
{
|
|
dev(qdec)->CR1 &= ~TIM_CR1_CEN;
|
|
}
|
|
|
|
static inline void irq_handler(qdec_t qdec)
|
|
{
|
|
uint32_t status = (dev(qdec)->SR & dev(qdec)->DIER);
|
|
|
|
if (status & (TIM_SR_UIF)) {
|
|
dev(qdec)->SR &= ~(TIM_SR_UIF);
|
|
isr_ctx[qdec].cb(isr_ctx[qdec].arg);
|
|
}
|
|
cortexm_isr_end();
|
|
}
|
|
|
|
|
|
#ifdef QDEC_0_ISR
|
|
void QDEC_0_ISR(void)
|
|
{
|
|
irq_handler(0);
|
|
}
|
|
#endif
|
|
|
|
#ifdef QDEC_1_ISR
|
|
void QDEC_1_ISR(void)
|
|
{
|
|
irq_handler(1);
|
|
}
|
|
#endif
|
|
|
|
#ifdef QDEC_2_ISR
|
|
void QDEC_2_ISR(void)
|
|
{
|
|
irq_handler(2);
|
|
}
|
|
#endif
|
|
|
|
#ifdef QDEC_3_ISR
|
|
void QDEC_3_ISR(void)
|
|
{
|
|
irq_handler(3);
|
|
}
|
|
#endif
|
|
|
|
#ifdef QDEC_4_ISR
|
|
void QDEC_4_ISR(void)
|
|
{
|
|
irq_handler(4);
|
|
}
|
|
#endif
|
|
|
|
#endif /* QDEC_NUMOF */
|