cortex m0 lpc1114 adc start位控制转换
“START位”位于AD模块控制寄存器bit24~bit26。
位 | 符号 | 值 | 描述 | 复位值 |
7:0 | SEL | 选择哪个引脚用作采集和转换,当bit0=1,用AD0;当bit1=1,…,当bit7=1,用AD7在软件控制模式,当(BURST=0),只允许选择一个引脚,也就是说,这7位当中,只允许有1个位是1
在硬件扫描模式,当(BURST=1),可以允许1个,也可以允许多个甚至允许8个引脚都开启。如果所有位都是0,将自动使得SEL=0X01,即选择AD0 |
0x00 | |
15:8 | CLKDIV | APB时钟(PCLK)除以(CLKDIV+1)就是AD模块的时钟,这个值必须≤4.5MHz。 | 0 | |
16 | BURST | 触发模式位(注意:当BURST=1的时候,AD0INTEN寄存器中的ADGINTEN位必须为0) | 0 | |
0 | 软件控制模式:需要11个时钟转换 | |||
1 | 硬件扫描模式:AD转换将会按照CLKS位设置的速度重复转换,扫描所有SEL设置成1的位。首先扫描SEL中最小的设置成1的位,然后扫描再大一些的设置成1的位。在转换过程中,把此位写0,可以停止转换。(注意:如果BRUST=1,START位必须为000,否则将不会开启转换。) | |||
19:17 | CLKS | 时钟精度选择位,精度越高,转换时间越长 | 000 | |
0x0 | 11 clocks / 10 bits | |||
0x1 | 10 clocks / 9 bits | |||
0x2 | 9 clocks / 8 bits | |||
0x3 | 8 clocks / 7 bits | |||
0x4 | 7 clocks / 6 bits | |||
0x5 | 6 clocks / 5 bits | |||
0x6 | 5 clocks / 4 bits | |||
0x7 | 4 clocks / 3 bits | |||
23:20 | – | 保留位,禁止给这些位写1 | Na | |
26:24 | START | 当BURST=0,这些位控制软件控制转换方式 | 0 | |
0x0 | 没有开始 | |||
0x1 | 开始转换 | |||
0x2 | 当PIO0_2/SSEL/CT16B0_CAP0引脚上产生bit27位设置的边沿信号,开始转换 | |||
0x3 | 当PIO1_5/DIR/CT32B0_CAP0引脚上产生bit27位设置的边沿信号,开始转换 | |||
0x4 | 当CT32B0_MAT0产生bit27位设置的边沿信号,开始转换 | |||
0x5 | 当CT32B0_MAT1产生bit27位设置的边沿信号,开始转换 | |||
0x6 | 当CT16B0_MAT0产生bit27位设置的边沿信号,开始转换 | |||
0x7 | 当CT16B0_MAT1产生bit27位设置的边沿信号,开始转换 | |||
27 | EDGE | 只有在START位为010~111时,这个位有效 | 0 | |
0 | 在CAP/MAT引脚上产生上升沿触发转换 | |||
1 | 在CAP/MAT引脚上产生下降沿触发转换 | |||
31:28 | – | 保留位,不允许写1到这些位 | Na |
在CR寄存器中,由SEL位选择输入通道,由CLKDIV决定AD时钟,由BURST位控制转换触发模式,CLKS位决定转换精度,START位和EDGE位决定了软件触发模式下的转换模式。
新建一个工程,结构如下图所示:
在adc.h中,加入以下代码:
- #ifndef __NXP_ADC_H
- #define __NXP_ADC_H
- #define? Vref?? 3300
- extern void ADC_Init(uint8_t Channel);
- extern uint32_t ADC_Read(uint8_t Channel);
- #endif
在adc.c文件中,加入以下代码:
- #include “lpc11xx.h”
- #include “adc.h”
- void ADC_Init(uint8_t Channel)
- {
- ???if(Channel>7) return;
- ???LPC_SYSCON->PDRUNCFG &= ~(0x1<<4);??????? // ADC模块上电
- ???LPC_SYSCON->SYSAHBCLKCTRL |= (1<<13);???? // 使能ADC时钟
- ???LPC_SYSCON->SYSAHBCLKCTRL |= (1<<16);???? // 使能IOCON时钟
- ???switch(Channel)
- ???{
- ??????case 0: // 通道0配置 set channel 0
- ??????LPC_IOCON->R_PIO0_11 &= ~0x07;????????????? //
- ??????LPC_IOCON->R_PIO0_11 |= 0x02;????? // 把P0.11引脚设置为AD0功能
- ??????LPC_IOCON->R_PIO0_11 &= ~(3<<3) ; ?// 去掉上拉和下拉电阻
- ??????LPC_IOCON->R_PIO0_11 &= ~(1<<7) ;?? // 模拟输入模式
- ??????break;
- ??????case 1:? // 通道1配置 set channel 1
- ??????LPC_IOCON->R_PIO1_0 &= ~0x07;????????????? //
- ??????LPC_IOCON->R_PIO1_0 |= 0x02;?????????????? // 把P1.0引脚设置为AD1功能
- ??????LPC_IOCON->R_PIO1_0 &= ~(3<<3) ;?????????? // 去掉上拉和下拉电阻
- ??????LPC_IOCON->R_PIO1_0 &= ~(1<<7) ;?????????? // 模拟输入模式
- ??????break;
- ??????case 2:? // 通道2配置 set channel 2
- ??????LPC_IOCON->R_PIO1_1 &= ~0x07;????????????? //
- ??????LPC_IOCON->R_PIO1_1 |= 0x02;?????????????? // 把P1.1引脚设置为AD2功能
- ??????LPC_IOCON->R_PIO1_1 &= ~(3<<3) ;?????????? // 去掉上拉和下拉电阻
- ??????LPC_IOCON->R_PIO1_1 &= ~(1<<7) ;?????????? // 模拟输入模式
- ??????break;
- ??????case 3:? // 通道3配置 set channel 3
- ??????LPC_IOCON->R_PIO1_2 &= ~0x07;????????????? //
- ??????LPC_IOCON->R_PIO1_2 |= 0x02;?????????????? // 把P1.2引脚设置为AD3功能
- ??????LPC_IOCON->R_PIO1_2 &= ~(3<<3) ;?????????? // 去掉上拉和下拉电阻
- ??????LPC_IOCON->R_PIO1_2 &= ~(1<<7) ;?????????? // 模拟输入模式
- ??????break;
- ??????case 4:? // 通道4配置 set channel 4
- ??????LPC_IOCON->SWDIO_PIO1_3 &= ~0x07;????????????? //
- ??????LPC_IOCON->SWDIO_PIO1_3 |= 0x02;??? // 把P1.3引脚设置为AD4功能
- ??????LPC_IOCON->SWDIO_PIO1_3 &= ~(3<<3) ;?????????? // 去掉上拉和下拉电阻
- ??????LPC_IOCON->SWDIO_PIO1_3 &= ~(1<<7) ;?????????? // 模拟输入模式
- ??????break;
- ??????case 5:? // 通道5配置 set channel 5
- ??????LPC_IOCON->PIO1_4 &= ~0x07;????????????? //
- ??????LPC_IOCON->PIO1_4 |= 0x01;?????????????? // 把P1.4引脚设置为AD5功能
- ??????LPC_IOCON->PIO1_4 &= ~(3<<3) ;?????????? // 去掉上拉和下拉电阻
- ??????LPC_IOCON->PIO1_4 &= ~(1<<7) ;?????????? // 模拟输入模式
- ??????break;
- ??????case 6:? // 通道6配置 set channel 6
- ??????LPC_IOCON->PIO1_10 &= ~0x07;????????????? //
- ??????LPC_IOCON->PIO1_10 |= 0x01;?????????????? // 把P1.10引脚设置为AD6功能
- ??????LPC_IOCON->PIO1_10 &= ~(3<<3) ;?????????? // 去掉上拉和下拉电阻
- ??????LPC_IOCON->PIO1_10 &= ~(1<<7) ;?????????? // 模拟输入模式
- ??????break;
- ??????case 7:? // 通道7配置 set channel 7
- ??????LPC_IOCON->PIO1_11 &= ~0x07;????????????? //
- ??????LPC_IOCON->PIO1_11 |= 0x01;?????????????? // 把P1.11引脚设置为AD7功能
- ??????LPC_IOCON->PIO1_11 &= ~(3<<3) ;?????????? // 去掉上拉和下拉电阻
- ??????LPC_IOCON->PIO1_11 &= ~(1<<7) ;?????????? // 模拟输入模式
- ??????break;
- ??????default:break;
- ???}
- ???LPC_SYSCON->SYSAHBCLKCTRL &= ~(1<<16);??? // 关闭IOCON时钟
- ???LPC_ADC->CR = (1<<Channel)|? ?/* bit7:bit0?? 选择通道Channel*/
- (24<<8);? ?/* bit15:bit8? 把采样时钟频率设置为2MHz 50/(24+1)*/
- }
- uint32_t ADC_Read(uint8_t Channel)
- {
- ???uint32_t adc_value=0;
- ???LPC_ADC->CR |= (1<<24); // 启动转换
- ???while((LPC_ADC->DR[Channel]&0x80000000)==0);
- ???adc_value = (LPC_ADC->DR[Channel]>>6)&0x3FF;
- ???adc_value = (adc_value*Vref)/1024; // 转换为真正的电压值
- ???return adc_value;????? ? // 返回结果
- }
此文件中,定义了两个函数,一个是ADC初始化函数ADC_Init(),一个是ADC读值函数ADC_Read()。
第3~63行,ADC初始化函数,入口参数是通道号。
第5行,如果入口参数大于7,入口参数错误,退出函数。
第6行,给PDRUNCFG寄存器bit4写1,给ADC模块上电。
第7行,给SYSAHBCLKCTRL寄存器bit13写1,开启ADC模块时钟。
第8行,给SYSAHBCLKCTRL寄存器bit16写1,开启IOCON模块时钟,因为接下来,我们要配置单片机的引脚了。
第9~60行的配置,请看IOCON模块中相关引脚的寄存器,根据引脚寄存器的定义进行配置。
第61行,关闭IOCON时钟,因为引脚配置完毕,用不着这个模块了,关闭时钟节省功耗。
第62行,配置CR寄存器,选择通道,配置时钟。
第64~72行,ADC读值函数。
第66行,定义一个存放ADC值的变量。
第67行,启动转换。
第68行,观察数据寄存器转换完成标志位,等待转换完成。
第69行,获取ADC值。ADC值位于DR寄存器bit6~bit15位。读出来DR寄存器的值右移6位,即LPC_ADC->DR[channel]>>6;再把移位后的数的bit0~bit9取出,即把bit10以上的数都写成0,即把数“与”0x1111111111,
0与任何数都为0,1与任何数都为任何数,所以结果不会改变bit0~bit9的值,bit10以上的位都为0,即(LPC_ADC->DR[channel]>>6&0x3FF)。很多童鞋看到这条语句很复杂,无形中给自己的大脑设立了第一个门槛,然后分析了半天,没有头绪。当我给他们解释一番之后,才发现,只要看了DR寄存器的定义,和C语言中“右移”运行于“与”的运行即可解决。所以Ration再次强调,高手并不是掌握了特有的技术,而是掌握了扎实的基础。
第70行,把ADC值转换成电压值。因为ADC值是10位精度,210=1024,也就是说,ADC值由0~1023来表示0~VDD的电压值。利用等比公式
x/adc_value=Vref/1024??? x表示真实电压值
由上面等比公式得出:x=adc_value*Vref/1024
DR寄存器:(DR[0]~DR[7])
位 | 符号 | 值 | 描述 | 复位值 |
5:0 | – | 保留位 | 0 | |
15:6 | V_VREF | 当DONE位为1时,这些位表示ADC引脚上的测出的ADC值 | 不定 | |
29:16 | – | 保留位 | 0 | |
30 | OVERRUN | 当转换完成的值没有被读出,而又一次转换完成,此位置1,读取DR寄存器值后,此位清0 | 0 | |
31 | DONE | 转换完成标志位,转换完成置1,读取DR寄存器后,此位清0 | 0 |
在main.c文件中,输入以下代码:
- #include “lpc11xx.h”
- #include “uart.h”
- #include “adc.h”
- void delay(void)
- {
- uint16_t i,j;
- for(j=0;j<5000;j++)
- for(i=0;i<500;i++);
- }
- int main()
- {
- uint16_t adc_value;
- UART_init(9600);
- ADC_Init(7);
- while(1)
- {
- delay();
- adc_value = ADC_Read(7);
- UART_send_byte(adc_value)>>8);
- UART_send_byte(adc_value);
- }
- }
打开串口调试助手,选好串口号,波特率调成9600,选择十六进制接收,即可以看到在AD7(P1.11)通道上测到的电压值。注意:输入到AD7(P1.11)引脚上的电压值不能超过VDD,否则将烧毁内部ADC模块,甚者烧毁整个芯片。
因为Vref=3300mV,所以发送到电脑的电压值单位也是mV,先发送高8位数据,后发送低8位数据。