CC2430 ADC使用——查询法
1 目标
熟悉使用CC2430的ADC功能。根据我自己开发板的情况,我使用P07作为AD转换的输入口,使用一个旋转电位器来调整输入端口的电压,通过串口发送AD转换结果。在这里还是说说ADC的结构。
CC2430的ADC是基于sigma-delta原理,而不是常用的逐次比较式,通过不同的抽取率来实现不同的转换精度。
2 代码总览
还是老规矩,先列出所有的代码。在这里除了使用到ADC模块,还使用了定时器和串口模块,串口模块用来输出转换结果,定时器模块用来间隔调用ADC转换函数。具体的代码如下所示:
//包含头文件 #include "hal.h" #include "stdio.h" //函数申明 UINT8 UART0_Init(); UINT8 ADC_Init(); UINT8 Timer1_Init(); UINT16 ADC_Convert(); void main(){ //使用外部晶振 SET_MAIN_CLOCK_SOURCE(CRYSTAL); //设定IO口 IO_DIR_PORT_PIN(0, 7, IO_IN); //初始化定时器 Timer1_Init(); //初始化串口 UART0_Init(); //初始化ADC ADC_Init(); //输出提示 printf("CC2430 ADC Test\n"); //无线循环 while(1){ } } UINT8 ADC_Init(){ //选择AD转换通道 ADC_ENABLE_CHANNEL(ADC_AIN7); //选择参考电压,分辨率,通道 ADC_SINGLE_CONVERSION(ADC_REF_AVDD | ADC_10_BIT | ADC_AIN7); return 0; } UINT16 ADC_Convert(){ //转换结果高位和低位 UINT8 adc_h; UINT8 adc_l; UINT16 adc_value; //开始转换 ADC_SAMPLE_SINGLE(); //等待转化结束 while(!ADC_SAMPLE_READY()); //获得转换结果 adc_h = ADCH; adc_l = ADCL; //获得AD转换结果,10位结果 adc_value = (( adc_h << 8) | adc_l) >> 6; //输出转换结果 printf("ADC 10bit = %d\n",adc_value); return adc_value ; } UINT8 UART0_Init(){ //UART0 IO口定位 IO_PER_LOC_UART0_AT_PORT0_PIN2345(); //UART0参数9600 8 N 1 UART_SETUP(0,9600,HIGH_STOP); UTX0IF = 1; return 0; } int putchar(int c){ if(c== '\n'){ while(!UTX0IF); UTX0IF = 0; U0DBUF = '\r'; } while(!UTX0IF); UTX0IF = 0; U0DBUF = c; return c; } UINT8 Timer1_Init(){ //定时器1复位 TIMER1_INIT(); //设定定时器相关参数 //溢出值低8位 T1CC0L=0x24; //溢出值高8位 T1CC0H=0xF4; //128分频0000 1100 T1CTL = 0x0c; //定时器T1溢出中断使能 TIMER1_ENABLE_OVERFLOW_INT(TRUE); //定时器T1中断使能 INT_ENABLE(INUM_T1,INT_ON); //启动定时器1 TIMER1_RUN(TRUE); //全局中断使能,注意 INT_GLOBAL_ENABLE(INT_ON); return 0; } //定时器1 中断函数 #pragma vector=T1_VECTOR __interrupt void T1_ISR(void) { //检查中断标志位 if(T1CTL & 0x10){ //ADC 通道参数初始化 ADC_Init(); //启动转换,通过串口输出结果 ADC_Convert(); //清中断标志 T1CTL &= ~0x10; } }
3 初始化其他内容
//使用外部晶振 SET_MAIN_CLOCK_SOURCE(CRYSTAL); //设定IO口 IO_DIR_PORT_PIN(0, 7, IO_IN); //初始化定时器 Timer1_Init(); //初始化串口 UART0_Init();
其他的不多说了,在使用AD转换功能之前,需要定义该IO口为输入状态。使用这个宏就可以了。IO_DIR_PORT_PIN(0, 7, IO_IN);
4 初始化ADC
//初始化ADC ADC_Init(); ADC_Init的函数如下所示: UINT8 ADC_Init(){ //选择AD转换通道 ADC_ENABLE_CHANNEL(ADC_AIN7); //选择参考电压,分辨率,通道 ADC_SINGLE_CONVERSION(ADC_REF_AVDD | ADC_10_BIT | ADC_AIN7); return 0; }
初始化ADC可以分为2步,
第一步,使能ADC转换通道(IO特性),在这里我们选择通道7,使用了一个动作宏
#define ADC_ENABLE_CHANNEL(ch) ADCCFG |= (0x01<<ch);
该宏操作了ADCCFG寄存器,这个寄存器的说明位于IO部分,而不是ADC部分。
第二步,选择ADC的参考电压,转换分辨率和ADC通道。在这里使用了另一个宏定义:
#define ADC_SINGLE_CONVERSION(settings) \ do{ ADCCON3 = settings; }while(0) // Reference voltage: #define ADC_REF_1_25_V 0x00 // Internal 1.25V reference #define ADC_REF_P0_7 0x40 // External reference on AIN7 pin #define ADC_REF_AVDD 0x80 // AVDD_SOC pin #define ADC_REF_P0_6_P0_7 0xC0 // External reference on AIN6-AIN7 differential input // Resolution (decimation rate): #define ADC_7_BIT 0x00 // 64 decimation rate #define ADC_9_BIT 0x10 // 128 decimation rate #define ADC_10_BIT 0x20 // 256 decimation rate #define ADC_12_BIT 0x30 // 512 decimation rate // Input channel: #define ADC_AIN0 0x00 // single ended P0_0 #define ADC_AIN1 0x01 // single ended P0_1 #define ADC_AIN2 0x02 // single ended P0_2 #define ADC_AIN3 0x03 // single ended P0_3 #define ADC_AIN4 0x04 // single ended P0_4 #define ADC_AIN5 0x05 // single ended P0_5 #define ADC_AIN6 0x06 // single ended P0_6 #define ADC_AIN7 0x07 // single ended P0_7 #define ADC_GND 0x0C // Ground #define ADC_TEMP_SENS 0x0E // on-chip temperature sensor #define ADC_VDD_3 0x0F // (vdd/3)
所有的参数都可以在数据手册中,ADCCON3部分找到,这里不多做说明。需要说明的一点是,原书代码中(ZigBee技术实践教程)转换分辨率的定义为8,10,12,14,通过我个人的多次试验和资料查证,分辨率实际为7,9,10,12,数据左对齐,以补码的形式保存。所以这里定义为10位分辨率时,最大的结果为511,最小的结果为-512。但是这里是不会有负结果出现的。(这个和实验的结果也是吻合的)
5 进行AD转换
UINT16 ADC_Convert(){ //转换结果高位和低位 UINT8 adc_h; UINT8 adc_l; UINT16 adc_value; //开始转换 ADC_SAMPLE_SINGLE(); //等待转化结束 while(!ADC_SAMPLE_READY()); //获得转换结果 adc_h = ADCH; adc_l = ADCL; //获得AD转换结果,10位结果 adc_value = (( adc_h << 8) | adc_l) >> 6; //输出转换结果 printf("ADC 10bit = %d\n",adc_value); return adc_value ; }
这里使用了最简单的等待方法,ADC还可以使用其他方法,比如说中断或者DMA传送等。先从简单的来,完成一次ADC转换可以分为3步:
第一步:启动AD转换
#define ADC_SAMPLE_SINGLE() \ do { ADC_STOP(); ADCCON1 |= 0x40; } while (0) #define ADC_STOP() \ do { ADCCON1 |= 0x30; } while (0)
第二步:等待AD转换结束
只需要查询标志位就可以了。
#define ADC_SAMPLE_READY() (ADCCON1 & 0x80)
ADCCON1的7位置位时代表AD转换完成,否则不断等待。
第三步:结果处理
AD转换的结果保存在ADCH和ADCL寄存器中,先把两个寄存器组成一个16位长度的整形数据,然后使用移位算法获得相应的结果,本例中使用了10位数据,所以右移6位即可。
转换完成后使用printf输出结果,这里就体现了串口的好处了。最大的作用方便调试。
6 定时输出结果
由于完成单次的转换没有意义,所以需要定时完成转换。在这里使用了定时器1,定时器1的使用前面的文章给你已经说过,这里不再重复。中断服务函数如下所示:
#pragma vector=T1_VECTOR __interrupt void T1_ISR(void) { //检查中断标志位 if(T1CTL & 0x10){ //ADC 通道参数初始化 ADC_Init(); //启动转换,通过串口输出结果 ADC_Convert(); //清中断标志 T1CTL &= ~0x10; } }
还请大家注意,每次转换完成后必须重新选择通道。数据手册中关于ADCCON3有明确提到。我一开始也以为只要设定好了,这个通道号是不会变化的,但是后面发现这样做转换的结果总是不变,打印了数据手册一页一页看才明白过来。
7 输出结果
没有实验结果的示例是站不住脚的。我把旋转定位器的先拧到最大,使7通道的输入电压最大,也就是达到VDD(3.3V),然后再慢慢减小。测试的结果如下图所示:
通过转动电位器使电压不断减小,转换的结果也随之不断减小。还可以看出,转换的最大结果为511,符合预计结果。幸好没有出现512,否则我又要郁闷好久。为了这个AD转换我还做了很多实验,在不同的抽取率情况下取出ADC结果,然后画图分析,这个以后再写文章说明。
CC2430串口使用说明
CC2430定时器使用说明
CC2430 hal.h