当前位置: 首页 > news >正文

Arduino CH552 ADC的使用

Arduino CH552 ADC的使用


📄CH552 ADC 简介

CH552 芯片提供 8 位的模拟数字转换器,包括电压比较器和 ADC 模块。该转换器具有 4 个模拟信号输入通道,可以分时采集,支持 0 到 VCC 模拟输入电压范围。

📗ADC 寄存器

在这里插入图片描述

    • ADC 控制寄存器(ADC_CTRL)
      在这里插入图片描述
      在这里插入图片描述
  • ADC 配置寄存器(ADC_CFG):
    在这里插入图片描述
  • ADC 数据寄存器(ADC_DATA):
    在这里插入图片描述

📗ADC 功能

  • 🌿ADC 采样模式配置步骤:
    (1)、设置 ADC_CFG 寄存器中的 ADC_EN 位为 1,开启 ADC 模块,设置 bADC_CLK 选择频率。
    (2)、设置 ADC_CTRL 寄存器中的 ADC_CHAN1/0,选择输入通道。
    (3)、可选的,清零中断标志 ADC_IF。可选的,如果使用中断模式,还需要在此使能中断。
    (4)、设置 ADC_CTRL 寄存器中的 ADC_START,启动一次 ADC 转换。
    (5)、等待 ADC_START 变为 0,或者 ADC_IF 被置 1(如果之前已清零),表示 ADC 转换结束,可通过ADC_DATA 读取结果数据。该数据是输入电压相对于 VCC 电源电压的 255 等份的值,例如,结果数据是 47,说明输入电压接近 VCC 电压的 47/255。如果 VCC 电源电压也不确定,那么可以另测一个确定的参考电压值,再按比例计算出被测的输入电压值和 VCC 电源电压值。
    (6)、如果再次设置 ADC_START 则可启动下一次 ADC 转换。
  • 🌿电压比较器模式配置步骤:
    (1)、设置 ADC_CFG 寄存器中的 CMP_EN 位为 1,开启电压比较器模块。
    (2)、设置 ADC_CTRL 寄存器中的 ADC_CHAN1/0 和 CMP_CHAN,选择正相和反相输入端。
    (3)、可选的,清零标志 CMP_IF。
    (4)、任何时候都可查询 CMPO 位的状态获得当前比较器的结果。
    (5)、如果 CMP_IF 变为 1,表示比较器的结果发生了变化。
    上述被选择的模拟信号输入通道,其所在 GPIO 引脚必须设置为高阻输入模式、或者开漏输出模式并
    且处于输出 1 的状态(相当于高阻输入),Pn_DIR_PU[x]=0,并且建议关闭上拉电阻和下拉电阻。

📝ADC 通道测试代码

✨开启ADC中断,需要在固件:C:\Users\Administrator\AppData\Local\Arduino15\packages\CH55xDuino\hardware\mcs51\0.0.23\cores\ch55xduino\main.c文件中,将PWM中断函数添加进去:void ADCInterrupt( void ) __interrupt(INT_NO_ADC);

/*ADC精度值:0 - 255AIN0(P1.1)
AIN1(P1.4)
AIN2(P1.5)
AIN3(P3.2)
*/
#include <Arduino.h>#define  LED_BUILTIN  17volatile uint16_t count = 0;  // 中断计数器
const uint16_t target = 500;  // 500次中断(每次1ms)
static bool LED_FLAG = 0;
uint16_t UserData;#define ADC_INTERRUPT  1void ADCInit(uint8_t div)
{ADC_CFG = ( ADC_CFG & ~bADC_CLK ) | div;ADC_CFG |= bADC_EN;                                                        //ADC电源使能
#if ADC_INTERRUPTADC_IF = 0;                                                                //清空中断IE_ADC = 1;                                                                //使能ADC中断
#endif
}/*******************************************************************************
* Function Name  : ADC_ChannelSelect(uint8_t ch)
* Description    : ADC采样启用
* Input          : uint8_t ch 采用通道
* Output         : None
* Return         : 成功 1失败 0
*******************************************************************************/
uint8_t ADC_ChannelSelect(uint8_t ch)
{if(ch == 0){ADC_CHAN1 =0;ADC_CHAN0=0;P1_DIR_PU &= ~bAIN0;}                 //AIN0else if(ch == 1){ADC_CHAN1 =0;ADC_CHAN0=1;P1_DIR_PU &= ~bAIN1;}            //AIN1else if(ch == 2){ADC_CHAN1 =1;ADC_CHAN0=0;P1_DIR_PU &= ~bAIN2;}            //AIN2else if(ch == 3){ADC_CHAN1 =1;ADC_CHAN0=1;P3_DIR_PU &= ~bAIN3;}            //AIN3else return 0;return 1;
}/*******************************************************************************
* Function Name  : VoltageCMPModeInit()
* Description    : 电压比较器模式初始化
* Input          : uint8_t fo 正向端口 0\1\2\3uint8_t re 反向端口 1\3
* Output         : None
* Return         : 成功 1失败 0
*******************************************************************************/
bool VoltageCMPModeInit(uint8_t fo,uint8_t re)
{ADC_CFG |= bCMP_EN;                                                        //电平比较电源使能if(re == 1){if(fo == 0) {ADC_CHAN1 =0;ADC_CHAN0=0;CMP_CHAN =0;}                      //AIN0和AIN1else if(fo == 2) {ADC_CHAN1 =1;ADC_CHAN0=0;CMP_CHAN =0;}                 //AIN2和AIN1else if(fo == 3) {ADC_CHAN1 =1;ADC_CHAN0=1;CMP_CHAN =0; }                //AIN3和AIN1			else return 0;}			     else if(re == 3){if(fo == 0) {ADC_CHAN1 =0;ADC_CHAN0=0;CMP_CHAN =0;}                      //AIN0和AIN1else if(fo == 1) {ADC_CHAN1 =0;ADC_CHAN0=1;CMP_CHAN =0;}                 //AIN1和AIN1else if(fo == 2) {ADC_CHAN1 =1;ADC_CHAN0=0;CMP_CHAN =0;}                 //AIN2和AIN1			else return 0;}else return 0;
#if ADC_INTERRUPTCMP_IF = 0;                                                                //清空中断IE_ADC = 1;                                                                //使能ADC中断
#endifreturn 1;
}#if ADC_INTERRUPT
/*******************************************************************************
* Function Name  : ADCInterrupt(void)
* Description    : ADC 中断服务程序
*******************************************************************************/
void ADCInterrupt( void ) __interrupt(INT_NO_ADC) __using(1)                       //ADC中断服务程序,使用寄存器组1
{ if(ADC_IF ==  1)                                                          //ADC完成中断{ UserData = ADC_DATA;                                                    //取走ADC采样数据ADC_IF = 0;		                                                          //清空ADC中断标志
//	  printf(" %d ",UserData);}if(CMP_IF ==  1)                                                          //电压比较完成中断{	
//       UserData = ADC_CTRL&0x80 >> 7);	                                        //保存比较器结果		CMP_IF = 0;		                                                          //清空比较器完成中断}
}
#endif// 定时器2中断服务函数
void Timer2Interrupt(void) __interrupt(INT_NO_TMR2) __using(3) 
{if (TF2) {TF2 = 0;  // 清除定时器2溢出标志count++;if (count >= target) {count = 0;LED_FLAG = !LED_FLAG;// 执行500ms任务digitalWrite(LED_BUILTIN, LED_FLAG);  // 翻转LED}}}void Timer2_init()
{// 配置定时器2T2CON = 0x00;    // 清除T2CON,设置为16位自动重载模式T2MOD = 0x00;    // 时钟=Fsys(24MHz)// 计算定时器初值(24MHz时钟,1ms中断)uint16_t timerValue = 65536 - 1000;//1msRCAP2L = timerValue & 0xFF;   // 低8位RCAP2H = (timerValue >> 8);   // 高8位// 装载初值TL2 = RCAP2L;TH2 = RCAP2H;// 使能定时器2中断ET2 = 1;  // 使能定时器2中断EA = 1;   // 使能全局中断// 启动定时器2TR2 = 1;
}// the setup function runs once when you press reset or power the board
void setup() {// initialize digital pin LED_BUILTIN as an output.pinMode(LED_BUILTIN, OUTPUT);digitalWrite(LED_BUILTIN, LOW);pinMode(LED_PIN, OUTPUT);digitalWrite(LED_PIN, LOW);  // 初始化为低电平(LED熄灭)ADCInit( 0 ); Timer2_init();
}// the loop function runs over and over again forever
void loop() {
#if ADC_INTERRUPT                                                              //ADC中断方式EA = 1;for(uint8_t i=0;i<4;i++){	USBSerial_print("AIN ");USBSerial_flush();USBSerial_print(i);USBSerial_flush();ADC_ChannelSelect( i );                                                //ADC采样电源开启和通道设置,i(0-3)表示采样通道ADC_START = 1;                                                         //开始采样,采样完成进入中断delay(100);  //>=30us等待采集完成才能切换至下一通道USBSerial_print(":");USBSerial_flush();USBSerial_println(UserData);USBSerial_flush();delay(1000);}#else	for(uint8_t i=0;i<4;i++){				USBSerial_print("AIN:");USBSerial_flush();USBSerial_print(i);USBSerial_flush();ADC_ChannelSelect( i );                                                //ADC采样初始化ADC_START = 1;                                                         //开始采样,采样完成进入中断while(ADC_START);                                                      //ADC_START变为0时,表示采样完成USBSerial_println(ADC_DATA);USBSerial_flush();//等待串口数据发送结束(串口数据发送必不可少)delay(100);                                                         //模拟单片机干其他事 }		
#endif//   USBSerial_println("Hello World123!");// USBSerial_flush();delay(1000);                       // wait for a second//   USBSerial_println("perseverance51");// wait for a second//   USBSerial_flush();//等待串口数据发送结束(串口数据发送必不可少)
}
http://www.dtcms.com/a/264968.html

相关文章:

  • NumPy 或 PyTorch/TensorFlow 中的张量理解
  • Servlet开发流程(包含IntelliJ IDEA项目添加Tomcat依赖的详细教程)
  • 【论文阅读】DeepEyes: Incentivizing “Thinking with Images” via Reinforcement Learning
  • 【新手小白的嵌入式学习之路】-STM32的学习_GPIO 8种模式学习心得
  • JavaWeb笔记03
  • GC393:一款低功耗双电压比较器芯片
  • 设计模式-责任链模式
  • SpringBoot控制反转
  • vue中的toRef
  • 【Redis】StringRedisTemplate 和 RedisTemplate 的区别
  • Python 数据分析与可视化 Day 14 - 建模复盘 + 多模型评估对比(逻辑回归 vs 决策树)
  • JavaEE==网站开发
  • Liunx 安装 MySQL 8.0
  • Selenium使用教程-爬虫版(超详细)
  • 数学建模_图论
  • 重塑智能体决策路径:深入理解 ReAct 框架
  • 【前端进阶】【实战】【性能优化】前端开发中的事件监听与DOM操作优化实践
  • Linux基本命令篇 —— whereis命令
  • 利用 Claude Opus 4 自动化 GitHub 工作流:从安装到实战详解
  • 新版本AI数字人全能管家,即将推出,还是开源免费。
  • [附源码+数据库+毕业论文+答辩PPT]基于Spring+MyBatis+MySQL+Maven+vue实现的中小型企业财务管理系统,推荐!
  • 多个单片机简单通讯框架
  • GO Web 框架 Gin 完全解析与实践
  • 数据结构与算法 第三章 栈和队列
  • 第一章 快速入门
  • DPI深度检索原理和架构
  • 人脸活体识别3:C/C++实现人脸眨眼 张嘴 点头 摇头识别(可实时检测)
  • 创客匠人解构知识付费爆单密码:产品力打造与 IP 变现的深度耦合
  • Kafka高级特性深度解析:构建企业级流处理平台的核心技术
  • IP地理定位技术综述:理论、方法与应用创新(三)