STM32F103利用模拟I2C驱动ADS1115

ADS1115通过模拟I2C驱动:(部分代码借鉴了网络上的几个,并且根据引脚进行了配置,都没有运行成功,今天调了一天,终于在晚上调了出来)

注意:本部分代码需要只是ADS1115的部分程序(一些用到的数组在此没有写),模拟II2C的各程序并未给出,大家根据需要进行裁剪,代码完全开源,希望能帮到大家,也希望大家乐于分享。

   调试时需特别注意I2C的通信问题,特别是应答信号的使用需要特别关注,本人的程序之前出错全是因为从机的应答信号的未使用造成。

           如有其它问题,欢迎指正批评,第一次写博客,感谢!


收获:对于I2C通信协议的理解更加深刻,对ADS1115也能进行单通道的使用。I2C的开始、发送/读写、应答、结束等一定要严格按照时序操作,ADS的操作还有所欠缺,仅仅会配置单通道,对于其他配置并没有关注,最近要把多通道的和阈值的配置也写好,到时再更新


附录代码:

#define CMD_Write      0x90                   写入命令 1001 0000  前7位表示ADS1115地址,最后1位0表示写
#define CMD_Read       0x91      读取命令  1001 001   前7位表示ADS1115地址,最后1位1表示读
#define CMD_POINT_REG  0x00         指向寄存器配置
#define CMD_CONF_REG   0x01        /配置寄存器配置
#define CONF_L    0xe3        低8位
#define ADS1115_ADDR   0x90   /* ADS1115的设备地址,需配置 */

/*******************************************************************************
* Function Name  : Confige1115
* 参数           : 通道0/1/2/3
* Attention :配置ADS1115,根据需要的通道进行配置
*******************************************************************************/

static void Confige1115 (unsigned char port)
{
    static unsigned char chnel, i;
    switch (port)
    {
      case 0:               //0通道
          chnel=0xC2;
      break;
      
      case 1:               //1通道  
          chnel=0xD2;
      break;
          
      case 2:               //2通道  
          chnel=0xE2;
      break;
          
      case 3:               //3通道
          chnel=0xF2;
      break;
          
      default:
      break;
    }
    WriteIntBuf[0] = CMD_Write;                  //0x90
    WriteIntBuf[1] = CMD_CONF_REG;     //0x01
    WriteIntBuf[2] = chnel;  // 写入要配置的通道
    WriteIntBuf[3] = CONF_L;                       //0xe3

    i2c_Start();   /开启I2C
    
for(i=0;i<4;i++)
{
i2c_SendByte(WriteIntBuf[i]);    /发送寄存器配置信息
ads1115_delay(20);
        i2c_WaitAck();            注意,此处在最初调试时没有,读出数据一直不变,加上后正常
}
    
       i2c_Stop();   /停止I2C
}

/*******************************************************************************
* Function Name  : PointRegister
* Attention   :  指向ADS1115指针寄存器用于准备读取数据
*******************************************************************************/
static void PointRegister (void)
{
    unsigned char i;
    WritepointBuf[0] = CMD_Write; //90
    WritepointBuf[1] = CMD_POINT_REG; //00

   i2c_Start();     /开启I2C
   for(i=0;i<2;i++)
   {
i2c_SendByte(WritepointBuf[i]);   /发送寄存器配置信息
i2c_WaitAck();
ads1115_delay(20);
   }
    i2c_Stop();  /停止I2C
}

/*******************************************************************************
* 读取ADS1115的16位数据
*******************************************************************************/

static uint16_t ReadData (unsigned char chnnal1)
{  

uint16_t  data;

i2c_Start();     /开启I2C  

i2c_SendByte(CMD_Read);   /发送读命令
        ads1115_delay(20);
i2c_WaitAck();      注意,加上此句后程序正常,不加时我测试的读错数据,大家可能没事

ReadBuffer[0] = i2c_ReadByte();   读取数据高位
i2c_Ack();   /主机向ADS1115发送应答信号
ads1115_delay(200);

ReadBuffer[1] = i2c_ReadByte(); 读取数据低位
i2c_Ack(); /主机向ADS1115发送应答信号
ads1115_delay(200);

i2c_Stop();   发送停止信号
data = ReadBuffer[0]*256+ReadBuffer[1];  数据处理   

       return data;
}

/*******************************************************************************
* Function Name  : Get_ATOD
* Attention :获取ADS1115模拟转换结果
*******************************************************************************/
float Get_ATOD (unsigned char channel)
{
        static unsigned char chn; 

chn = channel; 

  Confige1115(channel); 配置ADS1115转换通道
      ads1115_delay(1000);

        PointRegister();  指向ADS1115指针寄存器用于准备读取数据
        ads1115_delay(1000);
        data_get = ReadData(chn);  从通道中读数据
        ads1115_delay(1000);
/**用于测量负电压,负电压从8000~ffff,负电压与正关于0有类似对称关系,按位取反后+1相同**/
if(data_get>=0x8000)  
dianya=((float)(0xffff-data_get)/32768.0)*4.096;
else
dianya=((float)data_get/32768.0)*4.096;

return dianya;    
}



  • 7
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是基于STM32F103ADS1115驱动程序示例: ```c #include "stm32f10x.h" #include "delay.h" #include "ads1115.h" #define ADS1115_ADDRESS 0x90 // ADS1115的地址为0x90 void ADS1115_Init(void) { I2C1_Init(); // 初始化I2C总线 I2C1_Start(); // 启动I2C总线 I2C1_SendByte(ADS1115_ADDRESS); // 发送从设备地址 I2C1_SendByte(0x01); // 发送配置寄存器的地址 I2C1_SendByte(0xC5); // 发送配置字节MSB I2C1_SendByte(0x83); // 发送配置字节LSB I2C1_Stop(); // 停止I2C总线 } int16_t ADS1115_GetValue(uint8_t channel) { uint8_t data[2]; int16_t value; I2C1_Start(); // 启动I2C总线 I2C1_SendByte(ADS1115_ADDRESS); // 发送从设备地址 I2C1_SendByte(0x00); // 发送转换寄存器地址 I2C1_Stop(); // 停止I2C总线 delay_ms(1); // 延时等待转换完成 I2C1_Start(); // 启动I2C总线 I2C1_SendByte(ADS1115_ADDRESS | 0x01); // 发送读取命令 data[0] = I2C1_ReadByte(1); // 读取数据字节MSB并发送NACK data[1] = I2C1_ReadByte(0); // 读取数据字节LSB并发送ACK I2C1_Stop(); // 停止I2C总线 value = (int16_t)(data[0] << 8) | data[1]; // 合并数据字节 if (value < 0) // 如果是负数 { value = ~value + 1; // 取反加一 } switch (channel) // 根据通道号计算转换后的值 { case 0: value *= 6.144 / 32768; break; case 1: value *= 4.096 / 32768; break; case 2: value *= 2.048 / 32768; break; case 3: value *= 1.024 / 32768; break; } return value; } ``` 在上面的示例代码中,我们使用了ADS1115的单次转换模式,配置字节为0xC583,通道0-3的增益分别为6.144、4.096、2.048和1.024。函数`ADS1115_Init()`用于初始化ADS1115,函数`ADS1115_GetValue()`用于获取指定通道的转换值。在读取转换值之前需要延时一段时间等待转换完成。转换完成后,读取数据字节MSB和LSB,并根据通道号计算转换后的值。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值