×
嵌入式 > 嵌入式开发 > 详情

Arm学习笔记--ADC编程

发布时间:2020-08-25 发布时间:
|
A/D转换器的功能是将模拟输入信号采样得到可以提供计算机进行处理的数字信号。开发板的A/D输入模块电路图如下
这是一个测量电压值的电路。


LPC1788的ADC是一个12位的逐次逼近型模数转换器,有8个复用的输入管脚,它的时钟使用PCLK分频得到。从这段描述中我们可以分析出一下的内容:
1、编程的时候需要首先打开这个功能,所以需要配置寄存器PCONP
2、既然ADC有8个引脚,所以就有一个选择的寄存器,这个就是控制寄存器,因为时钟需要切换,控制寄存器也包含这个功能
3、ADC需要有数据寄存器来保存读取到的数据,那么每个输入管脚就对应一个寄存器用来存储对应引脚的数据,这就是数据寄存器
4、ADC当前状态的检测需要一个状态寄存器
5、从ADC的块图上我们可以看出有中断接口,所以肯定有一个中断使能寄存器表明是那个引脚引起的中断
6、从块图上看到有一个触发源,这里也是一个寄存器控制
从块图和ADC的原理上我们基本上能理出这些内容,下面我们就来看一个ADC的例程,从例程中分析LPC1788 ADC的编程思路。
这是关于LPC1788的ADC寄存器定义,跟我们的分析差不多

typedef struct
{
__IO uint32_t CR; /*!< Offset: 0x000 A/D Control Register (R/W) */
__IO uint32_t GDR; /*!< Offset: 0x004 A/D Global Data Register (R/W) */
uint32_t RESERVED0;
__IO uint32_t INTEN; /*!< Offset: 0x00C A/D Interrupt Enable Register (R/W) */
__IO uint32_t DR[8]; /*!< Offset: 0x010-0x02C A/D Channel 0..7 Data Register (R/W) */
__I uint32_t STAT; /*!< Offset: 0x030 A/D Status Register (R/ ) */
__IO uint32_t ADTRM;
} LPC_ADC_TypeDef;


ADC的初始化代码,主要设置了时钟频率,满足ADC操作的需求

void ADC_Init(LPC_ADC_TypeDef *ADCx, uint32_t rate)
{
uint32_t temp, tmp;

// Turn on power and clock
CLKPWR_ConfigPPWR(CLKPWR_PCONP_PCADC, ENABLE); //开启ADC功能

ADCx->CR = 0; //控制寄存器初始化

//Enable PDN bit
tmp = ADC_CR_PDN;

// Set clock frequency
temp = CLKPWR_GetCLK(CLKPWR_CLKTYPE_PER); //设置时钟频率

/* 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 12.4MHz.
* A fully conversion requires 31 of these clocks.
* ADC clock = PCLK_ADC0 / (CLKDIV + 1);
* ADC rate = ADC clock / 31;
*/
temp = (temp /(rate * 31)) - 1;
tmp |= ADC_CR_CLKDIV(temp);

ADCx->CR = tmp; //配置了时钟频率
}

中断配置

/*********************************************************************//**
* @brief ADC interrupt configuration
* @param[in] ADCx pointer to LPC_ADC_TypeDef, should be: LPC_ADC
* @param[in] IntType: type of interrupt, should be:
* - ADC_ADINTEN0: Interrupt channel 0
* - ADC_ADINTEN1: Interrupt channel 1
* ...
* - ADC_ADINTEN7: Interrupt channel 7
* - ADC_ADGINTEN: Individual channel/global flag done generate an interrupt
* @param[in] NewState:
* - SET : enable ADC interrupt
* - RESET: disable ADC interrupt
* @return None
**********************************************************************/
void ADC_IntConfig (LPC_ADC_TypeDef *ADCx, ADC_TYPE_INT_OPT IntType, FunctionalState NewState)
{
ADCx->INTEN &= ~ADC_INTEN_CH(IntType);
if (NewState){
ADCx->INTEN |= ADC_INTEN_CH(IntType);
}
}

设置通道

void ADC_ChannelCmd (LPC_ADC_TypeDef *ADCx, uint8_t Channel, FunctionalState NewState)
{
if (NewState == ENABLE) {
ADCx->CR |= ADC_CR_CH_SEL(Channel);
} else {
ADCx->CR &= ~ADC_CR_CH_SEL(Channel);
}
}

中断处理

void ADC_IRQHandler(void)
{
adc_value = 0;

if (ADC_ChannelGetStatus(LPC_ADC, TENGHUA_ADC_PREPARED_CHANNEL, ADC_DATA_DONE))
{
adc_value = ADC_ChannelGetData(LPC_ADC, TENGHUA_ADC_PREPARED_CHANNEL);
NVIC_DisableIRQ(ADC_IRQn);
}
}


其实ADC是很简单和基础的,主要还是掌握好寄存器的使用。



『本文转载自网络,版权归原作者所有,如有侵权请联系删除』

热门文章 更多
发明专利在疫情影响下的逆势增长