CC2541蓝牙学习——ADC

本文详细介绍了CC2541芯片中的ADC模块特性及其应用。内容涵盖ADC的主要特性、工作模式、输入配置、温度测量功能、序列及单次转换、中断与DMA触发等。此外,还提供了ADC配置与读取的示例代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转载于http://www.cnblogs.com/chenzhao207/p/4539197.html

CC2541的ADC支持多达14位的模拟数字转换与高达12位的有效位数。它包括一个模拟多路转换器,具有多达8个各自可独立配置的通道,一个参考电压发生器。转换结果通过DMA写入存储器。还具有若干运行模式。

ADC主要特性如下:

  1. 可选的抽取率,设置了7~12位的分辨率;
  2. 8个独立输入通道,可接受单端或差分信号;
  3. 参考电压可选为内部,外部单端,外部差分,或AVDD5;
  4. 产生中断请求;
  5. 转换结束时的DMA触发;
  6. 温度传感器输入;
  7. 电池测量功能。

                                                                    图1

P0引脚上的信号可以作为ADC输入来使用。在下面,这些引脚叫做AIN0—AIN7引脚,输入脚AIN0—AIN7与ADC连接。

输入脚可配置成单端或差动输入。如选择差动输入,包含成对输入AIN0-AIN1,AIN2-AIN3,AIN4-AIN5和AIN6-AIN7;注意这些引脚既不能加载负电压,也不能加载大于VDD的电压。

除了输入脚AIN0-AIN7外,片上的温度传感器也可以用来作为ADC温度测量的输入。如要实现这个功能,需设置寄存器TR0.ADCTM和ATEST.ATESTCTRL。

单端输入AIN0至AIN7可代表通道号0至7,通道号8至11分别代表差动输入AIN0-AIN1,AIN2-AIN3,AIN4-AIN5,AIN6-AIN7;通道12表示GND,通道13表示温度传感器,通道15表示AVDD5/3。这些值在ADCCON2.SCH和ADCCON3.SCH中设置。

我们看到ADCCON2和ADCCON3这两个寄存器的定义基本相同,但是用法不同,ADCCON2用于ADC序列转换的配置,而ADCCON3则用于单个ADC通道的配置。所谓ADC序列就是多个ADC通道按照次序分别转换。注意:不是同时转换的,从图1我们也可以看出,ADC的模拟输入接一个选择器,同一时刻只能选择一个通道接入进行ADC转换

如果选择片上的温度传感器作为ADC温度测量的输入,则需要通过配置寄存器TR0和ATEST来获得片上温度,不过这个温度测量误差很大,我们一般不用,这里也就不给出例程了。

启用片内温度采集配置寄存器:

1 TR0 |= 0x01;
2 ATEST |= 0x01;

 

1、ADC序列转换

ADC序列转换无需CPU的参与,ADC能够完成一个序列的转换,并通过DMA把结果写入内存。

寄存器APCFG影响转换序列,来自I/O引脚的8位模拟输入不一定是程序设置的模拟输入。如某一通道是序列的一部分,但在APCFG中相应模拟输入是禁止的,那此通道将被跳过。当使用差动输入时,两个输入脚在APCFG寄存器中必须被设置成模拟输入。

ADCCON2.SCH用来定义ADC输入的转换序列。如ADCCON2.SCH被设为小于8,转换序列包含一个通道(从0到ADCCON2.SCH中设置的通道号),当ADCCON2.SCH值设为8至12时,序列是差动输入,从通道8至程序设置的通道号;当大于12时,序列包含只选择的通道。

 

2、单个ADC转换

除了序列转换外,ADC可以通过编程执行单个转换。通过写入ADCCON3寄存器可以触发一个转换,转换立即启动,除非一个转换序列正在进行中,这种情况下,当序列完成后,马上执行单个转换。

3、寄存器ADCCON1

ADC的数字转换结果可以通过寄存器ADCCON1获得,寄存器ADCCON1的定义如下图所示。

  • ADCCON1.EOC:转换结束状态位,当转换结束时设高电平,当读取ADCH时低电平。
  • ADCCON1.ST位用来启动序列转换的,当这位设高电平、ADCCON1.STSEL是11且当前无转换运行时序列启动开始。当序列转换结束时,这位自动清除为低电平。
  •  ADCCON1.STSEL位用来选择哪个事件将启动一个新的序列转换。此项选择有:外部引脚P2.0上升沿事件,之前序列的结束事件,定时器通道0比较事件,或ADCCON1.ST设1事件。

 

4、ADC转换结果

数字转换结果以2进制补码形式表示的,最高位是符号位。

对于单端输入配置,由于ADC输入不能接负电压,转换结果总是正的当输入信号等于参考电压VREF时达到最大转换结果。

对于差分输入配置,ADC输入电压为两个引脚的电压之差,两脚的输入信号不同,结果可能是负的;当采样率为512,模拟输入Vconv=VREF时,12MSB的数字转换结果为2047,当模拟输入等于-VREF时,转换结果为-2048。

通过读ADCCON2.SCH位,知道正在转换的是哪个通道,在序列转换中,ADCL和ADCH中的结果是前一个通道ADC转换的值。如转换序列已结束,ADCCON2.SCH将有一个大于最后通道数一个以上的值,但如最后写入ADCCON2.SCH中的通道数是12或更大,读回的是相同的值。

 

5、ADC参考电压

模数转换的参考电压可选择于内部产生电压,AVDD5脚电压,应用于AIN7输入脚的外部电压,或应用于AIN6-AIN7输入的差动电压。内部参考电压对于CC2541来说是1.25V,比较小,能转换的最大模拟电压最大也只能是1.25V,AVDD5脚电压一般为3.3V,精度也不是很高。转换结果的准确度依靠于参考电压的稳定性和噪声度,所以对于要求较高的ADC转换建议从AIN7输入脚接入高精度的参考电压。

 

6、ADC转换时间

ADC只能运行于32MHZ XOSC。执行一个转换的时间依靠于被选择的采样率,一般上,转换时间由以下公式所得:

Tconv=(decimation rate+16)*0.25us.

可见分辨率越高,转换时间越长。

 

7、ADC中断

只有单通道ADC转换才有ADC中断,序列ADC转换没有ADC中断。

The ADC generates an interrupt when a single conversion triggered by writing to ADCCON3 has completed.No interrupt is generated when a conversion from the sequence is completed.

 

8、ADC DMA触发

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

在ADCCON2.SCH中设置8个通道,每个通道都有一个DMA触发。当通道转换中准备好一个采样时,将激活一个DMA触发。DMA触发命名为ADC_CHsd,s是单端通道,d是差动通道。

另外,当ADC序列转换通道中准备好一个新数据时,一个DMA触发(ADC_CHALL)将激活。

 

单个ADC转换读取ADC值的程序如下:

复制代码
 1 /******************************************************************************
 2 *函 数 名:InitADC
 3 *功    能:ADC初始化
 4 *入口参数:参考电压 reference、转换通道 channel、分辨率resolution
 5 *出口参数:ADC转换结果
 6 ******************************************************************************/
 7 uint Read_advalue(uchar reference, uchar channel, uchar  resolution)
 8 {
 9   uint value; 
10   uchar tmpADCCON3 = ADCCON3;
11   
12   APCFG |= 1 << channel ; //设置ADC输入通道,模拟I/O使能
13   
14   ADCCON3  = (reference | resolution | channel);
15   ADCIF = 0;                         //
16 
17   while(!ADCIF);               //等待 AD 转换完成 
18   value =  ADCL >> 2;          //ADCL 寄存器低 2 位无效
19   value |= ((uint)ADCH << 6);  //连接AD转换结果高位和低位 
20   
21 //根据分辨率获得ADC转换结果有效位 
22   switch(resolution)
23   {
24     case ADC_7_BIT:  value >>= 7;break;
25     case ADC_9_BIT:  value >>= 5;break;
26     case ADC_10_BIT: value >>= 4;break;
27     case ADC_12_BIT: value >>= 2;break;
28     default:;
29   }
30   
31   ADCCON3 = tmpADCCON3;
32   return (value);
33 }
复制代码

主程序:采集VDD值。

复制代码
 1 /******************************************************************************
 2 *程序入口函数
 3 ******************************************************************************/
 4 int main(void)
 5 { 
 6   uint vddvalue;       //ADC转换值
 7 
 8   InitClock();         //32MHz时钟
 9   InitUART();          //UART0串口初始化
10   
11   while(1)
12   {
13 //ADC参考电压AVDD5引脚电源电压:3.3V,分辨率12位,采集通道:VDD/3,VDD=3.3V
14     vddvalue = Read_advalue(ADC_REF_AVDD5, 0x0f, ADC_12_BIT);
15     vddvalue = (vddvalue*33) >> 11;
16     vddvalue = vddvalue*3;
17     buf[0] = vddvalue/10 + '0';
18     buf[1] = '.';
19     buf[2] =vddvalue%10 + '0';
20 
21     UartSendString(buf,strlen(buf));  //串口上传采样VDD值
22     Delay1ms(2000);                   //每隔2s上传一次值
23   }
24 }
复制代码

这里给出协议栈的adc转换函数参照对比。

 View Code

 

调试结果:显示VDD值3.3V。

 

关于程序注意以下几点:

1、要配置一个端口0脚为一个ADC输入,APCFG寄存器中相应的位必须设置为1。这个寄存器的默认值选择端口0引脚为非ADC,即数字输入输出。APCFG寄存器的设置将覆盖P0SEL的设置,所以无需再配置P0SEL,另外对于I/O口作为外设功能,都无需配置方向,即无需配置寄存器PxDIR。

 

2、对于单次ADC转换的配置,只需要配置寄存器ADCCON3,无需配置寄存器ADCCON1和ADCCON2。对于判断转换是否结束,还有一种判断方法:

1   ADCCON1 |=0X30;              //ADC启动方式选择为ADCCON1.ST=1事件
2   ADCCON1 |= 0x40;             //启动转换
3   while(!(ADCCON1 & 0x80));    //等待 AD 转换完成 

ADCCON1.STSEL是用于启动转换序列的触发方式的,对于单次ADC转换,个人感觉这样配置不好,以后对于单次ADC转换,不采用这种判断方式。

单次转换判断是否转换结束:判断ADC中断标志ADCIF。

 

3、ADCH的最高位是符号位,对于单次测量,结果总是正的,所以符号位总是0。14位的ADC转换值有效值并不是14位的。

有效分辨率如下:
00: 64 decimation rate (7 bits ENOB)----ADCH低7位
01: 128 decimation rate (9 bits ENOB)---ADCH低7位+ADCH高2位
10: 256 decimation rate (10 bits ENOB)--ADCH低7位+ADCH高3位
11: 512 decimation rate (12 bits ENOB)--ADCH低7位+ADCL高5位

例如:采集VDD/3值时,使用12位分辨率,参考电压AVDD5:3.3V

VDD/3 = vddvalue*3.3/2^11
扩大10倍
VDD/3 = vddvalue*33/2^11
为什么是除以2^11而不是2^12,因为最高位是符号位,12位分辨率实际上只有11位。
VDD = (vddvalue*33/2^11) * 3

4、差分输入可以用来做比较器。比如通道ADCCON3.ECH=1000,对应差分输入AIN0-AIN1。如果要比较一个模拟信号和另一个模拟信号的大小关系,只需要将这两个信号分别接入AIN0和AIN1,然后判断ADCH的最高位,如果是1,则AIN0<AIN1,如果是0,则AIN0>=AIN1。

 

5、最大转换电压等于参考电压,而参考电压的选择不能大于芯片的电源电压,一般为3.3V。虽然差分输入可以转换负电压,但是每一个模拟输入引脚都必须是正电压且小于电源电压VDD,负电压是指两个输入通道的差值。

### 单片机电子秤通过蓝牙与微信小程序的数据交互方案 为了实现单片机控制的电子秤通过蓝牙与微信小程序之间的数据交互,可以从以下几个方面展开: #### 1. **硬件设计** 硬件部分主要包括单片机、HX711芯片以及蓝牙模块。其中,单片机负责接收来自HX711的压力信号并进行初步处理,而蓝牙模块则用于无线传输这些数据至手机端的小程序。 - HX711作为高精度24位ADC模数转换器,在此场景中被用来采集负载单元产生的电信号,并将其转化为数字量供单片机读取[^3]。 - 蓝牙模块的选择需考虑功耗、通信距离等因素。常用的有HC-05/HC-06系列经典蓝牙模块或者BLE(Bluetooth Low Energy)类型的模块如CC2540、nRF51822等。对于低延迟实时性较高的应用场景建议优先采用支持BLE协议栈的产品[^2]。 #### 2. **固件编程** 针对所选型号编写相应的驱动代码完成初始化配置工作之后即可进入核心逻辑环节——即如何把经过计算后的最终测量结果打包发送出去并通过串口传递给外部设备也就是这里的蓝牙收发装置。 以下是简单的伪代码框架展示整个流程概览: ```c #include <reg52.h> // 假设使用的是STC89C52RC单片机 sbit LED=P1^0; // 定义指示灯引脚 unsigned char buffer[10]; void main(){ init_hardware(); // 初始化硬件资源包括定时器中断设置等等... while(1){ float weight=getWeightFromHX711(); if(weight>=threshold){ turnOnLed(LED); sprintf(buffer,"%.2f",weight); sendViaBluetooth(buffer,strlen((char*)buffer)); } delay_ms(500); // 添加适当延时减少CPU占用率 } } ``` 以上仅为示意性质并不完全适用于具体项目实施过程中还需要根据实际需求调整参数定义等内容[^3]。 #### 3. **微信小程序开发** 在客户端一侧,则需要利用腾讯官方提供的WeChat Mini Program Framework来构建图形化操作界面允许用户查看上传来的最新称重记录同时也能够下达某些特定指令反向影响物理世界中的行为动作比如开启关闭电源等功能开关选项卡之类的东西。 大致结构如下所示: ```javascript Page({ data:{ currentWeight:null, command:'' }, onLoad:function(options){ }, onShow:function(){ }, receiveData:async function(event){ const res=await wx.requestSocketMessage({url:'/api/socket'}); this.setData({ currentWeight:Number(res.data.value) }); }, sendCommand:function(e){ let cmd=e.detail.value; wx.sendSocketMessage({ url:'/api/command', method:"POST", header:{'content-type':'application/json'}, body:JSON.stringify({"action":cmd}) }) } }); ``` 注意这里只是非常基础的概念验证版本可能无法满足生产环境下的严格要求因此后续还需进一步优化完善诸如错误捕获机制安全性加固等方面的工作[^1]. --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值