I2C RepStart Mode 的编写
请问如何在应用层使用I2C_RDWR的ioctl 读的操作, 操作内核层的i2c dev.c结构。
如实现fun_read(int slave, int addr, uint8_t *rx, int rxlen) 对salve的设备对addr地址进行读取操作。
work_queue.nmsgs = 2;
work_queue.msgs = (struct i2c_msg*)malloc(work_queue.nmsgs * sizeof(struct i2c_msg));
(work_queue.msgs[0]).len = 1;
(work_queue.msgs[0]).addr = slave>>1;
(work_queue.msgs[0]).flags=0;
(work_queue.msgs[0]).buf = &addr;
(work_queue.msgs[1]).len = rxlen;
(work_queue.msgs[1]).addr = slave>>1;
(work_queue.msgs[1]).flags=1;
(work_queue.msgs[1]).buf = rx;
ret = ioctl(fd, I2C_RDWR, (unsigned long)&work_queue);
free(work_queue.msgs);
if(ret<0)
{
printf("Error dring I2C_RDWR ioctl with error code: %d\n",ret);
return ret;
}
return ret;
1. 请问 为什么要用2个msg? 不是一个吗?
2. 请问如何写操作是使用i2cdev_write()(即是直接fd=open /dev/i2c. 然后write(fd...))会有问题吗?
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(9)
1.如果才一个msgs 那么offset command放在哪个变量呢?
都往msg[0].buff里放。
目前你的代码只放了address, 而且buff是个指针,指向了slave_address.
你需要申请一个buffer, 用来暂存slave_address & offset, 然后把这个buffer的地址赋给msg[0].buff
2.单独write是没有问题的。 操作了read(re-start模式)后在写就有问题了。 why? investigating..
如果单独write没问题, 你试试write + read呢,是否正常?
或者说,你的单独的read正常吗?
1. 请问 为什么要用2个msg? 不是一个吗?
读操作, 是从哪个设备地址,哪个偏移,用什么读命令,来进行操作。对于读操作,其实包含两个过程:
a. master把设备相关的一些信息发过去,即往总线上发slave address, slave offset, command(command为可选项)
b. 再从slave读取数据。
2. 请问如何写操作是使用i2cdev_write()(即是直接fd=open /dev/i2c. 然后write(fd...))会有问题吗?
不是太肯定。可以试一下。你要保证:
a. 参数传递正确。
b. i2cdev_write能最终调用i2c的write命令。
加点调试信息,确保上面两步无误。
补充一下,
从下面的代码来看,
(work_queue.msgs[0]).len = 1;
(work_queue.msgs[0]).addr = slave>>1;
(work_queue.msgs[0]).flags=0;
(work_queue.msgs[0]).buf = &addr;
你的I2C设备,读命令时,只传递了slave address而已,不需要传递offset, command等。
问一下LZ,
I2C RepStart Mode, 这里的RepStart是I2C的Re-start吗?
是的。 repeat-start 重复开始模式。 就是I2C模式2.
要求时序为: Start | Slave_Address | 0 | Ack | Register_Address_MSB | Ack | Register_Address_LSB | Ack |
Re-start | Slave_Address | 1 | Ack | XXXX_MSB | Ack | XXXX_LSB | NAck or Stop.
1. 请问 为什么要用2个msg? 不是一个吗?
读操作, 是从哪个设备地址,哪个偏移,用什么读命令,来进行操作。对于读操作,其实包含两个过程:
a. master把设备相关的一些信息发过去,即往总线上发slave address, slave offset, command(command为可选项)
b. 再从slave读取数据。
从下面的代码来看,
(work_queue.msgs[0]).len = 1;
(work_queue.msgs[0]).addr = slave>>1;
(work_queue.msgs[0]).flags=0;
(work_queue.msgs[0]).buf = &addr;
你的I2C设备,读命令时,只传递了slave address而已,不需要传递offset, command等。
>>offset 是需要的。 但是offset是通过那个变量来储存和发送呢?是否msgs·[0]通过写发送addr=slave address buf=&addr偏移量(offset)。然后msgs[1]通过读操作读addr=slave 读到buf上去?
如果才一个msgs 那么offset command放在哪个变量呢?
2. 请问如何写操作是使用i2cdev_write()(即是直接fd=open /dev/i2c. 然后write(fd...))会有问题吗?
不是太肯定。可以试一下。你要保证:
a. 参数传递正确。
b. i2cdev_write能最终调用i2c的write命令。
加点调试信息,确保上面两步无误。
》单独write是没有问题的。 操作了read(re-start模式)后在写就有问题了。 why? investigating...
1.如果才一个msgs 那么offset command放在哪个变量呢?
都往msg[0].buff里放。
目前你的代码只放了address, 而且buff是个指针,指向了slave_address.
你需要申请一个buffer, 用来暂存slave_address & offset, 然后把这个buffer的地址赋给msg[0].buff
>>你的意思是说:
(work_queue.msgs[0]).len = 1;
(work_queue.msgs[0]).addr = slave>>1;
(work_queue.msgs[0]).flags=0;
(work_queue.msgs[0]).buf = slave_address & offset 还是 只是offset?
2.单独write是没有问题的。 操作了read(re-start模式)后在写就有问题了。 why? investigating..
如果单独write没问题, 你试试write + read呢,是否正常?
或者说,你的单独的read正常吗?
>> write->read 是成功的。 读不了数据的。 因为read是mode 1. 时序不对。
参考代码 http://blog.chinaunix.net/u3/91468/showart_1798987.html
linux下24c08的i2c读写程序
#include <stdio.h>
#include <linux/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/ioctl.h>
#include <errno.h>
#include <assert.h>
#include <string.h>
#define MAX_I2C_MSG 2
#define I2C_RETRIES 0x701
#define I2C_TIMEOUT 0x702
#define I2C_RDWR 0x707
#define I2C_M_RD 0x1
struct i2c_msg
{
__u16 addr;
__u16 flags;
__u16 len;
__u8 *buf;
};
struct i2c_rdwr_ioctl_data
{
struct i2c_msg *msgs;
int nmsgs;
};
void clear_whole_eeprom();
void set_whole_eeprom();
void read_whole_eeprom();
void print_usage(__u8 *argv)
{
printf("\nUsage: %s [-c\\-s\\-r]\n",argv);
}
int main(int argc, char **argv)
{
int cmd;
if (argc!=2)
{
print_usage(argv[0]);
return 0;
}
if(strcmp(argv[1],"-c") == 0)
cmd=1;
else if(strcmp(argv[1],"-s")==0)
cmd=2;
else if(strcmp(argv[1],"-r")==0)
cmd=3;
else
{
print_usage(argv[0]);
return 0;
}
int fd;
fd = open("/dev/i2c/0",O_RDWR);
if(fd<=0)
{
printf("ERROR on opening the device file,fd=%d\n",fd);
return 0;
}
switch(cmd)
{
case 1:
clear_whole_eeprom();
break;
case 2:
set_whole_eeprom();
break;
case 3:
read_whole_eeprom();
break;
default:
break;
}
close(fd);
return 0;
}
__s32 write_eeprom(__s32 fd,__u32 offset,__u32 len,__u8 *buf)
{
__s32 ret;
struct i2c_rdwr_ioctl_data work_queue;
__u8 tmp[24];
__u32 i,bytes;
bytes=len;
if((offset+len)>1024)
{
printf("write too long than 1024\n");
return -1;
}
work_queue.nmsgs = 1;
work_queue.msgs = (struct i2c_msg*)malloc(work_queue.nmsgs * sizeof(struct i2c_msg));
do{
work_queue.msgs->len=len>(16-offset%16)?(16-offset%16+1):len+1;
work_queue.msgs->addr=0x50+offset/256;
work_queue.msgs->flags=0;//write
tmp[0]=offset%256;
memcpy(tmp+1,buf,16);
work_queue.msgs->buf = tmp;
#if 0
printf("buf=%x,offset=%u,len=%u,addr=%x\n",buf,tmp[0],work_queue.msgs->len,work_queue.msgs->addr);
printf("tmp=");
for(i=0;i<(work_queue.msgs->len-1);i++)
{
printf("%u ",tmp[i+1]);
}
printf("\n");
#endif
ret = ioctl(fd, I2C_RDWR, (unsigned long)&work_queue);
if(ret<0)
{
printf("Error dring I2C_RDWR ioctl with error code: %d\n",ret);
return ret;
}
len = len - (work_queue.msgs->len-1);
buf=buf+16-offset%16;
offset=offset+(16-offset%16);
// usleep(1);
}while(len>0);
return bytes;
}
__s32 read_eeprom(__s32 fd,__u32 offset,__u32 len,__u8 *buf)
{
struct i2c_rdwr_ioctl_data work_queue;
__u8 tmp[24];
__s32 ret=0,bytes;
if((offset+len)>1024)
{
printf("read too long than 1024\n");
return -1;
}
bytes=len;
work_queue.nmsgs = 2;
work_queue.msgs = (struct i2c_msg*)malloc(work_queue.nmsgs * sizeof(struct i2c_msg));
(work_queue.msgs[0]).len = 1;
(work_queue.msgs[0]).addr = 0x50 + offset/256;
(work_queue.msgs[0]).flags=0;
tmp[0]=offset%256;
(work_queue.msgs[0]).buf = tmp;
(work_queue.msgs[1]).len = len;
(work_queue.msgs[1]).addr = 0x50 + offset/256;
(work_queue.msgs[1]).flags=1;
(work_queue.msgs[1]).buf = buf;
ret = ioctl(fd, I2C_RDWR, (unsigned long)&work_queue);
if(ret<0)
{
printf("Error dring I2C_RDWR ioctl with error code: %d\n",ret);
return ret;
}
return len;
}
void clear_whole_eeprom(int fd)
{
__s32 ret;
__u8 buf[1024];
bzero(buf,1024);
ret=write_eeprom(fd,0,1024,buf);
if(ret<0)
printf("write eeprom error\n");
else
printf("clear %u bytes\n",ret);
}
撞头ing
粗略看了内核i2c操作,应该这样
buf=malloc((rxlen)*sizeof(uint8_t));
memcpy(buf, &address, 2)
(work_queue.msgs[0]).len = rxlen;
(work_queue.msgs[0]).addr = slave>>1;
(work_queue.msgs[0]).flags=1;
(work_queue.msgs[0]).buf =buf;
ioctl(...)
memcpy(rx,buf,rxlen);
free
94. 你找个例子多好。 比我一步一步解释强多了。
还有问题时, 你三方面下手:
1. slave datasheet. 理解slave的时序,
2. kernel中i2c的框架, i2c module本身会帮你做哪些,
3. i2c master driver, 这是你要准备或处理的数据。
i2c module会把你准备好的数据,按固有的时序发出去,这个master时序严格匹配slave时序就能工作了。