打印
[Zigbee]

9、ZigBee之AD剖析——AD采集CC2530温度串口显示

[复制链接]
1146|28
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
1、ADC 简介
  ADC 支持多达14 位的模拟数字转换,具有多达12 位有效数字位。它包括一个模拟多路转换器,具有多达8 个各自可配置的通道;以及一个参考电压发生器。转换结果通过DMA 写入存储器。还具有若干运行模式。
  ADC 的主要特性如下:
● 可选的抽取率,这也设置了分辨率(7 到12 位)
● 8 个独立的输入通道,可接受单端或差分信号
● 参考电压可选为内部单端、外部单端、外部差分或AVDD5
● 产生中断请求
● 转换结束时的DMA 触发
● 温度传感器输入
● 电池测量功能

使用特权

评论回复

相关帖子

沙发
phosphate|  楼主 | 2020-1-11 17:25 | 只看该作者
2、ADC 操作

  本节描述了ADC 的一般安装和操作,并描述了CPU 存取的ADC 控制和状态寄存器的使用。

使用特权

评论回复
板凳
phosphate|  楼主 | 2020-1-11 17:25 | 只看该作者
2.1、ADC 输入

  The signals on the Port 0 pins can be used as ADC inputs. In the following, these port pins are referred to as the AIN0–AIN7 pins. The input pins AIN0–AIN7 are connected to the ADC.

  可以把输入配置为单端或差分输入。在选择差分输入的情况下,差分输入包括输入对AIN0-1、AIN2-3、AIN4-5 和AIN6-7。电压不能为负或者大于VDD。这些输入对之间的区别书他们采用不同的模式进行转换。

  除了输入引脚AIN0-AIN7,片上温度传感器的输出也可以选择作为ADC 的输入,用于温度测量。为此寄存器TR0.ADCTM 和ATEST.ATESTCTRL 必须分别按2.10 节和寄存器描述所述设置。

  还可以输入一个对应AVDD5/3 的电压作为一个ADC 输入。这个输入允许诸如需要在应用中实现一个电池监测器的功能。注意在这种情况下参考电压不能取决于电源电压,比如AVDD5 电压不能用作一个参考电压。

  单端电压输入AIN0 到AIN7 以通道号码0 到7 表示。通道号码8 到11 表示差分输入,由AIN0–AIN1、AIN2–AIN3、AIN4–AIN5 和AIN6–AIN7 组成。通道号码12 到15 表示G N D(12)温度传感器(14),和AVDD5/3(15)。这些值在ADCCON2.SCH 和ADCCON3.SCH 域中使用。

使用特权

评论回复
地板
phosphate|  楼主 | 2020-1-11 17:26 | 只看该作者
2.2、ADC 转换序列(暂时难理解)

  ADC将执行一系列的转换,并把结果移动到存储器(通过DMA),不需要任何CPU 干预。

  转换序列可以被APCFG 寄存器影响,八位模拟输入来自I/O 引脚,不必经过编程变为模拟输入。如果一个通道正常情况下应是序列的一部分,但是相应的模拟输入在APCFG 中禁用,那么通道将被跳过。当使用差分输入,处于差分对的两个引脚都必须在APCFG 寄存器中设置为模拟输入引脚。

  The ADCCON2.SCH(用于定义转换序列) register bits are used to define an ADC conversion sequence from the ADC inputs.

  If ADCCON2.SCH is set to a value less than 8, the conversion sequence contains a conversion from each channel from 0 up to and including the channel number programmed in ADCCON2.SCH.(当设置该寄存器值小于8时,转换序列为从通道0到SCH定义的值,包括该值)

  When ADCCON2.SCH is set to a value between 8 and 12, the sequence consists of differential inputs, starting at channel 8 and ending at the programmed channel.(在8~12之间,为缠粉输入,通道从8到定义的值)

  For ADCCON2.SCH greater than or equal to 12, the sequence consists of the selected channel only.(如果大于12,则定义哪个就是哪个)

  PS:the channel define in the last of 2.1

使用特权

评论回复
5
phosphate|  楼主 | 2020-1-11 17:26 | 只看该作者
2.3、Single ADC Conversion(单个ADC转换,2.2是设置一个ADC序列进行转换)

  In addition to this sequence of conversions, the ADC can be programmed to perform a single conversion from any channel(ADC能够配置从任何一个channel开执行一次单通道转换). Such a conversion is triggered by writing to the ADCCON3 register. (转换开始的条件->)The conversion starts immediately unless a conversion sequence is already ongoing, in which case the single conversion is performed as soon as that sequence is finished.

使用特权

评论回复
6
phosphate|  楼主 | 2020-1-11 17:27 | 只看该作者
2.4、ADC Operating Modes

  本节描述:operating modes and initialization of conversions.

  The ADC has three control registers: ADCCON1, ADCCON2, and ADCCON3. These registers are used to configure the ADC and to report status.

1、ADCCON1.EOC 位是一个状态位,当一个转换结束时,设置为高电平;当读取ADCH 时,它就被清除。
2、ADCCON1.ST 位用于启动一个转换序列。当这个位设置为高电平,ADCCON1.STSEL 是11,且当前没有转换正在运行时,就启动一个序列。当这个序列转换完成,这个位就被自动清除。
3、ADCCON1.STSEL 位选择哪个事件将启动一个新的转换序列。该选项可以选择为外部引脚P2.0 上升沿或外部引脚事件,之前序列的结束事件,定时器1 的通道0 比较事件或ADCCON1.ST 是1。
4、ADCCON2 寄存器控制转换序列是如何执行的。
5、ADCCON2.SREF 用于选择参考电压。参考电压只能在没有转换运行的时候修改。
6、ADCCON2.SDIV 位选择抽取率(并因此也设置了分辨率和完成一个转换所需的时间,或样本率)。抽取率只能在没有转换运行的时候修改。
7、转换序列的最后一个通道由ADCCON2.SCH 位选择,如上所述。
8、ADCCON3 寄存器控制单个转换的通道号码、参考电压和抽取率。单个转换在寄存器ADCCON3 写入后将立即发生,或如果一个转换序列正在进行,该序列结束之后立即发生。该寄存器位的编码和ADCCON2 是完全一样的。

使用特权

评论回复
7
phosphate|  楼主 | 2020-1-11 17:28 | 只看该作者
2.5、ADC 转换结果

  数字转换结果以2 的补码形式表示。对于单端配置,结果总是为正。这是因为结果是输入信号和地面之间的差值,它总是一个正符号数(Vconv=Vinp-Vinn,其中Vinn=0V)。当输入幅度等于所选的电压参考VREF时,达到最大值。

  对于差分配置,两个引脚对之间的差分被转换,这个差分可以是负符号数。对于抽取率是512的一个数字转换结果的12 位MSB,当模拟输入Vconv 等于VREF 时,数字转换结果是2047。当模拟输入等于
-VREF 时,数字转换结果是-2048。

  当ADCCON1.EOC 设置为1 时,数字转换结果是可以获得的,且结果放在ADCH 和ADCL 中。注意转换结果总是驻留在ADCH 和ADCL 寄存器组合的MSB 段中。

  当读取ADCCON2.SCH 位时,它们将指示转换在哪个通道上进行。ADCL 和ADCH 中的结果一般适用于之前的转换。如果转换序列已经结束, ADCCON2.SCH 的值大于最后一个通道号码,但是如果最后写入ADCCON2.SCH 的通道号码是12 或更大,将读回同一个值。

使用特权

评论回复
8
phosphate|  楼主 | 2020-1-11 17:28 | 只看该作者
2.6、ADC 参考电压

  模拟数字转换的正参考电压可选择为一个内部生成的电压,AVDD5 引脚,适用于AIN7 输入引脚的外部电压,或适用于AIN6-AIN7 输入引脚的差分电压。

  转换结果的准确性取决于参考电压的稳定性和噪音属性。希望的电压有偏差会导致ADC 增益误差,与希望电压和实际电压的比例成正比。参考电压的噪音必须低于ADC 的量化噪音,以确保达到规定的SNR。

使用特权

评论回复
9
phosphate|  楼主 | 2020-1-11 17:28 | 只看该作者
2.7、ADC 转换时间

  ADC 只能运行在32 MHz XOSC 上,用户不能整除系统时钟。实际ADC 采样的4 MHz 的频率由固定的内部划分器产生。执行一个转换所需的时间取决于所选的抽取率。总的来说,转换时间由以下公式给定:

Tconv = (抽取率+ 16) x 0.25 μs。

使用特权

评论回复
10
phosphate|  楼主 | 2020-1-11 17:29 | 只看该作者
2.8、ADC 中断

  当通过写ADCCON3 触发的一个单个转换完成时,ADC 将产生一个中断。当完成一个序列转换时,不产生一个中断。

使用特权

评论回复
11
phosphate|  楼主 | 2020-1-11 17:29 | 只看该作者
2.9、ADC DMA 触发(和ADC中断有种互补的感觉~)

  每完成一个序列转换,ADC 将产生一个DMA 触发。当完成一个单个转换,不产生DMA 触发。

  There is one DMA trigger for each of the eight channels defined by the first eight possible settings for ADCCON2.SCH。当通道中一个新的样本准备转换,DMA 触发是活动的。The DMA triggers are named ADC_CHsd in Following Table, where s is single-ended channel and d is differential channel。

  In addition, one DMA trigger, ADC_CHALL, is active when new data is ready from any of the channels in the ADC conversion sequence.

使用特权

评论回复
12
phosphate|  楼主 | 2020-1-11 17:30 | 只看该作者
3、工程解析

从下面main函数可以看出,整个流程是先初始化串口收发(这个和上一节介绍的串口收发一模一样,请参考上一节);接着是初始化ADC将片上片上温度传感器的输出选择作为ADC 的输入用于温度测量;在while大循环内则是连续读取64次温度数据并求平均(代码中求平均方法有点怪),最后通过串口将采集的片内温度传感器数据输出。
 void main(void) 
{   
     char i;
     float AvgTemp;   
     char strTemp[6];
     
     InitUART();                           //初始化串口
     InitSensor();                         //初始化 ADC
     
     while(1)
     {
         AvgTemp = GetTemperature();   
         
         for (i=0; i<63; i++)
         {   
             AvgTemp += GetTemperature();  
             AvgTemp = AvgTemp/2;          //每次累加后除 2
         }
        
         memset(strTemp, 0, 6);
         sprintf(strTemp,"%.02f", AvgTemp);//将浮点数转成字符串
         UartSendString(strTemp, 5);       //通过串口发给电脑显示芯片温度
         DelayMS(1000);                    //延时
     }
}

使用特权

评论回复
13
phosphate|  楼主 | 2020-1-11 17:31 | 只看该作者
其中initSensor()是对ADC进行初始化:

第3行#define DISABLE_ALL_INTERRUPTS() (IEN0 = IEN1 = IEN2 = 0x00)是关闭所有中断;

第4行为设置系统主时钟为32M,上一节中main函数最前面做的工作;

1 void InitSensor(void)
2 {
3    DISABLE_ALL_INTERRUPTS();     //关闭所有中断
4    InitClock();                  //设置系统主时钟为 32M
5    TR0=0x01;                     //设置为1来连接温度传感器到SOC_ADC
6    ATEST=0x01;                   //使能温度传感
7 }   
第5行TR0=1为设置温度传感器连接到SOC_ADC:



第6行ATEST=1为使能温度传感器:

使用特权

评论回复
14
phosphate|  楼主 | 2020-1-11 17:32 | 只看该作者
其中GetTemperature()函数用来获取温度传感器AD的值:

 /****************************************************************************
* 名    称: GetTemperature()
* 功    能: 获取温度传感器 AD 值
* 入口参数: 无
* 出口参数: 通过计算返回实际的温度值
****************************************************************************/
float GetTemperature(void)
{
    uint  value;
   
    ADCCON3  = (0x3E);            //选择1.25V为参考电压;14位分辨率;对片内温度传感器采样
    ADCCON1 |= 0x30;              //选择ADC的启动模式为手动
    ADCCON1 |= 0x40;              //启动AD转化  
    while(!(ADCCON1 & 0x80));     //等待 AD 转换完成
    value =  ADCL >> 4;           //ADCL 寄存器低 2 位无效,由于他只有12位有效,ADCL寄存器低4位无效。网络上很多代码这里都是右移两位,那是不对的
    value |= (((uint)ADCH) << 4);
   
    return (value-1367.5)/4.5-5;  //根据 AD 值,计算出实际的温度,芯片手册有错,温度系数应该是4.5 /℃
                                  //进行温度校正,这里减去5℃(不同芯片根据具体情况校正)
}

使用特权

评论回复
15
phosphate|  楼主 | 2020-1-11 17:32 | 只看该作者
其中ADCCON3设置为0x3E,即选择内联参考电压,512采样率(12位有效位数),对片内温度传感器单通道采样!

使用特权

评论回复
16
phosphate|  楼主 | 2020-1-11 17:33 | 只看该作者
其中 ADCCON1 |= 0x30 即Start select. Selects the event that starts a new conversion sequence(ADC启动模式为手动)

其中 ADCCON1 |= 0x40 即Start a conversion sequence if ADCCON1.STSEL = 11 and no sequence is running(启动ADC转换)

其中 while(!(ADCCON1 & 0x80)) 即等待一次转换完成

使用特权

评论回复
17
phosphate|  楼主 | 2020-1-11 17:33 | 只看该作者
第15、16行:是获得ADC采样的12位有效数据的值保存在value中

15    value =  ADCL >> 4;           //ADCL 寄存器低 2 位无效,由于他只有12位有效,ADCL寄存器低4位无效。网络上很多代码这里都是右移两位,那是不对的
16    value |= (((uint)ADCH) << 4);


但是value值只是ADC值,并不是温度值,需要转换,代码18、19行就是完成转换:(至于怎么算的我猜测应该有个公式对应!)

18    return (value-1367.5)/4.5-5;  //根据 AD 值,计算出实际的温度,芯片手册有错,温度系数应该是4.5 /℃
19                                  //进行温度校正,这里减去5℃(不同芯片根据具体情况校正)

使用特权

评论回复
18
phosphate|  楼主 | 2020-1-11 17:33 | 只看该作者
4、实验现象

将程序烧入CC2530,用USB连接开发板与PC,可以用串口助手观察zigbee发来的温度数据,当用手触摸芯片时温度会有明显变化:

使用特权

评论回复
19
phosphate|  楼主 | 2020-1-11 17:34 | 只看该作者
Zigbee系列**:

[ZigBee] 1、 ZigBee简介   https://bbs.21ic.com/icview-2894000-1-1.html

[ZigBee] 2、 ZigBee开发环境搭建   https://bbs.21ic.com/icview-2894002-1-1.html

[ZigBee] 3、ZigBee基础实验——GPIO输出控制实验-控制Led亮灭   https://bbs.21ic.com/icview-2894004-1-1.html

[ZigBee] 4、ZigBee基础实验——中断   https://bbs.21ic.com/icview-2894006-1-1.html

[ZigBee] 5、ZigBee基础实验——图文与代码详解定时器1(16位定时器)  https://bbs.21ic.com/icview-2894008-1-1.html

[ZigBee] 6、ZigBee基础实验——定时器3和定时器4(8 位定时器)  https://bbs.21ic.com/icview-2894896-1-1.html

[ZigBee] 7、ZigBee之UART剖析(ONLY串口发送)   https://bbs.21ic.com/icview-2894898-1-1.html

[ZigBee] 8、ZigBee之UART剖析·二(串口收发)   https://bbs.21ic.com/icview-2894904-1-1.html

使用特权

评论回复
20
zljiu| | 2020-2-1 10:46 | 只看该作者
非常感谢楼主分享

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

32

主题

393

帖子

1

粉丝