- fixed selection of resolution - fixed result masking - fixed channel selection (and optimized it) - fixed clock prescaler calculation
113 lines
2.4 KiB
C
113 lines
2.4 KiB
C
/*
|
|
* Copyright (C) 2015-2016 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_lpc11u34
|
|
* @ingroup drivers_periph_adc
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Low-level ADC driver implementation
|
|
*
|
|
* @author Paul Rathgeb <paul.rathgeb@skynet.be>
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
*
|
|
* @}
|
|
*/
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "cpu.h"
|
|
#include "mutex.h"
|
|
#include "assert.h"
|
|
#include "periph/adc.h"
|
|
|
|
#define START_CMD (0x1 << 24)
|
|
#define DONE_BIT (0x1 << 31)
|
|
#define RES_MASK (0xf << 17)
|
|
#define SAMPLE_SHIFT (6)
|
|
#define SAMPLE_MASK (0x3ff)
|
|
|
|
/* we chose an ADC clock smaller or equal than but close to 4.5MHz (as proposed
|
|
* in the reference manual) */
|
|
#define CLK_DIV (((CLOCK_CORECLOCK / 45000000) & 0xff) << 8)
|
|
|
|
/**
|
|
* @brief Mutex to synchronize ADC access from different threads
|
|
*/
|
|
static mutex_t lock = MUTEX_INIT;
|
|
|
|
static inline uint32_t *pincfg_reg(adc_t line)
|
|
{
|
|
int offset = (line < 6) ? (11 + line) : (16 + line);
|
|
return ((uint32_t *)(LPC_IOCON) + offset);
|
|
}
|
|
|
|
static inline void prep(void)
|
|
{
|
|
mutex_lock(&lock);
|
|
LPC_SYSCON->PDRUNCFG &= ~(1 << 4);
|
|
LPC_SYSCON->SYSAHBCLKCTRL |= (1 << 13);
|
|
}
|
|
|
|
static inline void done(void)
|
|
{
|
|
LPC_SYSCON->SYSAHBCLKCTRL &= ~(1 << 13);
|
|
LPC_SYSCON->PDRUNCFG |= (1 << 4);
|
|
mutex_unlock(&lock);
|
|
}
|
|
|
|
int adc_init(adc_t line)
|
|
{
|
|
uint32_t *pincfg;
|
|
|
|
prep();
|
|
|
|
/* configure the connected pin */
|
|
pincfg = pincfg_reg(line);
|
|
/* Put the pin in its ADC alternate function */
|
|
if (line < 5) {
|
|
*pincfg |= 2;
|
|
}
|
|
else {
|
|
*pincfg |= 1;
|
|
}
|
|
/* Configure ADMODE in analog input */
|
|
*pincfg &= ~(1 << 7);
|
|
|
|
done();
|
|
|
|
return 0;
|
|
}
|
|
|
|
int adc_sample(adc_t line, adc_res_t res)
|
|
{
|
|
int sample;
|
|
|
|
assert(line < ADC_NUMOF);
|
|
|
|
/* check if resolution is valid */
|
|
if (res & ~(RES_MASK)) {
|
|
return -1;
|
|
}
|
|
|
|
/* prepare the device */
|
|
prep();
|
|
|
|
/* start conversion */
|
|
LPC_ADC->CR = ((1 << line) | CLK_DIV | res | START_CMD);
|
|
/* Wait for the end of the conversion */
|
|
while (!(LPC_ADC->DR[line] & DONE_BIT)) {}
|
|
/* Read and return result */
|
|
sample = ((LPC_ADC->DR[line] >> SAMPLE_SHIFT) & SAMPLE_MASK);
|
|
|
|
done();
|
|
|
|
return sample;
|
|
}
|