IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> STM8单片机ADC带缓存的连续采样模式 -> 正文阅读

[嵌入式]STM8单片机ADC带缓存的连续采样模式

??在上一篇文章中说了STM8的ADC连续采样模式,为了提高采样的精度和速率,STM8单片机还提供了带缓存的连续采样模式,也就是说ADC会连续采集8个数据,放在缓存中,读取数据时可以一次从缓存中读取8个数据,这样就可以通过8个数据数据计算平均值,使得采样的结果更加准确。

下面看一下官方文档中的对缓存模式的介绍。

image.png

??通过文档中可以看出,要开启缓存模式,只需要将ADC_CR3寄存器中的COUNT为DBUF设置为1,就可以开启缓存模式了。
??当开启缓存模式后,采样的结果将不会存放在ADC_DR寄存器中,而是会将结果依次存放在ADC_DB0R寄存器到ADCDB7R寄存器,连续读取8次数据,存储在这8个寄存器中。读取数据的时候,依次从这8个寄存器中读取就行。

下面直接通过代码来实现带缓存功能的连续采样模式:

#include "adc.h"
#include "main.h"

u16  DATAH = 0;                          //ADC转换值高8位
u16  DATAL = 0;                          //ADC转换值低8位
_Bool ADC_flag = 0;                     //ADC转换成功标志

//AD通道引脚初始化
void ADC_GPIO_Init( void )
{
    PD_DDR &= ~( 1 << 3 );              //PD3 设置为输入      电流
    PD_CR1 &= ~( 1 << 3 );              //PD3 设置为悬空输入
}
/*
ch 为单片机ADC通道
通过置位ADC_CR1寄存器的ADON位来开启ADC。当第一次置位ADON时,ADC从低功耗模式唤醒。
为了启动转换必须第二次使用写指令来置位ADON位。
在转换结束时,ADC会保持在上电状态,用户只需要置位ADON位一次来启动下一次转换。
转换完成后,转换数据存储在ADC_DR寄存器中,EOC(转换结束)标志被置位,如果EOCIE被置位将产生一个中断
ADC输入通道初始化入口参数表示通道选择
*/
void ADC_CH_Init( u8 ch )
{
    char l = 0;
    ADC_CR1  = 0x00;                    //fADC = fMASTER/2, 8Mhz  单次转换,禁止转换
    
    ADC_CR1 |= ( 1 << 1 );              //开启连续转换模式
    
    ADC_CSR  = ch;                      //控制状态寄存器 选择要 AD输入通道  如:PD2(AIN3)
    ADC_CR2  = 0x00;                    //默认左对齐 读数据时先读高在读低
    ADC_TDRL = ( 1 << ch );             //禁止相应通道 施密特触发功能 1左移ch+1位
    ADC_CR1 |= 0x01;                    //使能ADC并开始转换

    //ADC_CSR |= 0x20;                    //EOCIE 使能转换结束中断  EOC中断使能
    ADC_CR3 |= ( 1 << 7 );              //数据缓存使能

    for( l = 0; l < 100; l++ );         //延时,保证ADC模块的上电完成 至少7us
    ADC_CR1 = ADC_CR1 | 0x01;           //再次将CR1寄存器的最低位置1 使能ADC 并开始转换
}

u16 databuf[8] = {0};
//采集PD2电压值
u16 ReadVol_CH3( void )
{
    u16 voltage = 0;

    while( ( ADC_CSR & 0x80 ) == 0 );      //等待转换结束
    if( ADC_CSR & 0x80 )
    {
        ADC_CSR &= 0x7F;
        /*
        使能缓存模式后,数据会存储在 ADC_DB0R ---- ADC_DB7R 寄存器中
        如果使能了扫描模式那么这几个寄存器存储的就是对应通道的数据
        如果没有使能扫描模式,那么这几个通道就存储的是连续转换的结果
        这里没有使用扫描模式,所以缓存器中存储的都是当前通道连续读取的数据
        */
        DATAH = ADC_DB0RH;                    // 读出ADC结果的高8位
        DATAL = ADC_DB0RL;                    // 读出ADC结果的低8位
        voltage = ( DATAH << 2 ) + DATAL ;    //得到十位精度的数据  0--1024
        databuf[0] = voltage;

        DATAH = ADC_DB1RH;
        DATAL = ADC_DB1RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[1] = voltage;

        DATAH = ADC_DB2RH;
        DATAL = ADC_DB2RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[2] = voltage;

        DATAH = ADC_DB3RH;
        DATAL = ADC_DB3RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[3] = voltage;

        DATAH = ADC_DB4RH;
        DATAL = ADC_DB4RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[4] = voltage;

        DATAH = ADC_DB5RH;
        DATAL = ADC_DB5RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[5] = voltage;

        DATAH = ADC_DB6RH;
        DATAL = ADC_DB6RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[6] = voltage;

        DATAH = ADC_DB7RH;
        DATAL = ADC_DB7RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[7] = voltage;
    }

    return voltage;
}

在初始化中设置 ADC_CR1 |= ( 1 << 1 ); 也就是将ADC_CR1寄存器的第一位设置为1。

image.png

也就是使能了连续转换模式。

然后设置ADC_CR3 |= ( 1 << 7 ); 也就是将ADC_CR3寄存器的第7位设置为1.

image.png

也就是开启了ADC的数据缓存功能,当数据转换完成之后,ADC_CSR寄存器的EOC位会置1,此时如果程序判断到EOC位为1时,就可以去数据缓存寄存器读取数据了。也就是代码中ReadVol_CH3()函数实现的功能。

image.png

这里是通过查询标志位的方法读取数据,也可以通过中断的方式读取数据。

通过中断方式读取代码如下:

#include "adc.h"
#include "main.h"

u16  DATAH = 0;                          //ADC转换值高8位
u16  DATAL = 0;                          //ADC转换值低8位
_Bool ADC_flag = 0;                     //ADC转换成功标志

//AD通道引脚初始化
void ADC_GPIO_Init( void )
{
    PD_DDR &= ~( 1 << 3 );              //PD3 设置为输入      电流
    PD_CR1 &= ~( 1 << 3 );              //PD3 设置为悬空输入
}
/*
ch 为单片机ADC通道
通过置位ADC_CR1寄存器的ADON位来开启ADC。当第一次置位ADON时,ADC从低功耗模式唤醒。
为了启动转换必须第二次使用写指令来置位ADON位。
在转换结束时,ADC会保持在上电状态,用户只需要置位ADON位一次来启动下一次转换。
转换完成后,转换数据存储在ADC_DR寄存器中,EOC(转换结束)标志被置位,如果EOCIE被置位将产生一个中断
ADC输入通道初始化入口参数表示通道选择
*/
void ADC_CH_Init( u8 ch )
{
    char l = 0;
    ADC_CR1  = 0x00;                    //fADC = fMASTER/2, 8Mhz  单次转换,禁止转换
    
    ADC_CR1 |= ( 1 << 1 );              //开启连续转换模式
    
    ADC_CSR  = ch;                      //控制状态寄存器 选择要 AD输入通道  如:PD2(AIN3)
    ADC_CR2  = 0x00;                    //默认左对齐 读数据时先读高在读低
    ADC_TDRL = ( 1 << ch );             //禁止相应通道 施密特触发功能 1左移ch+1位
    ADC_CR1 |= 0x01;                    //使能ADC并开始转换

    ADC_CSR |= 0x20;                    //EOCIE 使能转换结束中断  EOC中断使能
    ADC_CR3 |= ( 1 << 7 );              //数据缓存使能

    for( l = 0; l < 100; l++ );         //延时,保证ADC模块的上电完成 至少7us
    ADC_CR1 = ADC_CR1 | 0x01;           //再次将CR1寄存器的最低位置1 使能ADC 并开始转换
}

u16 databuf[8] = {0};
//采集PD2电压值
u16 ReadVol_CH3( void )
{
    u16 voltage = 0;

    if( ADC_flag )
    {
        ADC_flag = 0;
        /*
        使能缓存模式后,数据会存储在 ADC_DB0R ---- ADC_DB7R 寄存器中
        如果使能了扫描模式那么这几个寄存器存储的就是对应通道的数据
        如果没有使能扫描模式,那么这几个通道就存储的是连续转换的结果
        这里没有使用扫描模式,所以缓存器中存储的都是当前通道连续读取的数据
        */
        DATAH = ADC_DB0RH;                    // 读出ADC结果的高8位
        DATAL = ADC_DB0RL;                    // 读出ADC结果的低8位
        voltage = ( DATAH << 2 ) + DATAL ;    //得到十位精度的数据  0--1024
        databuf[0] = voltage;

        DATAH = ADC_DB1RH;
        DATAL = ADC_DB1RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[1] = voltage;

        DATAH = ADC_DB2RH;
        DATAL = ADC_DB2RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[2] = voltage;

        DATAH = ADC_DB3RH;
        DATAL = ADC_DB3RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[3] = voltage;

        DATAH = ADC_DB4RH;
        DATAL = ADC_DB4RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[4] = voltage;

        DATAH = ADC_DB5RH;
        DATAL = ADC_DB5RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[5] = voltage;

        DATAH = ADC_DB6RH;
        DATAL = ADC_DB6RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[6] = voltage;

        DATAH = ADC_DB7RH;
        DATAL = ADC_DB7RL;
        voltage = ( DATAH << 2 ) + DATAL ;
        databuf[7] = voltage;     
    }
    return voltage;
}


//AD中断服务函数 中断号22
#pragma vector = 24                     // IAR中的中断号,要在STVD中的中断号上加2
__interrupt void ADC_Handle( void )
{
    ADC_CSR &= ~0x80;                   // 转换结束标志位清零  EOC
    ADC_flag = 1;                       // ADC中断标志 置1
}

当数据采集完成后,就会产生一次中断,然后在中断中设置一个标志位,然后ReadVol_CH3()函数发现标志位置1后,就直接读取采样的数据。

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2021-08-15 15:45:24  更:2021-08-15 15:45:30 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 20:47:52-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码