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;
}