mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
2d81738cb9
- Previously the pulse counter used GCC's built-in functions for atomic memory access. This PR changes this to use C11 atomics instead: - Use of C11 atomics instead of atomic build-in functions makes the code more portable and readable. - Previously pulse_counter_reset() did not use an atomic function, so that resets would be racy. (E.g. on the 8-bit AVR platform an 16-bit store is not atomic, unless special care (like using C11 atomics) is taken.)
73 lines
1.6 KiB
C
73 lines
1.6 KiB
C
/*
|
|
* Copyright (C) 2017 UC Berkeley
|
|
*
|
|
* 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 drivers_pulse_counter
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Driver for the PULSE COUNTER.
|
|
*
|
|
* @author Hyung-Sin Kim <hs.kim@cs.berkeley.edu>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include "pulse_counter_params.h"
|
|
#include "pulse_counter.h"
|
|
|
|
#define ENABLE_DEBUG (0)
|
|
#include "debug.h"
|
|
|
|
/* Accumulate pulse count */
|
|
static void pulse_counter_trigger(void *arg)
|
|
{
|
|
pulse_counter_t *dev = (pulse_counter_t *)arg;
|
|
|
|
/* Use C11 atomic operations to avoid messing with IRQ flags */
|
|
atomic_fetch_add(&(dev->pulse_count), 1);
|
|
}
|
|
|
|
/* Initialize pulse counter */
|
|
int pulse_counter_init(pulse_counter_t *dev, const pulse_counter_params_t *params)
|
|
{
|
|
gpio_mode_t gpio_mode;
|
|
if (params->gpio_flank == GPIO_FALLING) {
|
|
gpio_mode = GPIO_IN_PU;
|
|
}
|
|
else {
|
|
gpio_mode = GPIO_IN_PD;
|
|
}
|
|
|
|
if (gpio_init_int(params->gpio, gpio_mode, params->gpio_flank, pulse_counter_trigger, dev)) {
|
|
return -1;
|
|
}
|
|
|
|
atomic_init(&dev->pulse_count, 0);
|
|
return 0;
|
|
}
|
|
|
|
/* Return the accumulated pulse counts and reset the count to zero */
|
|
int16_t pulse_counter_read_with_reset(pulse_counter_t *dev)
|
|
{
|
|
return atomic_exchange(&(dev->pulse_count), 0);
|
|
}
|
|
|
|
/* Return the accumulated pulse counts */
|
|
int16_t pulse_counter_read_without_reset(pulse_counter_t *dev)
|
|
{
|
|
return atomic_load(&dev->pulse_count);
|
|
}
|
|
|
|
/* Reset the pulse count value to zero */
|
|
void pulse_counter_reset(pulse_counter_t *dev)
|
|
{
|
|
atomic_store(&(dev->pulse_count), 0);
|
|
}
|