zoukankan      html  css  js  c++  java
  • 【STM32H7教程】第93章 STM32H7的SPI总线应用之驱动ADS1256(8通道24bit ADC, 增益可编程)

    完整教程下载地址:http://www.armbbs.cn/forum.php?mod=viewthread&tid=86980

    第93章       STM32H7的SPI总线应用之驱动ADS1256(8通道24bit ADC, 增益可编程)

    本章节为大家讲解8通道24bit ADC芯片驱动实现。

    93.1 初学者重要提示

    93.2 ADC结构分类

    93.3 ADS1256硬件设计

    93.4 ADS1256关键知识点整理(重要)

    93.5 ADS1256驱动设计

    93.6 ADS1256板级支持包(bsp_spi_ads1256)

    93.7 ADS1256实际测试效果(10uV抖动)

    93.8 ADS1256驱动移植和使用

    93.9 实验例程设计框架

    93.10 实验例程说明(MDK)

    93.11 实验例程说明(IAR)

    93.12 总结

    93.1 初学者重要提示

    1.   ADS1256的模拟部分供电5V,SPI数字接口电平3.3V。
    2.   ADS1256的PGA可以编程增益支持: 1、2、4、8、16、32、64。
    3.   ADS1256支持自动校准 (当设置了PGA,BUF使能、数据采样率时,会启动自校准)。
    4.   ADS1256支持8通道单端ADC采集或者4通道差分采集。
    5.   ADS1256支持正负5V差分采集,但不支持负压,使用时要注意。
    6.   ADS1256时序操作稍有点特殊,所以本章是采用的模拟SPI控制。
    7.   ADS1256数据手册,模块原理图(通用版)和接线图都已经放到本章教程配置例子的Doc文件里。
    8.   ADC 的专业术语诠释文档,推荐大家看看:http://www.armbbs.cn/forum.php?mod=viewthread&tid=89414
    9.   测试时,务必使用外置电源为开发板供电,因为ADS1256需要5V供电电压。板子上插入ADS1256模块时,注意对齐。

    93.2 ADC结构分类

    这里将六种DAC结构为大家做个普及。注,这些知识翻译自美信和TI的英文技术手册。

     

     

    93.2.1 SAR ADC(逐次逼近型)

    逐次逼近型ADC通常是中高分辨率的首选架构,采样速率通常低于5Msps。SAR ADC最常见的分辨率范围是8位到20位,并具有低功耗和小尺寸的特点。这种组合使其非常适合各种应用,例如自动测试设备,电池供电的设备,数据采集系统,医疗仪器,电机和过程控制,工业自动化,电信,测试和测量,便携式系统,高速闭环系统和窄带接收器。

    93.2.2 Sigma-Delta ADC

    Sigma-delta ADC主要用于低速应用中,该应用需要通过过采样来权衡速度和分辨率,然后进行滤波以降低噪声。24位sigma-delta转换器用于自动化测试设备,高精度便携式传感器,医疗和科学仪器以及地震数据采集等应用中。

    93.2.3 Integrating ADC

    集成ADC提供高分辨率,并且可以提供良好的线路频率和噪声抑制。集成架构提供了一种新颖且直接的方法,可将低带宽模拟信号转换为数字表示形式。这些类型的转换器通常包括用于LCD或LED显示器的内置驱动器,并且在许多便携式仪器应用中都可以找到,包括数字面板表和数字万用表。

    93.2.4 FLASH ADC

    Flash ADC是将模拟信号转换为数字信号的最快方法。它们适用于需要非常大带宽的应用。然而,闪存转换器功率高,具有相对较低的分辨率,并且可能非常昂贵。这将它们限制在通常无法以其他任何方式解决的高频应用中。示例包括数据采集,卫星通信,雷达处理,示波器和高密度磁盘驱动器。

    93.2.5 Pipelined ADC

    流水线ADC已成为最受欢迎的ADC体系结构,其采样率从每秒几兆采样(MS / s)到最高100MS / s +,分辨率为8至16位。它们提供的分辨率和采样率,可覆盖各种应用,包括CCD成像,超声医学成像,数字接收器,基站,数字视频(例如HDTV),xDSL,电缆调制解调器和快速以太网。

    93.2.6 Two Step ADC

    两步ADC也称为子范围转换器,有时也称为多步或half flash(比Flash架构慢)。这是Flash ADC和流水线ADC的交叉点。与Flash ADC相比,可以实现更高的分辨率或更小的裸片尺寸。

    93.3 ADS1256硬件设计

    这里将开发板上的ADS1256硬件接口,ADS1256模块为大家做个说明。

    ADS1256的原理图论坛下载:

    http://www.armbbs.cn/forum.php?mod=viewthread&tid=97547

    93.3.1 ADS1256硬件接口

    V7板子上ADS1256模块的插座的原理图如下:

     

    实际对应开发板的位置如下:

     

    为了方便大家更好的理解接线,下面是框图:

     

    93.3.2 ADS1256模块

    产品规格:

    1、 单电源5.0V DC供电,提供正负5V信号采样功能。

    2、 MCU接口:SPI。

    3、 主ADC芯片:ADS1256  (全新进口原装正品)。

    4、 电压基准,之前采用的LM285-2.5,现在采用的REF5025(全新进口原装正品)。

    5、 输入电路带分压电阻和R-C滤波,方便客户自己变更增益范围。

    6、 芯片内带可编程增益放大器,增益范围:1-64倍。

    7、 芯片内部输入带缓冲放大器,可以直接连接传感器。

    正面:

     

     

    反面:

     

     

    接线图:

     

     

    93.4 ADS1256关键知识点整理(重要)

    驱动ADS1256需要对下面这些知识点有个认识。

    93.4.1 ADS1256基础信息

    ADS1256是TI公司推出的微功耗、高精度、8通道、 24位△-∑型高性能模数转换器(ADC)。该器件提供高达23比特的无噪声精度、数据速率高达30kSPS(次采样/秒)、0.0010%非线性特性(最大值)以及众多的板上外设(输入模拟多路开关、输入缓冲器、可编程增益放大器和可编程数字滤波器等),可为设计人员带来完整而高分辨率的量测解决方案。

    •   24位无数据丢失;
    •   高达23比特的无噪声精度;
    •   低非线性度: ±0.0010% ;
    •   数据采样率可达30kSPS ;
    •   采用单周期转换模式;
    •   带有模拟多路开关,具有传感器接口(可配置为4路差动输入或8路单极输入) ;
    •   带有输入缓冲器( BUF) ;
    •   带有串行外设接口( SPI) ;
    •   内含低噪声可编程增益放大器( PGA ) ,所有的PGA均具有自校准和系统校准;
      •  PGA= 1时,可提供高达25.3位的有效分辨率;
      •  PGA = 64时,可提供高达22.5位的有效分辨率;
    •   模拟输入电压为5V ,数字电压为1.8~3.6V ;
    •   正常模式下功耗低至38mW ,备用模式下功耗为0.4mW。

    93.4.2 ADS1256常用引脚的作用

    ADS1256的封装形式:

     

     

    这里把常用的几个引脚做个说明:

    •   AVDD

    模拟电源供电。

    •   AGND

    模拟地。

    •   VREFN

    负参考电压输入。

    •   VREFP

    正参考电压输入。

    •   AINCOM

    模拟公共输入。

    •   AIN0 – AIN7

    模拟输入通道0到通道7

    •   SYNC/PDWN

    同步,掉电输入。

    •   RESET

    复位引脚。

    •   DVDD

    数字电源。

    •   DGND

    数字地。

    •   XTAL1,XTAL2

    晶振输入端。

    •   CS

    片选输入端。

    •   DRDY

    数据就绪输出信号引脚。

    •   DOUT

    数据输出。

    •   DIN

    数据输入。

    •   SCLK

    时钟引脚。

    •   D0,D1,D2,D3

    通用GPIO

    93.4.3 ADS1256输出电压计算公式

    ADS1256的计算公式如下:

     

     

    最小单位值是2 * VREF/(PGA * (2^23 − 1))

    采用二进制补码表示(其实就是24bit有符号数,我们将转换结果定义为int32_t即可)。

    93.4.4 ADS1256时序图

    驱动ADS1256主要是两个时序图需要了解,读写时序:

     

     

    •   t1:SPI时钟周期:

    外部晶振频率 = 7.68MHz,

                  时钟频率 tCLK = 1/7.68M = 0.13uS

                  输出数据周期 tDATA =  1 / 30K = 0.033mS  (按30Ksps计算)

           对SPI的时钟速度要求:

                  最快 4个tCLK = 0.52uS

                  最慢 10个tDATA = 0.3mS (按 30Ksps 计算)

     

     

    •   t2H,t2L :脉冲高低电平:

    SCL高电平和低电平持续时间最小 200ns。

     

     

    数据读取流程:

     

     

    •   第1步:ADS1256_SetChannal(g_tADS1256.Channel)切换模拟通道。
    •   第2步:发送SYNC同步命令。
    •   第3步:唤醒。
    •   第4步:读取数据。

    这个过程中特别注意读取的是上次转换的数据。

    93.4.5 ADS1256的增益和测量范围问题

    ADS1256的增益和测量范围关系如下:

     

     

    比如增益是1时,测量范围是正负5V,增益是64时,测量范围是正负78.125mV。

    93.4.6 ADS1256输入缓冲器

    开关输入缓冲器时,影响到的几个参数,大家需要做个了解。

    开缓冲的情况下,输入参考值噪声。

     

     

    关闭缓冲时的输入参考噪声:

     

     

    开缓冲的情况下,有效分辨率:

     

     

    关闭缓冲时的有效位数:

     

     

    打开缓冲器后的输入阻抗:

     

     

     

    关闭缓冲器后的输入阻抗:

     

     

     

    93.4.7 ADS1256支持的采样率

    ADS1256支持的采样率如下,这里特别注意,因为切换通道和读数据耗时 123微秒, 因此扫描中断模式工作时,最大速率 = DRATE_1000SPS。

     

     

    93.4.8 ADS1256的多路选择器,单端和多端输入

    ADS1256的8路支持是通过多路选择器实现,8路采样时是选择相应通道进行采样:

     

     

    实际应用的的8路单端采样和4路差分采样效果如下:

     

     

    93.5 ADS1256驱动设计

    ADS1256的程序驱动框架设计如下:

     

    有了这个框图,程序设计就比较好理解了。

    93.5.1 第1步,ADS1256所涉及到的GPIO配置

    这里需要把用到的GPIO时钟、GPIO引脚配置好:

    /*
        ADS1256模块    STM32-V7开发板(示波器接口)
          +5V   <------  5.0V      5V供电
          GND   -------  GND       地
          DRDY  ------>  PC6       准备就绪
          CS    <------  PC7       SPI_CS
          DIN   <------  PG10      SPI_MOSI
          DOUT  ------>  PA5       SPI_MISO
          SCLK  <------  PA4       SPI时钟
          GND   -------  GND       地
          PDWN  <------  PB7       掉电控制
          RST   <------  PC3       复位信号
          NC   空脚
          NC   空脚
    */
    
    #ifdef SOFT_SPI        /* 软件SPI */
        /* 定义GPIO端口 */    
        #define SCK_CLK_ENABLE()     __HAL_RCC_GPIOA_CLK_ENABLE()
        #define SCK_GPIO            GPIOA
        #define SCK_PIN            GPIO_PIN_4
        #define SCK_1()            SCK_GPIO->BSRR = SCK_PIN
        #define SCK_0()            SCK_GPIO->BSRR = ((uint32_t)SCK_PIN << 16U)    
    
        #define DIN_CLK_ENABLE()     __HAL_RCC_GPIOG_CLK_ENABLE()
        #define DIN_GPIO            GPIOG
        #define DIN_PIN            GPIO_PIN_10
        #define DIN_1()            DIN_GPIO->BSRR = DIN_PIN
        #define DIN_0()            DIN_GPIO->BSRR = ((uint32_t)DIN_PIN << 16U)    
    
        #define CS_CLK_ENABLE()     __HAL_RCC_GPIOC_CLK_ENABLE()
        #define CS_GPIO            GPIOC
        #define CS_PIN            GPIO_PIN_7
        #define CS_1()            CS_GPIO->BSRR = CS_PIN
        #define CS_0()            CS_GPIO->BSRR = ((uint32_t)CS_PIN << 16U)    
    
        #define DOUT_CLK_ENABLE()     __HAL_RCC_GPIOA_CLK_ENABLE()
        #define DOUT_GPIO            GPIOA
        #define DOUT_PIN            GPIO_PIN_5
        #define DOUT_IS_HIGH()        ((DOUT_GPIO->IDR & DOUT_PIN) != 0)
    
        #define DRDY_CLK_ENABLE()     __HAL_RCC_GPIOC_CLK_ENABLE()
        #define DRDY_GPIO            GPIOC
        #define DRDY_PIN            GPIO_PIN_6
        #define DRDY_IS_LOW()        ((DRDY_GPIO->IDR & DRDY_PIN) == 0)
        #define DRDY_IRQn             EXTI9_5_IRQn
        #define DRDY_IRQHandler        EXTI9_5_IRQHandler    
    
        /* PDWN  <------  PB7       掉电控制 */
        #define PWDN_CLK_ENABLE()     __HAL_RCC_GPIOB_CLK_ENABLE()
        #define PWDN_GPIO            GPIOB
        #define PWDN_PIN            GPIO_PIN_7
        #define PWDN_1()            PWDN_GPIO->BSRR = PWDN_PIN
        #define PWDN_0()            PWDN_GPIO->BSRR = ((uint32_t)PWDN_PIN << 16U)            
        
        /*  RST   <------  PC3       复位信号     */
        #define RST_CLK_ENABLE()     __HAL_RCC_GPIOC_CLK_ENABLE()
        #define RST_GPIO            GPIOC
        #define RST_PIN            GPIO_PIN_3
        #define RST_1()            RST_GPIO->BSRR = RST_PIN
        #define RST_0()            RST_GPIO->BSRR = ((uint32_t)RST_PIN << 16U)            
    #endif
    
    /*
    *********************************************************************************************************
    *    函 数 名: bsp_InitADS1256
    *    功能说明: 配置STM32的GPIO和SPI接口,用于连接 ADS1256
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void bsp_InitADS1256(void)
    {
    #ifdef SOFT_SPI
        GPIO_InitTypeDef gpio_init;
        
        RST_1();
        PWDN_1();
        CS_1();
        SCK_0();        /* SPI总线空闲时,钟线是低电平 */
        DIN_1();
    
        /* 打开GPIO时钟 */
        SCK_CLK_ENABLE();
        DIN_CLK_ENABLE();
        CS_CLK_ENABLE();
        DOUT_CLK_ENABLE();
        DRDY_CLK_ENABLE();
        PWDN_CLK_ENABLE();
        RST_CLK_ENABLE();
    
        /* 配置几个推完输出IO */
        gpio_init.Mode = GPIO_MODE_OUTPUT_PP;        /* 设置推挽输出 */
        gpio_init.Pull = GPIO_NOPULL;                /* 上下拉电阻不使能 */
        gpio_init.Speed = GPIO_SPEED_FREQ_HIGH;      /* GPIO速度等级 */        
        
        gpio_init.Pin = SCK_PIN;    
        HAL_GPIO_Init(SCK_GPIO, &gpio_init);    
    
        gpio_init.Pin = DIN_PIN;    
        HAL_GPIO_Init(DIN_GPIO, &gpio_init);    
        
        gpio_init.Pin = CS_PIN;    
        HAL_GPIO_Init(CS_GPIO, &gpio_init);    
    
        gpio_init.Pin = PWDN_PIN;    
        HAL_GPIO_Init(PWDN_GPIO, &gpio_init);    
    
        /* DRDY 设置为输入 */
        gpio_init.Mode = GPIO_MODE_INPUT;        /* 设置输入 */
        gpio_init.Pull = GPIO_NOPULL;            /* 上下拉电阻不使能 */
        gpio_init.Speed = GPIO_SPEED_FREQ_HIGH;      /* GPIO速度等级 */
        
        gpio_init.Pin = DRDY_PIN;    
        HAL_GPIO_Init(DRDY_GPIO, &gpio_init);    
    
        gpio_init.Pin = DOUT_PIN;    
        HAL_GPIO_Init(DOUT_GPIO, &gpio_init);    
    #endif
    }

    这里重点注意DRDY转换就绪引脚的配置,DRDY_IRQn和DRDY_IRQHandler不要配置错了。

    93.5.2 第2步,ADS1256的8bit读写函数实现

    读写函数实现如下:

    /*
    *********************************************************************************************************
    *    函 数 名: ADS1256_Send8Bit
    *    功能说明: 向SPI总线发送8个bit数据。 不带CS控制。
    *    形    参: _data : 数据
    *    返 回 值: 无
    *********************************************************************************************************
    */
    static void ADS1256_Send8Bit(uint8_t _data)
    {
        uint8_t i;
    
        /* 连续发送多个字节时,需要延迟一下 */
        ADS1256_DelaySCLK();
        ADS1256_DelaySCLK();
    
        /* ADS1256 要求 SCL高电平和低电平持续时间最小 200ns  */
        for(i = 0; i < 8; i++)
        {
            if (_data & 0x80)
            {
                DIN_1();
            }
            else
            {
                DIN_0();
            }
            SCK_1();                
            ADS1256_DelaySCLK();        
            _data <<= 1;        
            SCK_0();            /* <----  ADS1256 是在SCK下降沿采样DIN数据, 数据必须维持 50nS */
            ADS1256_DelaySCLK();        
        }
    }
    
    /*
    *********************************************************************************************************
    *    函 数 名: ADS1256_Recive8Bit
    *    功能说明: 从SPI总线接收8个bit数据。 不带CS控制。
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    static uint8_t ADS1256_Recive8Bit(void)
    {
        uint8_t i;
        uint8_t read = 0;
    
        ADS1256_DelaySCLK();
        /* ADS1256 要求 SCL高电平和低电平持续时间最小 200ns  */
        for (i = 0; i < 8; i++)
        {
            SCK_1();
            ADS1256_DelaySCLK();
            read = read<<1;
            SCK_0();
            if (DOUT_IS_HIGH())
            {
                read++;
            }        
            ADS1256_DelaySCLK();
        }
        return read;
    }

    读写的实现完全按照下面的时序实现:

     

     

    这里主要注意时间实现:

    t1:SPI时钟周期:

    外部晶振频率 = 7.68MHz,

                  时钟频率 tCLK = 1/7.68M = 0.13uS

                  输出数据周期 tDATA =  1 / 30K = 0.033mS  (按30Ksps计算)

           对SPI的时钟速度要求:

                  最快 4个tCLK = 0.52uS

                  最慢 10个tDATA = 0.3mS (按 30Ksps 计算)

     

     

    t2H,t2L :脉冲高低电平:

    SCL高电平和低电平持续时间最小 200ns。

     

     

    93.5.3 第3步,ADS1256的24bit ADC数据读取

    实现代码如下:

    /*
    *********************************************************************************************************
    *    函 数 名: ADS1256_ReadData
    *    功能说明: 读ADC数据
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    static int32_t ADS1256_ReadData(void)
    {
        uint32_t read = 0;
    
        CS_0();    /* SPI片选 = 0 */
    
        ADS1256_Send8Bit(CMD_RDATA);    /* 读数据的命令 */
        
        ADS1256_DelayDATA();    /* 必须延迟才能读取芯片返回数据 */
    
        /* 读采样结果,3个字节,高字节在前 */
        read = ADS1256_Recive8Bit() << 16;
        read += ADS1256_Recive8Bit() << 8;
        read += ADS1256_Recive8Bit() << 0;
    
        CS_1();    /* SPI片选 = 1 */
        
        /* 负数进行扩展。24位有符号数扩展为32位有符号数 */
        if (read & 0x800000)
        {
            read += 0xFF000000;
        }
        
        return (int32_t)read;
    }

    这段代码里面关键是24bit数据的补码处理。对负数进行扩展,24位有符号数扩展为32位有符号数。

    93.5.4 第3步,ADS1256增益和采样率配置

    代码如下:

    /*
    *********************************************************************************************************
    *    函 数 名: ADS1256_CfgADC
    *    功能说明: 配置ADC参数,增益和数据输出速率
    *    形    参: _gain : 支持增益参数。
    *                    ADS1256_GAIN_1
    *                    ADS1256_GAIN_2    
    *                    ADS1256_GAIN_4
    *                    ADS1256_GAIN_8
    *                    ADS1256_GAIN_16
    *                    ADS1256_GAIN_32
    *                    ADS1256_GAIN_64
    *
    *             _drate : 数据输出速率,不推荐超过1000SPS
    *                    ADS1256_30000SPS
    *                    ADS1256_15000SPS
    *                    ADS1256_7500SPS
    *                    ADS1256_3750SPS
    *                    ADS1256_2000SPS
    *                    ADS1256_1000SPS
    *                    ADS1256_500SPS
    *                    ADS1256_100SPS
    *                    ADS1256_60SPS
    *                    ADS1256_50SPS
    *                    ADS1256_30SPS
    *                    ADS1256_25SPS
    *                    ADS1256_15SPS
    *                    ADS1256_10SPS
    *                    ADS1256_5SPS
    *                    ADS1256_2d5SPS
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void ADS1256_CfgADC(ADS1256_GAIN_E _gain, ADS1256_DRATE_E _drate)
    {    
        g_tADS1256.Gain = _gain;
        g_tADS1256.DataRate = _drate;
        
        ADS1256_StopScan();            /* 暂停CPU中断 */
        
        ADS1256_ResetHard();        /* 硬件复位 */
    
        ADS1256_WaitDRDY();
    
        {
            uint8_t buf[4];        /* 暂存ADS1256 寄存器配置参数,之后连续写4个寄存器 */
        
            buf[0] = (0 << 3) | (1 << 2) | (1 << 1);
            
            buf[1] = 0x08;    /* 高四位0表示AINP接 AIN0,  低四位8表示 AINN 固定接 AINCOM */
    
            buf[2] = (0 << 5) | (0 << 2) | (_gain << 1);
    
            /* 因为切换通道和读数据耗时 123uS, 因此扫描中断模式工作时,最大速率 = DRATE_1000SPS */
            buf[3] = s_tabDataRate[_drate];    // DRATE_10SPS;    /* 选择数据输出速率 */
            
            CS_0();    /* SPI片选 = 0 */
            ADS1256_Send8Bit(CMD_WREG | 0);    /* 写寄存器的命令, 并发送寄存器地址 */
            ADS1256_Send8Bit(0x03);            /* 寄存器个数 - 1, 此处3表示写4个寄存器 */
            
            ADS1256_Send8Bit(buf[0]);    /* 设置状态寄存器 */
            ADS1256_Send8Bit(buf[1]);    /* 设置输入通道参数 */
            ADS1256_Send8Bit(buf[2]);    /* 设置ADCON控制寄存器,增益 */
            ADS1256_Send8Bit(buf[3]);    /* 设置输出数据速率 */
            
            CS_1();    /* SPI片选 = 1 */        
        }
    
        bsp_DelayUS(50);    
    }

    这个函数主要配置了4个ADS1256寄存器。

    •   设置状态寄存器:

     

     

    程序中的配置为:buf[0] = (0 << 3) | (1 << 2) | (1 << 1) ,意思是LSB传输,自动校准,使能模拟输入缓冲。

    •   设置输入通道参数

     

     

    程序中的配置为:buf[1] = 0x08, 高四位0表示AINp接通的 AIN0,  低四位8表示 AINn接通的 AINCOM。AINp和AINn表示当前的多路选择器选通的AIN0:

     

     

    •   设置ADCON控制寄存器,主要用于增益设置

     

     

    程序中的配置为:buf[2] = (0 << 5) | (0 << 2) | (_gain << 1), 意思是关闭CLKOUT引脚输出,关闭传感器检测,设置增益为形参_gain。

    •   设置ADC采样率

     

     

    程序中的配置为:buf[3] = s_tabDataRate[_drate],用于设置波特率:

    static const uint8_t s_tabDataRate[ADS1256_DRATE_MAX] = 
    {
        0xF0,        /* 复位时缺省值 */
        0xE0,
        0xD0,
        0xC0,
        0xB0,
        0xA1,
        0x92,
        0x82,
        0x72,
        0x63,
        0x53,
        0x43,
        0x33,
        0x20,
        0x13,
        0x03
    };
    typedef enum
    {
        ADS1256_30000SPS = 0,
        ADS1256_15000SPS,
        ADS1256_7500SPS,
        ADS1256_3750SPS,
        ADS1256_2000SPS,
        ADS1256_1000SPS,
        ADS1256_500SPS,
        ADS1256_100SPS,
        ADS1256_60SPS,
        ADS1256_50SPS,
        ADS1256_30SPS,
        ADS1256_25SPS,
        ADS1256_15SPS,
        ADS1256_10SPS,
        ADS1256_5SPS,
        ADS1256_2d5SPS,
        
        ADS1256_DRATE_MAX
    }ADS1256_DRATE_E;

    93.5.5 第4步,ADS1256启动采样

    代码实现如下

    /*
    *********************************************************************************************************
    *    函 数 名: ADS1256_StartScan
    *    功能说明: 将 DRDY引脚 (PC6 )配置成外部中断触发方式, 中断服务程序中扫描8个通道的数据。
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void ADS1256_StartScan(void)
    {
        /* PC6 外部中断,BUSY 
            配置 BUSY 作为中断输入口,下降沿触发 */
        {
            GPIO_InitTypeDef   GPIO_InitStructure;
            
            DRDY_CLK_ENABLE();    /* 打开GPIO时钟 */
    
            GPIO_InitStructure.Mode = GPIO_MODE_IT_FALLING;
            GPIO_InitStructure.Pull = GPIO_NOPULL;
            GPIO_InitStructure.Pin = DRDY_PIN;
            HAL_GPIO_Init(DRDY_GPIO, &GPIO_InitStructure);    
    
            HAL_NVIC_SetPriority(DRDY_IRQn, 2, 0);
            HAL_NVIC_EnableIRQ(DRDY_IRQn);    
        }
        
        /* 开始扫描前, 清零结果缓冲区 */    
        {
            uint8_t i;
            
            g_tADS1256.Channel = 0;
            
            for (i = 0; i < 8; i++)
            {
                g_tADS1256.AdcNow[i] = 0;
            }    
        }
    }

    代码比较简单,主要是配置PC6的EXTI外部中断,并初始化变量。

    93.5.6 第5步,ADS1256的中断处理(8通道数据读取)

    代码如下:

    /*
    *********************************************************************************************************
    *    函 数 名: EXTI9_5_IRQHandler
    *    功能说明: 外部中断服务程序.  此程序执行时间约 123uS
    *    形    参:无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    #ifdef EXTI9_5_ISR_MOVE_OUT        /* bsp.h 中定义此行,表示本函数移到 stam32f4xx_it.c。 避免重复定义 */
    void EXTI9_5_IRQHandler(void)
    {
        HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_6);
    }
    
    /*
    *********************************************************************************************************
    *    函 数 名: EXTI9_5_IRQHandler
    *    功能说明: 外部中断服务程序入口。PI6 / AD7606_BUSY 下降沿中断触发
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
    {
        if (GPIO_Pin == GPIO_PIN_6)
        {
    
            ADS1256_ISR();
        }
    }
    #endif
    
    /*
    *********************************************************************************************************
    *    函 数 名: ADS1256_ISR
    *    功能说明: 定时采集中断服务程序
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void ADS1256_ISR(void)
    {
        /* 读取采集结构,保存在全局变量 */                    
        ADS1256_SetChannal(g_tADS1256.Channel);    /* 切换模拟通道 */    
        bsp_DelayUS(5);
        
        ADS1256_WriteCmd(CMD_SYNC);
        bsp_DelayUS(5);
        
        ADS1256_WriteCmd(CMD_WAKEUP);
        bsp_DelayUS(25);
        
        if (g_tADS1256.Channel == 0)
        {
            g_tADS1256.AdcNow[7] = ADS1256_ReadData();    /* 注意保存的是上一个通道的数据 */
        }
        else
        {
            g_tADS1256.AdcNow[g_tADS1256.Channel-1] = ADS1256_ReadData(); /* 注意保存的是上一个通道的数据 */
        }
                    
        if (++g_tADS1256.Channel >= 8)
        {
            g_tADS1256.Channel = 0;
        }
    }

    中断服务程序的代码完全是按照下面的时序时序:

     

     

    •   第1步:ADS1256_SetChannal(g_tADS1256.Channel)切换模拟通道。
    •   第2步:发送SYNC同步命令。
    •   第3步:唤醒。
    •   第4步:读取数据。

    这个过程中特别注意读取的是上次转换的数据。

    93.6 ADS1256板级支持包(bsp_spi_ads1256.c)

    ADS1256驱动文件bsp_spi_ads1256.c主要实现了如下几个API供用户调用:

    •   bsp_InitADS1256
    •   ADS1256_CfgADC
    •   ADS1256_StartScan
    •   ADS1256_SetChannal
    •   ADS1256_SetDiffChannal

    93.6.1 函数bsp_InitADS1256

    函数原型:

    void bsp_InitADS1256(void)

    函数描述:

    主要用于ADS1256的初始化。

    93.6.2 函数ADS1256_CfgADC

    函数原型:

    void ADS1256_CfgADC(ADS1256_GAIN_E _gain, ADS1256_DRATE_E _drate)

    函数描述:

    用于配置ADS1256的增益和采样率。

    函数参数:

    •   第1个参数用于设置增益,支持的参数如下:

    ADS1256_GAIN_1

    ADS1256_GAIN_2     

    ADS1256_GAIN_4

    ADS1256_GAIN_8

    ADS1256_GAIN_16

    ADS1256_GAIN_32

    ADS1256_GAIN_64

    •   第2个参数用于设置采样率,支持的参数如下:

    ADS1256_30000SPS

    ADS1256_15000SPS

    ADS1256_7500SPS

    ADS1256_3750SPS

    ADS1256_2000SPS

    ADS1256_1000SPS

    ADS1256_500SPS

    ADS1256_100SPS

    ADS1256_60SPS

    ADS1256_50SPS

    ADS1256_30SPS

    ADS1256_25SPS

    ADS1256_15SPS

    ADS1256_10SPS

    ADS1256_5SPS

    ADS1256_2d5SPS

    93.6.3 函数ADS1256_StartScan

    函数原型:

    void ADS1256_StartScan(void)

    函数描述:

    此函数用于启动扫描,采样的中断方式。

    93.6.4 函数ADS1256_SetChannal

    函数原型:

    static void ADS1256_SetChannal(uint8_t _ch)

    函数描述:

    此函数用于设置单端采样的通道。

    函数参数:

    •   第1个参数支持0到7,0表示采样的通道0, 1表示采样的通道1,依次类推,范围0-7,共8个通道。

    93.7 ADS1256实际测量效果(10uV抖动)

    测试LM285-2.5V稳压效果,抖动40uV:

     

     

    测试干电池效果,抖动10uV左右,注意,这个级别的抖动容易受环境温度的影响,特别是开关空调,最明显。

     

     

    93.8 ADS1256驱动移植和使用

    移植步骤如下:

    •   第1步:复制bsp_spi_ads1256.c和bsp_spi_ads1256.h到自己的工程目录,并添加到工程里面。
    •   第2步:根据使用的SPI引脚,DRDY就绪引脚,RST复位引脚,修改bsp_spi_ads1256.c开头的宏定义。
    /* 定义GPIO端口 */    
    #define SCK_CLK_ENABLE()     __HAL_RCC_GPIOA_CLK_ENABLE()
    #define SCK_GPIO            GPIOA
    #define SCK_PIN                GPIO_PIN_4
    #define SCK_1()                SCK_GPIO->BSRR = SCK_PIN
    #define SCK_0()                SCK_GPIO->BSRR = ((uint32_t)SCK_PIN << 16U)    
    
    #define DIN_CLK_ENABLE()     __HAL_RCC_GPIOG_CLK_ENABLE()
    #define DIN_GPIO            GPIOG
    #define DIN_PIN                GPIO_PIN_10
    #define DIN_1()                DIN_GPIO->BSRR = DIN_PIN
    #define DIN_0()                DIN_GPIO->BSRR = ((uint32_t)DIN_PIN << 16U)    
    
    #define CS_CLK_ENABLE()     __HAL_RCC_GPIOC_CLK_ENABLE()
    #define CS_GPIO                GPIOC
    #define CS_PIN                GPIO_PIN_7
    #define CS_1()                CS_GPIO->BSRR = CS_PIN
    #define CS_0()                CS_GPIO->BSRR = ((uint32_t)CS_PIN << 16U)    
    
    #define DOUT_CLK_ENABLE()     __HAL_RCC_GPIOA_CLK_ENABLE()
    #define DOUT_GPIO            GPIOA
    #define DOUT_PIN            GPIO_PIN_5
    #define DOUT_IS_HIGH()        ((DOUT_GPIO->IDR & DOUT_PIN) != 0)
    
    #define DRDY_CLK_ENABLE()     __HAL_RCC_GPIOC_CLK_ENABLE()
    #define DRDY_GPIO            GPIOC
    #define DRDY_PIN            GPIO_PIN_6
    #define DRDY_IS_LOW()        ((DRDY_GPIO->IDR & DRDY_PIN) == 0)
    #define DRDY_IRQn             EXTI9_5_IRQn
    #define DRDY_IRQHandler        EXTI9_5_IRQHandler    
    
    /* PDWN  <------  PB7       掉电控制 */
    #define PWDN_CLK_ENABLE()     __HAL_RCC_GPIOB_CLK_ENABLE()
    #define PWDN_GPIO            GPIOB
    #define PWDN_PIN            GPIO_PIN_7
    #define PWDN_1()            PWDN_GPIO->BSRR = PWDN_PIN
    #define PWDN_0()            PWDN_GPIO->BSRR = ((uint32_t)PWDN_PIN << 16U)            
    
    /*  RST   <------  PC3       复位信号     */
    #define RST_CLK_ENABLE()     __HAL_RCC_GPIOC_CLK_ENABLE()
    #define RST_GPIO            GPIOC
    #define RST_PIN                GPIO_PIN_3
    #define RST_1()                RST_GPIO->BSRR = RST_PIN
    #define RST_0()                RST_GPIO->BSRR = ((uint32_t)RST_PIN << 16U)    
    •   第3步:特别注意中断服务程序的入口要根据使用的DRDY引脚修改。
    /*
    *********************************************************************************************************
    *    函 数 名: EXTI9_5_IRQHandler
    *    功能说明: 外部中断服务程序.  此程序执行时间约 123uS
    *    形    参:无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    #ifdef EXTI9_5_ISR_MOVE_OUT        /* bsp.h 中定义此行,表示本函数移到 stam32f4xx_it.c。 避免重复定义 */
    void EXTI9_5_IRQHandler(void)
    {
        HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_6);
    }
    
    /*
    *********************************************************************************************************
    *    函 数 名: EXTI9_5_IRQHandler
    *    功能说明: 外部中断服务程序入口。PI6 / AD7606_BUSY 下降沿中断触发
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
    {
        if (GPIO_Pin == GPIO_PIN_6)
        {
    
            ADS1256_ISR();
        }
    }
    #endif
    •   第4步:应用方法看本章节配套例子即可。

    93.9 实验例程设计框架

    通过程序设计框架,让大家先对配套例程有一个全面的认识,然后再理解细节,本次实验例程的设计框架如下:

     

     

      第1阶段,上电启动阶段:

    • 这部分在第14章进行了详细说明。

      第2阶段,进入main函数:

    •   第1部分,硬件初始化,主要是MPU,Cache,HAL库,系统时钟,滴答定时器和LED。
    •   第2部分,应用程序设计部分,测试ADS1256。

    93.10          实验例程说明(MDK)

    配套例子:

    V7-068_ADS1256(8通道带PGA的24位ADC)

    实验目的:

    1. 学习ADS1256, 8通道带PGA的24bit ADC。

    重要提示:

    1. 开发板请使用外置电源供电。

    实验内容:

    1. 启动一个自动重装软件定时器,每100ms翻转一次LED2。
    2. 上电后插入ADS1256模块到右上角CN26(2*6P双排母), 程序每1秒打印一次8通道采样数据。

    上电后串口打印的信息:

    波特率 115200,数据位 8,奇偶校验位无,停止位 1。

     

     

    模块插入位置:

     

     

    程序设计:

      系统栈大小分配:

     

     

      RAM空间用的DTCM:

     

     

      硬件外设初始化

    硬件外设的初始化是在 bsp.c 文件实现:

    /*
    *********************************************************************************************************
    *    函 数 名: bsp_Init
    *    功能说明: 初始化所有的硬件设备。该函数配置CPU寄存器和外设的寄存器并初始化一些全局变量。只需要调用一次
    *    形    参:无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void bsp_Init(void)
    {
        /* 配置MPU */
        MPU_Config();
        
        /* 使能L1 Cache */
        CPU_CACHE_Enable();
    
        /* 
           STM32H7xx HAL 库初始化,此时系统用的还是H7自带的64MHz,HSI时钟:
           - 调用函数HAL_InitTick,初始化滴答时钟中断1ms。
           - 设置NVIV优先级分组为4。
         */
        HAL_Init();
    
        /* 
           配置系统时钟到400MHz
           - 切换使用HSE。
           - 此函数会更新全局变量SystemCoreClock,并重新配置HAL_InitTick。
        */
        SystemClock_Config();
    
        /* 
           Event Recorder:
           - 可用于代码执行时间测量,MDK5.25及其以上版本才支持,IAR不支持。
           - 默认不开启,如果要使能此选项,务必看V7开发板用户手册第xx章
        */    
    #if Enable_EventRecorder == 1  
        /* 初始化EventRecorder并开启 */
        EventRecorderInitialize(EventRecordAll, 1U);
        EventRecorderStart();
    #endif
        
    bsp_InitDWT();      /* 初始化DWT时钟周期计数器 */       
        bsp_InitKey();         /* 按键初始化,要放在滴答定时器之前,因为按钮检测是通过滴答定时器扫描 */
        bsp_InitTimer();       /* 初始化滴答定时器 */
        bsp_InitLPUart();     /* 初始化串口 */
        bsp_InitExtIO();     /* 初始化FMC总线74HC574扩展IO. 必须在 bsp_InitLed()前执行 */    
        bsp_InitLed();         /* 初始化LED */    
    bsp_InitExtSDRAM(); /* 初始化SDRAM */
    }

      MPU配置和Cache配置:

    数据Cache和指令Cache都开启。配置了AXI SRAM区(本例子未用到AXI SRAM)和FMC的扩展IO区。

    /*
    *********************************************************************************************************
    *    函 数 名: MPU_Config
    *    功能说明: 配置MPU
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    static void MPU_Config( void )
    {
        MPU_Region_InitTypeDef MPU_InitStruct;
    
        /* 禁止 MPU */
        HAL_MPU_Disable();
    
        /* 配置AXI SRAM的MPU属性为Write back, Read allocate,Write allocate */
        MPU_InitStruct.Enable           = MPU_REGION_ENABLE;
        MPU_InitStruct.BaseAddress      = 0x24000000;
        MPU_InitStruct.Size             = MPU_REGION_SIZE_512KB;
        MPU_InitStruct.AccessPermission = MPU_REGION_FULL_ACCESS;
        MPU_InitStruct.IsBufferable     = MPU_ACCESS_BUFFERABLE;
        MPU_InitStruct.IsCacheable      = MPU_ACCESS_CACHEABLE;
        MPU_InitStruct.IsShareable      = MPU_ACCESS_NOT_SHAREABLE;
        MPU_InitStruct.Number           = MPU_REGION_NUMBER0;
        MPU_InitStruct.TypeExtField     = MPU_TEX_LEVEL1;
        MPU_InitStruct.SubRegionDisable = 0x00;
        MPU_InitStruct.DisableExec      = MPU_INSTRUCTION_ACCESS_ENABLE;
    
        HAL_MPU_ConfigRegion(&MPU_InitStruct);
        
        
        /* 配置FMC扩展IO的MPU属性为Device或者Strongly Ordered */
        MPU_InitStruct.Enable           = MPU_REGION_ENABLE;
        MPU_InitStruct.BaseAddress      = 0x60000000;
        MPU_InitStruct.Size             = ARM_MPU_REGION_SIZE_64KB;    
        MPU_InitStruct.AccessPermission = MPU_REGION_FULL_ACCESS;
        MPU_InitStruct.IsBufferable     = MPU_ACCESS_BUFFERABLE;
        MPU_InitStruct.IsCacheable      = MPU_ACCESS_NOT_CACHEABLE;    
        MPU_InitStruct.IsShareable      = MPU_ACCESS_NOT_SHAREABLE;
        MPU_InitStruct.Number           = MPU_REGION_NUMBER1;
        MPU_InitStruct.TypeExtField     = MPU_TEX_LEVEL0;
        MPU_InitStruct.SubRegionDisable = 0x00;
        MPU_InitStruct.DisableExec      = MPU_INSTRUCTION_ACCESS_ENABLE;
        
        HAL_MPU_ConfigRegion(&MPU_InitStruct);
    
        /*使能 MPU */
        HAL_MPU_Enable(MPU_PRIVILEGED_DEFAULT);
    }
    
    /*
    *********************************************************************************************************
    *    函 数 名: CPU_CACHE_Enable
    *    功能说明: 使能L1 Cache
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    static void CPU_CACHE_Enable(void)
    {
        /* 使能 I-Cache */
        SCB_EnableICache();
    
        /* 使能 D-Cache */
        SCB_EnableDCache();
    }

      每10ms调用一次按键处理:

    按键处理是在滴答定时器中断里面实现,每10ms执行一次检测。

    /*
    *********************************************************************************************************
    *    函 数 名: bsp_RunPer10ms
    *    功能说明: 该函数每隔10ms被Systick中断调用1次。详见 bsp_timer.c的定时中断服务程序。一些处理时间要求
    *              不严格的任务可以放在此函数。比如:按键扫描、蜂鸣器鸣叫控制等。
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void bsp_RunPer10ms(void)
    {
        bsp_KeyScan10ms();
    }

      主功能:

    主程序实现如下操作:

    •   启动一个自动重装软件定时器,每100ms翻转一次LED2。
    •   上电后插入ADS1256模块到右上角CN26(2*6P双排母), 程序每1秒打印一次8通道采样数据。
    /*
    *********************************************************************************************************
    *    函 数 名: main
    *    功能说明: c程序入口
    *    形    参: 无
    *    返 回 值: 错误代码(无需处理)
    *********************************************************************************************************
    */
    int main(void)
    {
        uint8_t i;
        int32_t iTemp;
        float fTemp;
    
        
        bsp_Init();        /* 硬件初始化 */
        PrintfLogo();    /* 打印例程信息到串口1 */
    
        PrintfHelp();    /* 打印操作提示信息 */
        
        
        bsp_DelayMS(500);    /* 等待上电稳定,等基准电压电路稳定, bsp_InitADS1256() 内部会进行自校准 */
    
        bsp_InitADS1256();    /* 初始化配置ADS1256.  PGA=1, DRATE=30KSPS, BUFEN=1, 输入正负5V */
        
        
        /* 打印芯片ID (通过读ID可以判断硬件接口是否正常) , 正常时状态寄存器的高4bit = 3 */
    #if 0
        {
            uint8_t id;
    
            id = ADS1256_ReadChipID();
    
            if (id != 3)
            {
                printf("Error, ASD1256 Chip ID = 0x%X\r\n", id);
            }
            else
            {
                printf("Ok, ASD1256 Chip ID = 0x%X\r\n", id);
            }
        }
    #endif
        
        ADS1256_CfgADC(ADS1256_GAIN_1, ADS1256_30SPS);    /* 配置ADC参数: 增益1:1, 数据输出速率 30Hz */
    
    /* 启动中断扫描模式, 轮流采集8个通道的ADC数据. 通过 ADS1256_GetAdc() 函数来读取这些数据 */
        ADS1256_StartScan();    
        
        bsp_StartAutoTimer(0, 1000);    /* 启动1个100ms的自动重装的定时器 */
    
        /* 进入主程序循环体 */
        while (1)
        {
            bsp_Idle();        /* 这个函数在bsp.c文件。用户可以修改这个函数实现CPU休眠和喂狗 */
            
    
            if (bsp_CheckTimer(0))    /* 判断定时器超时时间 */
            {
                /* 每隔1000ms 进来一次 */
                bsp_LedToggle(2);    /* 翻转LED的状态 */
                
                /* 打印采集数据 */
                for (i = 0; i < 8; i++)
                {
    
                      /*
                        计算公式:2 * VREF/(PGA * (2^23 - 1)) ,这里VREF是2.5V,PGA = 1
                      */
                       /* 计算实际电压值(近似估算的),如需准确,请进行校准 */
                    iTemp = ((int64_t)g_tADS1256.AdcNow[i] * 2500000) / 4194303; 
    
                    
                    fTemp = (float)iTemp / 1000000;   
    
                    printf("CH%d=%07d(%fV) ", i, g_tADS1256.AdcNow[i], fTemp);
    
                    if(i == 3)
                    {
                        printf("\r\n");
                    }
                }
                
                printf("\r\n\r\n");
            }
        }
    }

    93.11          实验例程说明(IAR)

    配套例子:

    V7-068_ADS1256(8通道带PGA的24位ADC)

    实验目的:

    1. 学习ADS1256, 8通道带PGA的24bit ADC。

    重要提示:

    1. 开发板请使用外置电源供电。

    实验内容:

    1. 启动一个自动重装软件定时器,每100ms翻转一次LED2。
    2. 上电后插入ADS1256模块到右上角CN26(2*6P双排母), 程序每1秒打印一次8通道采样数据。

    上电后串口打印的信息:

    波特率 115200,数据位 8,奇偶校验位无,停止位 1。

     

     

    模块插入位置:

     

     

    程序设计:

      系统栈大小分配:

     

     

      RAM空间用的DTCM:

     

     

      硬件外设初始化

    硬件外设的初始化是在 bsp.c 文件实现:

    /*
    *********************************************************************************************************
    *    函 数 名: bsp_Init
    *    功能说明: 初始化所有的硬件设备。该函数配置CPU寄存器和外设的寄存器并初始化一些全局变量。只需要调用一次
    *    形    参:无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void bsp_Init(void)
    {
        /* 配置MPU */
        MPU_Config();
        
        /* 使能L1 Cache */
        CPU_CACHE_Enable();
    
        /* 
           STM32H7xx HAL 库初始化,此时系统用的还是H7自带的64MHz,HSI时钟:
           - 调用函数HAL_InitTick,初始化滴答时钟中断1ms。
           - 设置NVIV优先级分组为4。
         */
        HAL_Init();
    
        /* 
           配置系统时钟到400MHz
           - 切换使用HSE。
           - 此函数会更新全局变量SystemCoreClock,并重新配置HAL_InitTick。
        */
        SystemClock_Config();
    
        /* 
           Event Recorder:
           - 可用于代码执行时间测量,MDK5.25及其以上版本才支持,IAR不支持。
           - 默认不开启,如果要使能此选项,务必看V7开发板用户手册第xx章
        */    
    #if Enable_EventRecorder == 1  
        /* 初始化EventRecorder并开启 */
        EventRecorderInitialize(EventRecordAll, 1U);
        EventRecorderStart();
    #endif
        
    bsp_InitDWT();      /* 初始化DWT时钟周期计数器 */       
        bsp_InitKey();         /* 按键初始化,要放在滴答定时器之前,因为按钮检测是通过滴答定时器扫描 */
        bsp_InitTimer();       /* 初始化滴答定时器 */
        bsp_InitLPUart();     /* 初始化串口 */
        bsp_InitExtIO();     /* 初始化FMC总线74HC574扩展IO. 必须在 bsp_InitLed()前执行 */    
        bsp_InitLed();         /* 初始化LED */    
    bsp_InitExtSDRAM(); /* 初始化SDRAM */
    
        /* 针对不同的应用程序,添加需要的底层驱动模块初始化函数 */    
        bsp_InitAD7606();    /* 配置AD7606所用的GPIO */
    }

      MPU配置和Cache配置:

    数据Cache和指令Cache都开启。配置了AXI SRAM区(本例子未用到AXI SRAM)和FMC的扩展IO区。

    /*
    *********************************************************************************************************
    *    函 数 名: MPU_Config
    *    功能说明: 配置MPU
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    static void MPU_Config( void )
    {
        MPU_Region_InitTypeDef MPU_InitStruct;
    
        /* 禁止 MPU */
        HAL_MPU_Disable();
    
        /* 配置AXI SRAM的MPU属性为Write back, Read allocate,Write allocate */
        MPU_InitStruct.Enable           = MPU_REGION_ENABLE;
        MPU_InitStruct.BaseAddress      = 0x24000000;
        MPU_InitStruct.Size             = MPU_REGION_SIZE_512KB;
        MPU_InitStruct.AccessPermission = MPU_REGION_FULL_ACCESS;
        MPU_InitStruct.IsBufferable     = MPU_ACCESS_BUFFERABLE;
        MPU_InitStruct.IsCacheable      = MPU_ACCESS_CACHEABLE;
        MPU_InitStruct.IsShareable      = MPU_ACCESS_NOT_SHAREABLE;
        MPU_InitStruct.Number           = MPU_REGION_NUMBER0;
        MPU_InitStruct.TypeExtField     = MPU_TEX_LEVEL1;
        MPU_InitStruct.SubRegionDisable = 0x00;
        MPU_InitStruct.DisableExec      = MPU_INSTRUCTION_ACCESS_ENABLE;
    
        HAL_MPU_ConfigRegion(&MPU_InitStruct);
        
        
        /* 配置FMC扩展IO的MPU属性为Device或者Strongly Ordered */
        MPU_InitStruct.Enable           = MPU_REGION_ENABLE;
        MPU_InitStruct.BaseAddress      = 0x60000000;
        MPU_InitStruct.Size             = ARM_MPU_REGION_SIZE_64KB;    
        MPU_InitStruct.AccessPermission = MPU_REGION_FULL_ACCESS;
        MPU_InitStruct.IsBufferable     = MPU_ACCESS_BUFFERABLE;
        MPU_InitStruct.IsCacheable      = MPU_ACCESS_NOT_CACHEABLE;    
        MPU_InitStruct.IsShareable      = MPU_ACCESS_NOT_SHAREABLE;
        MPU_InitStruct.Number           = MPU_REGION_NUMBER1;
        MPU_InitStruct.TypeExtField     = MPU_TEX_LEVEL0;
        MPU_InitStruct.SubRegionDisable = 0x00;
        MPU_InitStruct.DisableExec      = MPU_INSTRUCTION_ACCESS_ENABLE;
        
        HAL_MPU_ConfigRegion(&MPU_InitStruct);
    
        /*使能 MPU */
        HAL_MPU_Enable(MPU_PRIVILEGED_DEFAULT);
    }
    
    /*
    *********************************************************************************************************
    *    函 数 名: CPU_CACHE_Enable
    *    功能说明: 使能L1 Cache
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    static void CPU_CACHE_Enable(void)
    {
        /* 使能 I-Cache */
        SCB_EnableICache();
    
        /* 使能 D-Cache */
        SCB_EnableDCache();
    }

      每10ms调用一次按键处理:

    按键处理是在滴答定时器中断里面实现,每10ms执行一次检测。

    /*
    *********************************************************************************************************
    *    函 数 名: bsp_RunPer10ms
    *    功能说明: 该函数每隔10ms被Systick中断调用1次。详见 bsp_timer.c的定时中断服务程序。一些处理时间要求
    *              不严格的任务可以放在此函数。比如:按键扫描、蜂鸣器鸣叫控制等。
    *    形    参: 无
    *    返 回 值: 无
    *********************************************************************************************************
    */
    void bsp_RunPer10ms(void)
    {
        bsp_KeyScan10ms();
    }

      主功能:

    主程序实现如下操作:

    •   启动一个自动重装软件定时器,每100ms翻转一次LED2。
    •   上电后插入ADS1256模块到右上角CN26(2*6P双排母), 程序每1秒打印一次8通道采样数据。
    /*
    *********************************************************************************************************
    *    函 数 名: main
    *    功能说明: c程序入口
    *    形    参: 无
    *    返 回 值: 错误代码(无需处理)
    *********************************************************************************************************
    */
    int main(void)
    {
        uint8_t i;
        int32_t iTemp;
        float fTemp;
    
        
        bsp_Init();        /* 硬件初始化 */
        PrintfLogo();    /* 打印例程信息到串口1 */
    
        PrintfHelp();    /* 打印操作提示信息 */
        
        
        bsp_DelayMS(500);    /* 等待上电稳定,等基准电压电路稳定, bsp_InitADS1256() 内部会进行自校准 */
    
        bsp_InitADS1256();    /* 初始化配置ADS1256.  PGA=1, DRATE=30KSPS, BUFEN=1, 输入正负5V */
        
        
        /* 打印芯片ID (通过读ID可以判断硬件接口是否正常) , 正常时状态寄存器的高4bit = 3 */
    #if 0
        {
            uint8_t id;
    
            id = ADS1256_ReadChipID();
    
            if (id != 3)
            {
                printf("Error, ASD1256 Chip ID = 0x%X\r\n", id);
            }
            else
            {
                printf("Ok, ASD1256 Chip ID = 0x%X\r\n", id);
            }
        }
    #endif
        
        ADS1256_CfgADC(ADS1256_GAIN_1, ADS1256_30SPS);    /* 配置ADC参数: 增益1:1, 数据输出速率 30Hz */
    
    /* 启动中断扫描模式, 轮流采集8个通道的ADC数据. 通过 ADS1256_GetAdc() 函数来读取这些数据 */
        ADS1256_StartScan();    
        
        bsp_StartAutoTimer(0, 1000);    /* 启动1个100ms的自动重装的定时器 */
    
        /* 进入主程序循环体 */
        while (1)
        {
            bsp_Idle();        /* 这个函数在bsp.c文件。用户可以修改这个函数实现CPU休眠和喂狗 */
            
    
            if (bsp_CheckTimer(0))    /* 判断定时器超时时间 */
            {
                /* 每隔1000ms 进来一次 */
                bsp_LedToggle(2);    /* 翻转LED的状态 */
                
                /* 打印采集数据 */
                for (i = 0; i < 8; i++)
                {
    
                      /*
                        计算公式:2 * VREF/(PGA * (2^23 - 1)) ,这里VREF是2.5V,PGA = 1
                      */
                       /* 计算实际电压值(近似估算的),如需准确,请进行校准 */
                    iTemp = ((int64_t)g_tADS1256.AdcNow[i] * 2500000) / 4194303; 
    
                    
                    fTemp = (float)iTemp / 1000000;   
    
                    printf("CH%d=%07d(%fV) ", i, g_tADS1256.AdcNow[i], fTemp);
    
                    if(i == 3)
                    {
                        printf("\r\n");
                    }
                }
                
                printf("\r\n\r\n");
            }
        }
    }

    93.12   总结

    本章节涉及到的知识点非常多,主要为大家讲解了ADS1256的常用玩法,如果实际项目中用到此芯片需要熟练运用。

    微信公众号:armfly_com 安富莱论坛:www.armbbs.cn 安富莱淘宝:https://armfly.taobao.com
  • 相关阅读:
    SMO学习笔记(五)——附加数据库
    SMO学习笔记(二)——还原(恢复)篇之完整恢复
    加深C# 中字符串前加@符号理解以及使用~~
    Oracle Sys用户用默认密码change_on_install 无法登录的问题(错误代码:ORA28009)
    Reflector for .NET 下载问题
    SQLSERVER拆分字符串的函数(表值函数)
    AjaxPro使用Session出错(AjaxPro "Session"引发了"System.NullReferenceException"类型的异常)
    ASP.NET Web页面(.aspx)添加用户控件(.ascx)无显示的问题
    公积金贷款与商业贷款的区别(在废打印纸中的意外收获... :))
    SMO学习笔记(三)——效验数据库备份文件
  • 原文地址:https://www.cnblogs.com/armfly/p/15634067.html
Copyright © 2011-2022 走看看