diff --git a/cpu/nrf51/include/periph_cpu.h b/cpu/nrf51/include/periph_cpu.h index 251bd319c7166af7f5b2e14b8fc4c46791740e84..bc368152fd0f77a78bb25951847405a0bf4f189b 100644 --- a/cpu/nrf51/include/periph_cpu.h +++ b/cpu/nrf51/include/periph_cpu.h @@ -54,6 +54,21 @@ typedef enum { } i2c_speed_t; /** @} */ +/** + * @brief Override ADC resolution values + * @{ + */ +#define HAVE_ADC_RES_T +typedef enum { + ADC_RES_6BIT = 0xf0, /**< ADC resolution: 6 bit (not supported) */ + ADC_RES_8BIT = 0x00, /**< ADC resolution: 8 bit */ + ADC_RES_10BIT = 0x02, /**< ADC resolution: 10 bit */ + ADC_RES_12BIT = 0xf1, /**< ADC resolution: 12 bit (not supported) */ + ADC_RES_14BIT = 0xf2, /**< ADC resolution: 14 bit (not supported) */ + ADC_RES_16BIT = 0xf3 /**< ADC resolution: 16 bit (not supported) */ +} adc_res_t; +/** @} */ + /** * @brief I2C (TWI) configuration options */ diff --git a/cpu/nrf52/Makefile.features b/cpu/nrf52/Makefile.features index 7d12728b90a68dc1f4ce134b53fd28fe4cb22e6b..56ecba5a416cb9b33fafce76076a583a17a5805d 100644 --- a/cpu/nrf52/Makefile.features +++ b/cpu/nrf52/Makefile.features @@ -1 +1,4 @@ +# The ADC does not depend on any board configuration, so always available +FEATURES_PROVIDED += periph_adc + -include $(RIOTCPU)/nrf5x_common/Makefile.features diff --git a/cpu/nrf52/include/periph_cpu.h b/cpu/nrf52/include/periph_cpu.h index 38c7726fc95c6637cac5abcf571664fafd3d83d5..bac6b6487e7c8d310ded0c09efe5b846338108a8 100644 --- a/cpu/nrf52/include/periph_cpu.h +++ b/cpu/nrf52/include/periph_cpu.h @@ -40,6 +40,41 @@ extern "C" { #define SPI_MISOSEL (dev(bus)->PSEL.MISO) /** @} */ +/** + * @brief The nRF52 family of CPUs provides a fixed number of 9 ADC lines + */ +#define ADC_NUMOF (9U) + +/** + * @brief nRF52 specific naming of ADC lines (for convenience) + */ +enum { + NRF52_AIN0 = 0, /**< Analog Input 0 */ + NRF52_AIN1 = 1, /**< Analog Input 1 */ + NRF52_AIN2 = 2, /**< Analog Input 2 */ + NRF52_AIN3 = 3, /**< Analog Input 3 */ + NRF52_AIN4 = 4, /**< Analog Input 4 */ + NRF52_AIN5 = 5, /**< Analog Input 5 */ + NRF52_AIN6 = 6, /**< Analog Input 6 */ + NRF52_AIN7 = 7, /**< Analog Input 7 */ + NRF52_VDD = 8, /**< VDD, not useful if VDD is reference... */ +}; + +/** + * @brief Override ADC resolution values + * @{ + */ +#define HAVE_ADC_RES_T +typedef enum { + ADC_RES_6BIT = 0xf0, /**< not supported by hardware */ + ADC_RES_8BIT = 0x00, /**< ADC resolution: 8 bit */ + ADC_RES_10BIT = 0x01, /**< ADC resolution: 10 bit */ + ADC_RES_12BIT = 0x02, /**< ADC resolution: 12 bit */ + ADC_RES_14BIT = 0xf1, /**< supported with oversampling, not implemented */ + ADC_RES_16BIT = 0xf2 /**< not supported by hardware */ +} adc_res_t; +/** @} */ + #ifdef __cplusplus } #endif diff --git a/cpu/nrf52/periph/adc.c b/cpu/nrf52/periph/adc.c new file mode 100644 index 0000000000000000000000000000000000000000..7ecd219784d1a7780004e7b07b2e4c2e0139cdb7 --- /dev/null +++ b/cpu/nrf52/periph/adc.c @@ -0,0 +1,146 @@ +/* + * Copyright (C) 2017 HAW Hamburg + * 2017 Freie Universität Berlin + * + * 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_nrf52 + * @{ + * + * @file + * @brief Low-level ADC driver implementation + * + * @author Dimitri Nahm <dimitri.nahm@haw-hamburg.de> + * @author Hauke Petersen <hauke.petersen@fu-berlin.de> + * + * @} + */ + +#include "cpu.h" +#include "mutex.h" +#include "periph/adc.h" +#include "periph_conf.h" + +/** + * @name Default ADC reference, gain configuration and acquisition time + * + * Can be overridden by the board configuration if needed. The default + * configuration uses the full VDD (typically 3V3) as reference and samples for + * 10us. + * @{ + */ +#ifndef ADC_REF +#define ADC_REF SAADC_CH_CONFIG_REFSEL_VDD1_4 +#endif +#ifndef ADC_GAIN +#define ADC_GAIN SAADC_CH_CONFIG_GAIN_Gain1_4 +#endif +#ifndef ADC_TACQ +#define ADC_TACQ SAADC_CH_CONFIG_TACQ_10us +#endif +/** @} */ + +/** + * @brief Lock to prevent concurrency issues when used from different threads + */ +static mutex_t lock = MUTEX_INIT; + +/** + * @brief We use a static result buffer so we do not have to reprogram the + * result pointer register + */ +static int16_t result; + +static inline void prep(void) +{ + mutex_lock(&lock); + NRF_SAADC->ENABLE = 1; +} + +static inline void done(void) +{ + NRF_SAADC->ENABLE = 0; + mutex_unlock(&lock); +} + +int adc_init(adc_t line) +{ + if (line >= ADC_NUMOF) { + return -1; + } + + prep(); + + /* prevent multiple initialization by checking the result ptr register */ + if (NRF_SAADC->RESULT.PTR != (uint32_t)&result) { + /* set data pointer and the single channel we want to convert */ + NRF_SAADC->RESULT.MAXCNT = 1; + NRF_SAADC->RESULT.PTR = (uint32_t)&result; + + /* configure the first channel (the only one we use): + * - bypass resistor ladder+ + * - acquisition time as defined by board (or 10us as default) + * - reference and gain as defined by board (or VDD as default) + * - no oversampling */ + NRF_SAADC->CH[0].CONFIG = ((ADC_GAIN << SAADC_CH_CONFIG_GAIN_Pos) | + (ADC_REF << SAADC_CH_CONFIG_REFSEL_Pos) | + (ADC_TACQ << SAADC_CH_CONFIG_TACQ_Pos)); + NRF_SAADC->CH[0].PSELN = SAADC_CH_PSELN_PSELN_NC; + NRF_SAADC->OVERSAMPLE = SAADC_OVERSAMPLE_OVERSAMPLE_Bypass; + + /* calibrate SAADC */ + NRF_SAADC->EVENTS_CALIBRATEDONE = 0; + NRF_SAADC->TASKS_CALIBRATEOFFSET = 1; + while (NRF_SAADC->EVENTS_CALIBRATEDONE == 0) {} + } + + done(); + + return 0; +} + +int adc_sample(adc_t line, adc_res_t res) +{ + assert(line < ADC_NUMOF); + + /* check if resolution is valid */ + if (res > 2) { + return -1; + } + + /* prepare device */ + prep(); + + /* set resolution */ + NRF_SAADC->RESOLUTION = res; + /* set line to sample */ + NRF_SAADC->CH[0].PSELP = (line + 1); + + /* start the SAADC and wait for the started event */ + NRF_SAADC->EVENTS_STARTED = 0; + NRF_SAADC->TASKS_START = 1; + while (NRF_SAADC->EVENTS_STARTED == 0) {} + + /* trigger the actual conversion */ + NRF_SAADC->EVENTS_END = 0; + NRF_SAADC->TASKS_SAMPLE = 1; + while (NRF_SAADC->EVENTS_END == 0) {} + + /* stop the SAADC */ + NRF_SAADC->EVENTS_STOPPED = 0; + NRF_SAADC->TASKS_STOP = 1; + while (NRF_SAADC->EVENTS_STOPPED == 0) {} + + /* free device */ + done(); + + /* hack -> the result can be a small negative number when a AINx pin is + * connected via jumper wire a the board's GND pin. There seems to be a + * slight difference between the internal CPU GND and the board's GND + * voltage levels?! (observed on nrf52dk and nrf52840dk) */ + return (result < 0) ? 0 : (int)result; +} diff --git a/cpu/nrf5x_common/include/periph_cpu_common.h b/cpu/nrf5x_common/include/periph_cpu_common.h index 78d1ab0491bc050bc2cc63fe8142d0f12edecafd..608883c8e6a8f53379826f81b44b84b17edafa08 100644 --- a/cpu/nrf5x_common/include/periph_cpu_common.h +++ b/cpu/nrf5x_common/include/periph_cpu_common.h @@ -109,21 +109,6 @@ typedef enum { GPIO_BOTH = 3 /**< emit interrupt on both flanks */ } gpio_flank_t; /** @} */ - -/** - * @brief Override ADC resolution values - * @{ - */ -#define HAVE_ADC_RES_T -typedef enum { - ADC_RES_6BIT = 0xf0, /**< ADC resolution: 6 bit */ - ADC_RES_8BIT = 0x00, /**< ADC resolution: 8 bit */ - ADC_RES_10BIT = 0x02, /**< ADC resolution: 10 bit */ - ADC_RES_12BIT = 0xf1, /**< ADC resolution: 12 bit */ - ADC_RES_14BIT = 0xf2, /**< ADC resolution: 14 bit */ - ADC_RES_16BIT = 0xf3 /**< ADC resolution: 16 bit */ -} adc_res_t; -/** @} */ #endif /* ndef DOXYGEN */ /**