2017-05-10 21:23:00 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2017 Dan Evans <photonthunder@gmail.com>
|
|
|
|
* Copyright (C) 2017 Travis Griggs <travisgriggs@gmail.com>
|
|
|
|
* Copyright (C) 2017 Dylan Laduranty <dylanladuranty@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.
|
|
|
|
*/
|
|
|
|
|
2017-06-22 15:43:17 +02:00
|
|
|
/**
|
|
|
|
* @ingroup cpu_sam0_common
|
|
|
|
* @ingroup drivers_periph_adc
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Low-level ADC driver implementation
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
2017-05-10 21:23:00 +02:00
|
|
|
#include <stdint.h>
|
|
|
|
#include "cpu.h"
|
|
|
|
#include "periph/gpio.h"
|
|
|
|
#include "periph/adc.h"
|
|
|
|
#include "periph_conf.h"
|
|
|
|
#include "mutex.h"
|
|
|
|
|
2017-05-15 16:37:55 +02:00
|
|
|
#define ENABLE_DEBUG (0)
|
2017-05-10 21:23:00 +02:00
|
|
|
#include "debug.h"
|
|
|
|
|
|
|
|
/* Prototypes */
|
|
|
|
static bool _adc_syncing(void);
|
|
|
|
static void _adc_poweroff(void);
|
|
|
|
static int _adc_configure(adc_res_t res);
|
|
|
|
|
|
|
|
static mutex_t _lock = MUTEX_INIT;
|
|
|
|
|
|
|
|
static inline void _prep(void)
|
|
|
|
{
|
|
|
|
mutex_lock(&_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void _done(void)
|
|
|
|
{
|
|
|
|
mutex_unlock(&_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool _adc_syncing(void)
|
|
|
|
{
|
|
|
|
#ifdef CPU_SAMD21
|
2019-10-01 21:42:42 +02:00
|
|
|
if (ADC->STATUS.reg & ADC_STATUS_SYNCBUSY) {
|
2017-05-10 21:23:00 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#else /* CPU_SAML21 */
|
2019-10-01 21:42:42 +02:00
|
|
|
if (ADC->SYNCBUSY.reg) {
|
2017-05-10 21:23:00 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void _adc_poweroff(void)
|
|
|
|
{
|
|
|
|
while (_adc_syncing()) {}
|
|
|
|
/* Disable */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->CTRLA.reg &= ~ADC_CTRLA_ENABLE;
|
2017-05-10 21:23:00 +02:00
|
|
|
while (_adc_syncing()) {}
|
|
|
|
/* Disable bandgap */
|
|
|
|
#ifdef CPU_SAMD21
|
2019-10-01 21:42:42 +02:00
|
|
|
if (ADC_REF_DEFAULT == ADC_REFCTRL_REFSEL_INT1V) {
|
2017-05-10 21:23:00 +02:00
|
|
|
SYSCTRL->VREF.reg &= ~SYSCTRL_VREF_BGOUTEN;
|
|
|
|
}
|
|
|
|
#else /* CPU_SAML21 */
|
2019-10-01 21:42:42 +02:00
|
|
|
if (ADC_REF_DEFAULT == ADC_REFCTRL_REFSEL_INTREF) {
|
2017-05-10 21:23:00 +02:00
|
|
|
SUPC->VREF.reg &= ~SUPC_VREF_VREFOE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static int _adc_configure(adc_res_t res)
|
|
|
|
{
|
2017-05-31 18:39:32 +02:00
|
|
|
/* Individual comparison necessary because ADC Resolution Bits are not
|
|
|
|
* numerically in order and 16Bit (averaging - not currently supported)
|
|
|
|
* falls between 12bit and 10bit. See datasheet for details */
|
2019-11-27 21:01:32 +01:00
|
|
|
if (!((res == ADC_RES_8BIT) || (res == ADC_RES_10BIT) ||
|
|
|
|
(res == ADC_RES_12BIT))){
|
|
|
|
return -1;
|
|
|
|
}
|
2017-05-10 21:23:00 +02:00
|
|
|
_adc_poweroff();
|
2019-10-01 21:42:42 +02:00
|
|
|
if (ADC->CTRLA.reg & ADC_CTRLA_SWRST ||
|
|
|
|
ADC->CTRLA.reg & ADC_CTRLA_ENABLE ) {
|
2017-05-10 21:23:00 +02:00
|
|
|
DEBUG("adc: not ready\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#ifdef CPU_SAMD21
|
|
|
|
/* Power On */
|
|
|
|
PM->APBCMASK.reg |= PM_APBCMASK_ADC;
|
|
|
|
/* GCLK Setup */
|
|
|
|
GCLK->CLKCTRL.reg = (uint32_t)(GCLK_CLKCTRL_CLKEN | GCLK_CLKCTRL_GEN_GCLK0 |
|
|
|
|
(GCLK_CLKCTRL_ID(ADC_GCLK_ID)));
|
|
|
|
/* Configure CTRLB Register HERE IS THE RESOLUTION SET! */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->CTRLB.reg = ADC_PRESCALER | res;
|
2017-05-10 21:23:00 +02:00
|
|
|
/* Load the fixed device calibration constants */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->CALIB.reg =
|
2017-05-10 21:23:00 +02:00
|
|
|
ADC_CALIB_BIAS_CAL((*(uint32_t*)ADC_FUSES_BIASCAL_ADDR >>
|
|
|
|
ADC_FUSES_BIASCAL_Pos)) |
|
|
|
|
ADC_CALIB_LINEARITY_CAL((*(uint64_t*)ADC_FUSES_LINEARITY_0_ADDR >>
|
|
|
|
ADC_FUSES_LINEARITY_0_Pos));
|
|
|
|
/* Set Voltage Reference */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->REFCTRL.reg = ADC_REF_DEFAULT;
|
2017-05-10 21:23:00 +02:00
|
|
|
/* Disable all interrupts */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->INTENCLR.reg = (ADC_INTENCLR_SYNCRDY) | (ADC_INTENCLR_WINMON) |
|
2017-05-10 21:23:00 +02:00
|
|
|
(ADC_INTENCLR_OVERRUN) | (ADC_INTENCLR_RESRDY);
|
|
|
|
while (_adc_syncing()) {}
|
|
|
|
/* Enable bandgap if VREF is internal 1V */
|
2019-10-01 21:42:42 +02:00
|
|
|
if (ADC_REF_DEFAULT == ADC_REFCTRL_REFSEL_INT1V) {
|
2017-05-10 21:23:00 +02:00
|
|
|
SYSCTRL->VREF.reg |= SYSCTRL_VREF_BGOUTEN;
|
|
|
|
}
|
|
|
|
#else /* CPU_SAML21 */
|
|
|
|
/* Power on */
|
2019-01-21 17:06:58 +01:00
|
|
|
#ifdef CPU_SAML1X
|
|
|
|
MCLK->APBCMASK.reg |= MCLK_APBCMASK_ADC;
|
|
|
|
#else
|
2017-05-10 21:23:00 +02:00
|
|
|
MCLK->APBDMASK.reg |= MCLK_APBDMASK_ADC;
|
2019-01-21 17:06:58 +01:00
|
|
|
#endif
|
2017-05-10 21:23:00 +02:00
|
|
|
/* GCLK Setup */
|
|
|
|
GCLK->PCHCTRL[ADC_GCLK_ID].reg = GCLK_PCHCTRL_CHEN | GCLK_PCHCTRL_GEN_GCLK0;
|
|
|
|
/* Set Voltage Reference */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->REFCTRL.reg = ADC_REF_DEFAULT;
|
2017-05-10 21:23:00 +02:00
|
|
|
/* Configure CTRLB & CTRLC Register */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->CTRLB.reg = ADC_PRESCALER;
|
|
|
|
ADC->CTRLC.reg |= res;
|
2017-05-10 21:23:00 +02:00
|
|
|
/* Disable all interrupts */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->INTENCLR.reg = ADC_INTENCLR_WINMON | ADC_INTENCLR_OVERRUN |
|
2017-05-10 21:23:00 +02:00
|
|
|
ADC_INTENCLR_RESRDY;
|
|
|
|
/* Set default calibration from NVM */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->CALIB.reg =
|
2017-05-10 21:23:00 +02:00
|
|
|
ADC_FUSES_BIASCOMP((*(uint32_t*)ADC_FUSES_BIASCOMP_ADDR)) >>
|
|
|
|
ADC_CALIB_BIASCOMP_Pos |
|
|
|
|
ADC_FUSES_BIASREFBUF((*(uint32_t*)ADC_FUSES_BIASREFBUF_ADDR) >>
|
|
|
|
ADC_FUSES_BIASREFBUF_Pos);
|
|
|
|
while (_adc_syncing()) {}
|
|
|
|
/* Enable bandgap if necessary */
|
2019-10-01 21:42:42 +02:00
|
|
|
if (ADC_REF_DEFAULT == ADC_REFCTRL_REFSEL_INTREF) {
|
2017-05-10 21:23:00 +02:00
|
|
|
SUPC->VREF.reg |= SUPC_VREF_VREFOE;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
/* Enable ADC Module */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->CTRLA.reg |= ADC_CTRLA_ENABLE;
|
2017-05-10 21:23:00 +02:00
|
|
|
while (_adc_syncing()) {}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int adc_init(adc_t line)
|
|
|
|
{
|
2019-02-03 23:17:22 +01:00
|
|
|
if (line >= ADC_NUMOF) {
|
|
|
|
DEBUG("adc: line arg not applicable\n");
|
|
|
|
return -1;
|
|
|
|
}
|
2017-05-10 21:23:00 +02:00
|
|
|
_prep();
|
|
|
|
gpio_init(adc_channels[line].pin, GPIO_IN);
|
|
|
|
gpio_init_mux(adc_channels[line].pin, GPIO_MUX_B);
|
|
|
|
_done();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int adc_sample(adc_t line, adc_res_t res)
|
|
|
|
{
|
|
|
|
if (line >= ADC_NUMOF) {
|
|
|
|
DEBUG("adc: line arg not applicable\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
_prep();
|
|
|
|
if (_adc_configure(res) != 0) {
|
|
|
|
_done();
|
|
|
|
DEBUG("adc: configuration failed\n");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#ifdef CPU_SAMD21
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->INPUTCTRL.reg = ADC_GAIN_FACTOR_DEFAULT |
|
|
|
|
adc_channels[line].muxpos | ADC_NEG_INPUT;
|
2017-05-10 21:23:00 +02:00
|
|
|
#else /* CPU_SAML21 */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->INPUTCTRL.reg = adc_channels[line].muxpos | ADC_NEG_INPUT;
|
2017-05-10 21:23:00 +02:00
|
|
|
#endif
|
|
|
|
while (_adc_syncing()) {}
|
|
|
|
/* Start the conversion */
|
2019-10-01 21:42:42 +02:00
|
|
|
ADC->SWTRIG.reg = ADC_SWTRIG_START;
|
2017-05-10 21:23:00 +02:00
|
|
|
/* Wait for the result */
|
2019-10-01 21:42:42 +02:00
|
|
|
while (!(ADC->INTFLAG.reg & ADC_INTFLAG_RESRDY)) {}
|
|
|
|
int result = ADC->RESULT.reg;
|
2017-05-10 21:23:00 +02:00
|
|
|
_adc_poweroff();
|
|
|
|
_done();
|
|
|
|
return result;
|
|
|
|
}
|