• 外设驱动库开发笔记48:MCP4725单通道DAC驱动


      在产品设计过程中,我们经常会遇到数模转换的应用需求。在本篇种我们就来讨论一下MCP4725单通道数模转换器的驱动设计与实现。

    1、功能概述

      MCP4725是一个低功耗,高精度,单通道,12位缓冲电压输出数字到模拟转换器(DAC)与非易失性存储器(EEPROM)。它的板载精度输出放大器允许它实现轨到轨模拟输出摆动。
      DAC输入和配置数据可以被编程到非易失性存储器(EEPROM)由用户使用I2C接口命令。非易失性存储器特性使DAC设备能够在断电时保存DAC输入代码,并且在通电后立即提供DAC输出。当DAC设备被用作网络中其他设备的支持设备时,这个特性非常有用。MCP4725的引脚定义及排布如下:

      MCP4725有一个外部A0地址位选择引脚。这个A0引脚可以绑定到用户应用PCB板的VDD或VSS上。这个引脚被用户用来选择A0地址位。用户可以将这个引脚绑定到VSS(逻辑’ 0 ‘),或VDD(逻辑’ 1 '),或可以由数字逻辑级别主动驱动,如I2C主输出。
      MCP4725的地址字节由两个部分组成,第一部分为4位设备代码,固定设置为1100的,设备代码后面是三位为地址位(A2, A1, A0),如下图所示:

      A2和A1位的选择可由客户提供,作为订购过程的一部分。两位在出厂前设定好,如果客户没有特别要求的话,A2和A1会默认编程为“00”。而A0位则由A0引脚的逻辑状态决定。
      MCP4725设备包括一个上电复位(POR)电路,以确保可靠的上电,以及一个用于EEPROM编程电压的板载电荷泵。DAC引用是直接从VDD驱动的。在down模式下,输出放大器可以配置为已知的低、中或高阻输出负载,如下图。

      MCP472的写命令用于将配置位和DAC输入码加载到DAC寄存器,或写入设备的EEPROM。写命令类型由三个写命令类型位(C2、C1、C0)定义。写命令类型及其作用如下表所示。

    2、驱动设计与实现

      在前一节中,我们梳理了MCP4725单通道数模转换器的基本技术参数。在这一节中,我们将依据这些技术参数来设计MCP4725单通道数模转换器的驱动程序。我们依然是基于对象的思想来实现之。

    2.1、对象定义

      我们基于对象来实现驱动程序,所以我们就需要先得到对象,在这里我们首先将抽象出MCP4725单通道数模转换器的对象类型。一版来说,对象皆包含属性与操作两个方面的内容。在抽象对象类型的过程中,我们需要分析MCP4725单通道数模转换器都有哪些属性和操作。
      我们先来分析MCP4725单通道数模转换器的对象的属性。每一台I2C从设备都有一个设备地址,这个地址实际上标识了总线上设备的身份,MCP4725亦如此,所以我们将设备地址作为对象的一个属性。对于MCP4725单通道数模转换器,有一个掉电处理模式是需要配置的,为了掌握其配置状态我们将其作为对象的一个属性记录下来。
      接下来分析MCP4725单通道数模转换器的对象的操作。MCP4725单通道数模转换器的基本操作无非就是读写数据,而要实现读和写则依赖于具体的软硬件平台,所以我们将读和写MCP4725单通道数模转换器都作为对象的操作来实现。
      根据上述关于MCP4725单通道数模转换器对象属性和操作的分析,我们可以抽象得到其对象类型如下:

    /*定义MCP4725对象类型*/
    typedef struct Mcp4725Object {
        uint8_t devAddress;
        Mcp4725PDModeType pdMode;
        void (*Write)(struct Mcp4725Object *mcp,uint8_t *wData, uint16_t wSize);
        void (*Read)(struct Mcp4725Object *mcp,uint8_t *rData, uint16_t rSize);
    }Mcp4725ObjectType;
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

      抽象了对象类型后就可声明对象变量,可是这个对象变量必须作必要的初始化才能使用。所以我们需要一个初始化函数来对其进行初始化。在此函数中,我们将检测变量的有效性和初始状态赋值,并对设备进行必要的配置。根据这些要求我们设计MCP4725单通道数模转换器的对象初始化函数如下:

    /*MCP4725初始化配置*/
    void Mcp4725Initialization(Mcp4725ObjectType *mcp,  //MCP4725对象变量
                           uint8_t slaveAddress,    //从站设备的地址
                           Mcp4725PDModeType pdMode,//掉电操作模式
                           Mcp4725Write write,      //写数据函数指针
                            Mcp4725Read read         //读数据函数指针
                            )
    {
        if((mcp==NULL)||(write==NULL)||(read==NULL))
        {
            return;
        }
        
        mcp->Write=write;
        mcp->Read=read;
        
        if((slaveAddress==0x60)||(slaveAddress==0x61))
        {
            mcp->devAddress=(slaveAddress<<1);
        }
        else if((slaveAddress==0xC0)||(slaveAddress==0xC2))
        {
            mcp->devAddress=slaveAddress;
        }
        else
        {
            mcp->devAddress=0x00;
        }
        
        mcp->pdMode=pdMode;
        
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32

    2.2、对象操作

      有了对象变量,也完成了初始化,那么我们就可以用其来操作MCP4725单通道数模转换器了。所以我们来看看实现对MCP4725单通道数模转换器的读写操作。
    首先我们来看看写MCP4725单通道数模转换器的实现。写MCP4725单通道数模转换器有两种模式:快速模式和正常模式。快速模式就是将命令与数据结合在一起,这要只需发送三个字节就可完成写数据的过程。具体的操作时序如下:

      而正常模式则是命令是单独的字节,数据是另外的2个字节,所以正常模式一次发送4个字节才能完成写的过程。正常模式可以操作寄存器也可操作EEPROM,这一点与快速模式是不一样的。具体的操作时序如下:

      根据前面的描述和时序图,我们可以设计写MCP4725单通道数模转换器的函数。下面的函数可以快速模式和普通模式,有命令类型来决定最终的操作方式。

    /*设置MCP4725输出*/
    void Mcp4725SetDatas(Mcp4725ObjectType *mcp,Mcp4725CommandType cmd,uint16_t data)
    {
        uint8_t wData[3];
        uint8_t pdMode=0;
        uint16_t wSize=0;
        uint8_t command[]={Fast_Mode,Write_DAC_Register,Write_DAC_Register_EEPROM};
        
        pdMode=(uint8_t)(mcp->pdMode);
        
        if(cmd==Mcp4725_Fast_Mode)  //快速模式
        {
            wData[1]=(uint8_t)data;
            wData[0]=(uint8_t)(data>>8);
            wData[0]=wData[0]|command[cmd];
            wData[0]=wData[0]|(pdMode<<4);
            wSize=2;
        }
        else    //普通模式
        {
            wData[0]=command[cmd];
            wData[0]=wData[0]|(pdMode<<1);
            wData[1]=(uint8_t)(data>>4);
            wData[2]=(uint8_t)(data<<4);
            wSize=3;
        }
        
        mcp->Write(mcp,wData,wSize);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29

      MCP4725单通道数模转换器不断可以写数据也可以读数据。读回来的数据包括状态命令字、DAC寄存器数据以及EEPROM数据,总共是5个字节。具体的操作时序如下:

      根据前速的分析以及时序图,我们可以简单实现读操作如下:

    /*读取MCP4725数据*/
    void Mcp4725GetDatas(Mcp4725ObjectType *mcp,uint8_t *rData)
    {
        mcp->Read(mcp,rData,5);
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5

    3、驱动的使用

      我们已经实现了MCP4725单通道数模转换器的驱动程序。我们还需要将这一驱动程序实际应用一下以确认驱动程序的正确性。

    3.1、声明并初始化对象

      同样,我们先声明一个MCP4725单通道数模转换器对象变量。前面我们已经抽象了对象类型,使用MCP4725单通道数模转换器对象类型声明如下:

    Mcp4725ObjectType mcp4725;
    
    • 1

      对于这个对象变量,我们还需要使用Mcp4725Initialization函数对它进行初始化才能使用。这个初始化函数有多个输入参数:

    Mcp4725ObjectType *mcp,  //MCP4725对象变量
    uint8_t slaveAddress,    //从站设备的地址
    Mcp4725PDModeType pdMode,//掉电操作模式
    Mcp4725Write write,      //写数据函数指针
    Mcp4725Read read         //读数据函数指针
    
    • 1
    • 2
    • 3
    • 4
    • 5

      这些参数中,第一个参数是我们要初始化的对象变量,已经在前面声明了。slaveAddress是指MCP4725单通道数模转换器的设备地址。掉电操作模式是枚举类型,根据使用需要选择就可以了。最后两个读写操作函数指针则需要我们实现相应的函数。这两个函数的原型定义如下:

    typedef void (*Mcp4725Write)(struct Mcp4725Object *mcp,uint8_t *wData, uint16_t wSize);
    typedef void (*Mcp4725Read)(struct Mcp4725Object *mcp,uint8_t *rData, uint16_t rSize);
    
    • 1
    • 2

      读写操作函数的实现与具体的软硬件平台是相关的,这里我们实现STM32F103硬件平台和HAL库的对应函数:

    /*通过I2C1端口写MCP4725*/
    static void BmcbMcp4725Write(struct Mcp4725Object *mcp,uint8_t *wData, uint16_t wSize)
    {
        HAL_I2C_Master_Transmit(&hi2c1,mcp->devAddress,wData,wSize,1000);
    }
    
    /*通过I2C1端口读MCP4725*/
    static void BmcbMcp4725Read(struct Mcp4725Object *mcp,uint8_t *rData, uint16_t rSize)
    {
        HAL_I2C_Master_Receive(&hi2c1,mcp->devAddress,rData,rSize,1000);
    }
    有了这些参数后,我们就可以使用这些参数来初始化MCP4725单通道数模转换器的对象变量了。
    Mcp4725Initialization(&mcp4725,         //MCP4725对象变量
                              0xC0,             //从站设备的地址
                              MCP4725_Normal,   //掉电操作模式
                              BmcbMcp4725Write, //写数据函数指针
                              BmcbMcp4725Read   //读数据函数指针
                                   );
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    3.2、基于对象进行操作

      关于对象的应用这块,我们将实际工程中的应用代码节选过来。具体很简单就是计算当前应该下发的数字编码并将其下发给MCP4725单通道数模转换器就可以了。

    uint16_t code=0;
    code=(uint16_t)((aPara.phyPara.presControl/100.0)*4095.0);
    Mcp4725SetDatas(&mcp4725,Mcp4725_Write_DAC,code);
    
    • 1
    • 2
    • 3

    4、应用总结

      我们设计并实现了MCP4725单通道数模转换器的驱动程序,而且将其运用到了实际的工程当中,使用情况符合我们的预期。
    源码下载:https://github.com/foxclever/ExPeriphDriver

    欢迎关注:

  • 相关阅读:
    NetBeans IDE8.0.2下git使用
    独处是一种修行
    数电学习(八、九、可编程逻辑器件)
    LeetCode 278. 第一个错误的版本
    【0day】泛微e-office OA未授权访问漏洞学习
    盲盒商城系统玩法大讲坛
    一招解决所有依赖冲突
    为github项目提交贡献步骤
    python编程用什么软件?Pycharm好不好?
    VSCode远程开发 Windows11 Linux
  • 原文地址:https://blog.csdn.net/foxclever/article/details/127716256