mirror of
https://github.com/RT-Thread/rt-thread.git
synced 2025-01-28 02:20:26 +08:00
171 lines
5.3 KiB
C
171 lines
5.3 KiB
C
/*
|
|
* @brief ADC Registers and control functions
|
|
*
|
|
* @note
|
|
* Copyright(C) NXP Semiconductors, 2012
|
|
* All rights reserved.
|
|
*
|
|
* @par
|
|
* Software that is described herein is for illustrative purposes only
|
|
* which provides customers with programming information regarding the
|
|
* LPC products. This software is supplied "AS IS" without any warranties of
|
|
* any kind, and NXP Semiconductors and its licensor disclaim any and
|
|
* all warranties, express or implied, including all implied warranties of
|
|
* merchantability, fitness for a particular purpose and non-infringement of
|
|
* intellectual property rights. NXP Semiconductors assumes no responsibility
|
|
* or liability for the use of the software, conveys no license or rights under any
|
|
* patent, copyright, mask work right, or any other intellectual property rights in
|
|
* or to any products. NXP Semiconductors reserves the right to make changes
|
|
* in the software without notification. NXP Semiconductors also makes no
|
|
* representation or warranty that such application will be suitable for the
|
|
* specified use without further testing or modification.
|
|
*
|
|
* @par
|
|
* Permission to use, copy, modify, and distribute this software and its
|
|
* documentation is hereby granted, under NXP Semiconductors' and its
|
|
* licensor's relevant copyrights in the software, without fee, provided that it
|
|
* is used in conjunction with NXP Semiconductors microcontrollers. This
|
|
* copyright, permission, and disclaimer notice must appear in all copies of
|
|
* this code.
|
|
*/
|
|
|
|
#include "adc_001.h"
|
|
|
|
/*****************************************************************************
|
|
* Private types/enumerations/variables
|
|
****************************************************************************/
|
|
|
|
/*****************************************************************************
|
|
* Public types/enumerations/variables
|
|
****************************************************************************/
|
|
|
|
/*****************************************************************************
|
|
* Private functions
|
|
****************************************************************************/
|
|
|
|
/* Configure clock for ADC */
|
|
static void SetClock(IP_ADC_001_Type *pADC, uint32_t adcRate, uint32_t adcPerClock, uint8_t bitsAccuracy)
|
|
{
|
|
uint32_t temp, adcBitRate;
|
|
|
|
/* The APB clock (PCLK_ADC0) is divided by (CLKDIV+1) to produce the clock for
|
|
A/D converter, which should be less than or equal to 4.5MHz.
|
|
A fully conversion requires (bits_accuracy+1) of these clocks.
|
|
ADC clock = PCLK_ADC0 / (CLKDIV + 1);
|
|
ADC rate = ADC clock / (bits_accuracy+1);
|
|
*/
|
|
adcBitRate = (adcRate * (11 - bitsAccuracy));
|
|
|
|
/* Get the round value by fomular: (2*A + B)/(2*B) */
|
|
temp = ((adcPerClock * 2 + adcBitRate) / (adcBitRate * 2)) - 1;
|
|
|
|
/* Enable PDN bit and clock bits */
|
|
pADC->CR &= ~(ADC_CR_CLKDIV(0xFF) | ADC_CR_BITACC(0x07));
|
|
pADC->CR |= ADC_CR_CLKDIV(temp) | ADC_CR_BITACC(bitsAccuracy);
|
|
}
|
|
|
|
/*****************************************************************************
|
|
* Public functions
|
|
****************************************************************************/
|
|
|
|
/* Initialize the ADC */
|
|
void IP_ADC_Init(IP_ADC_001_Type *pADC, uint32_t adcRate, uint32_t adcPerClock, uint8_t bitsAccuracy)
|
|
{
|
|
pADC->INTEN = 0; /* Disable all interrupts */
|
|
pADC->CR |= ADC_CR_PDN; /* Set PDN bit for ADC*/
|
|
SetClock(pADC, adcRate, adcPerClock, bitsAccuracy);
|
|
}
|
|
|
|
/* Shutdown ADC */
|
|
void IP_ADC_DeInit(IP_ADC_001_Type *pADC)
|
|
{
|
|
pADC->INTEN = 0x00000100;
|
|
pADC->CR = 0;
|
|
}
|
|
|
|
/* Set burst mode for ADC */
|
|
void IP_ADC_SetBurstMode(IP_ADC_001_Type *pADC, FunctionalState NewState)
|
|
{
|
|
if (NewState == DISABLE) {
|
|
pADC->CR &= ~ADC_CR_BURST;
|
|
}
|
|
else {
|
|
pADC->CR |= ADC_CR_BURST;
|
|
}
|
|
}
|
|
|
|
/* Get the ADC value */
|
|
Status IP_ADC_Get_Val(IP_ADC_001_Type *pADC, uint8_t channel, uint16_t *data)
|
|
{
|
|
uint32_t temp;
|
|
temp = pADC->DR[channel];
|
|
if (!ADC_DR_DONE(temp)) {
|
|
return ERROR;
|
|
}
|
|
// if(ADC_DR_OVERRUN(temp) && (pADC->CR & ADC_CR_BURST))
|
|
// return ERROR;
|
|
*data = (uint16_t) ADC_DR_RESULT(temp);
|
|
return SUCCESS;
|
|
}
|
|
|
|
/* Get ADC Channel status from ADC data register */
|
|
FlagStatus IP_ADC_GetStatus(IP_ADC_001_Type *pADC, uint8_t channel, uint32_t StatusType)
|
|
{
|
|
switch (StatusType) {
|
|
case ADC_DR_DONE_STAT:
|
|
return (pADC->STAT & (1UL << channel)) ? SET : RESET;
|
|
|
|
case ADC_DR_OVERRUN_STAT:
|
|
channel += 8;
|
|
return (pADC->STAT & (1UL << channel)) ? SET : RESET;
|
|
|
|
case ADC_DR_ADINT_STAT:
|
|
return pADC->STAT >> 16 ? SET : RESET;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
return RESET;
|
|
}
|
|
|
|
/* Set the edge start condition */
|
|
void IP_ADC_EdgeStartConfig(IP_ADC_001_Type *pADC, uint8_t edge_mode)
|
|
{
|
|
if (edge_mode) {
|
|
pADC->CR |= ADC_CR_EDGE;
|
|
}
|
|
else {
|
|
pADC->CR &= ~ADC_CR_EDGE;
|
|
}
|
|
}
|
|
|
|
/* Enable/Disable ADC channel number */
|
|
void IP_ADC_SetChannelNumber(IP_ADC_001_Type *pADC, uint8_t channel, FunctionalState NewState)
|
|
{
|
|
if (NewState == ENABLE) {
|
|
pADC->CR |= ADC_CR_CH_SEL(channel);
|
|
}
|
|
else {
|
|
pADC->CR &= ~ADC_CR_START_MASK;
|
|
pADC->CR &= ~ADC_CR_CH_SEL(channel);
|
|
}
|
|
}
|
|
|
|
/* Set start mode for ADC */
|
|
void IP_ADC_SetStartMode(IP_ADC_001_Type *pADC, uint8_t start_mode)
|
|
{
|
|
pADC->CR &= ~ADC_CR_START_MASK;
|
|
pADC->CR |= ADC_CR_START_MODE_SEL((uint32_t) start_mode);
|
|
}
|
|
|
|
/* Enable/Disable interrupt for ADC channel */
|
|
void IP_ADC_Int_Enable(IP_ADC_001_Type *pADC, uint8_t channel, FunctionalState NewState)
|
|
{
|
|
if (NewState == ENABLE) {
|
|
pADC->INTEN |= (1UL << channel);
|
|
}
|
|
else {
|
|
pADC->INTEN &= (~(1UL << channel));
|
|
}
|
|
}
|