1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/drivers/pulse_counter/pulse_counter.c
Marian Buschsieweke 2d81738cb9
drivers/pulse_counter: Use C11 atomics & bugfix
- 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.)
2019-11-26 14:04:42 +01:00

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);
}