iic是驱动非常常见的一种驱动总线,分为scl,sda,这里主要讲在rtt中的使用方法。当总线空闲时,两条线电平都是拉高的,传输每个字节都是8位,分别从高位到低位,
主机发送的第一个字节为从机地址,高 7 位为地址,最低位为 R/W 读写控制位,1 表示读操作,0 表示写操作。一般从机地址有 7 位地址模式和 10 位地址模式两种,如果是 10 位地址模式,第一个字节的头 7 位 是 11110XX 的组合,其中最后两位(XX)是 10 位地址的两个最高位,第二个字节为 10 位从机地址的剩下8位,如下图所示:
rt_device_find() //查找设备并获得设备句柄
rt_size_t rt_i2c_transfer(struct rt_i2c_bus_device *bus, struct rt_i2c_msg msgs[], rt_uint32_t num);//数据传输
首先第一步配置env工具,这里选择的模拟iic
其中地址位都是右移一位的,到这里,几乎是配置完成,另外在线库提供iic的例程
然后进入里面选择某个,然后更新最后生成工程文件即可。
参考写的24c02一个字节读写代码
#include
#include
#include "drv_i2c.h"
#include
#include "string.h"
#define EEP_ADDR 0x50
static struct rt_i2c_bus_device *eep_i2c_bus = RT_NULL;
#define EEP_I2CBUS_NAME "i2c1" /* 连接的I2C总线设备名称 */
static rt_err_t write_data(struct rt_i2c_bus_device *bus, rt_uint8_t addr, rt_uint8_t data)
{
rt_uint8_t buf[3],ret;
struct rt_i2c_msg msgs;
buf[0] = addr; //cmd
buf[1] = data;
msgs.addr = EEP_ADDR;
msgs.flags = RT_I2C_WR;
msgs.buf = buf;
msgs.len = 2;
ret=rt_i2c_transfer(bus, &msgs, 1);
rt_thread_mdelay(1);//要加延时才能成功
if(ret==1)
{
return RT_EOK;
}
else
{
return -RT_ERROR;
}
}
static rt_err_t read_regs(struct rt_i2c_bus_device *bus, rt_uint8_t addr, rt_uint8_t *buf)
{
rt_uint8_t buf1[2],ret;
struct rt_i2c_msg msgs;
buf1[0]=addr;
msgs.addr = EEP_ADDR;
msgs.flags = RT_I2C_WR;
msgs.buf = buf1;
msgs.len = 1;
ret=rt_i2c_transfer(bus, &msgs, 1);
if(ret==1)
{
rt_kprintf("1 ok
");
}
else
{
rt_kprintf(" 1fail
");
}
msgs.addr = EEP_ADDR;
msgs.flags = RT_I2C_RD;;
msgs.buf = buf;
msgs.len = 1;
ret=rt_i2c_transfer(bus, &msgs, 1);
if(ret==1)
{
rt_kprintf("2 ok
");
}
else
{
rt_kprintf("2 fail
");
}
}
int i2c_eeprom(void)
{
rt_uint8_t buf;
eep_i2c_bus = rt_i2c_bus_device_find(EEP_I2CBUS_NAME );
rt_kprintf("set i2c %s
", EEP_I2CBUS_NAME );
write_data(eep_i2c_bus,0x00,0x06);
read_regs(eep_i2c_bus,0x00,&buf);
rt_kprintf("EEP read data is: 0x%02x
",buf);
return 0;
}
int main()
{
i2c_eeprom();
return 0;
}
原作者:且偷浮生半日闲