Linux驱动之i2c用户态调用.doc_第1页
Linux驱动之i2c用户态调用.doc_第2页
Linux驱动之i2c用户态调用.doc_第3页
Linux驱动之i2c用户态调用.doc_第4页
Linux驱动之i2c用户态调用.doc_第5页
已阅读5页,还剩4页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

1、 概述I2C只有两条线,一条串行数据线:SDA,一条是时钟线SCL.正因为这样,它方便了工程人员的布线.2、 用户态实现设备驱动 在Linux内核代码文件i2c-dev.c中实现了I2C适配器设备文件的功能,针对每个适配器生成一个主设备号为89的设备节点(次设备号为0-255),I2c-dev.c并没有针对特定的设备而设计,只是提供了通用的read(),write(),和ioctl()等文件操作接口,在用户空间的应用层就可以借用这些接口访问挂接在适配器上的I2C设备的存储空间或寄存器,并控制I2C设备的工作方式。i2c适配器的设备节点是/dev/i2c-x,其中x是数字。由于适配器编号是动态分配的(和注册次序有关),所以想了解哪一个适配器对应什么编号,可以查看/sys/class/i2c-dev/目录下的文件内容。3、 用户态调用3.1、i2c-dev 用户空间操作i2c,需要包含以下头文件。 打开适配器对应的设备节点i2c-dev为打开的线程建立一个i2c_client,但是这个i2c_client并不加到i2c_adapter的client链表当中。他是一个虚拟的临时client,当用户打开设备节点时,它自动产生,当用户关闭设备节点时,它自动被释放。3.2、ioctl()查看include/linux/i2c-dev.h文件,可以看到i2c支持的IOCTL命令1. #defineI2C_RETRIES0x0701/*设置收不到ACK时的重试次数*/2. #defineI2C_TIMEOUT0x0702/*设置超时时限的jiffies*/3. #defineI2C_SLAVE0x0703/*设置从机地址*/4. #defineI2C_SLAVE_FORCE0x0706/*强制设置从机地址*/5. #defineI2C_TENBIT0x0704/*选择地址位长:=0for7bit,!=0for10bit*/6. #defineI2C_FUNCS0x0705/*获取适配器支持的功能*/7. #defineI2C_RDWR0x0707/*CombinedR/Wtransfer(oneSTOPonly)*/8. #defineI2C_PEC0x0708/*!=0tousePECwithSMBus*/9. #defineI2C_SMBUS0x0720/*SMBustransfer*/例如:1、设置重试次数:ioctl(fd, I2C_RETRIES,m);设置适配器收不到ACK时重试的次数为m。默认的重试次数为12、设置超时ioctl(fd, I2C_TIMEOUT,m);设置SMBus的超时时间为m,单位为jiffies。3、设置从机地址ioctl(fd, I2C_SLAVE,addr);ioctl(fd, I2C_SLAVE_FORCE, addr);在调用read()和write()函数之前必须设置从机地址。这两行都可以设置从机的地址,区别是第二行无论内核中是否已有驱动在使用这个地址都会成功,第一行则只在该地址空闲的情况下成功。由于i2c-dev创建的i2c_client不加入i2c_adapter的client列表,所以不能防止其它线程使用同一地址,也不能防止驱动模块占用同一地址。 4、设置地址模式ioctl(file,I2C_TENBIT,select)如果select不等于0选择10bit地址模式,如果等于0选择7bit模式,默认7位模式。3.3数据包i2c发送或者接收一次数据都以数据包( struct i2c_msg )封装addr是设备从地址。 flags是通信标志,发送数据为0, 接收数据为I2C_M_RD。len是数据长度buf是传输数据3.4、接受数据设备驱动中我们通常调用 /driver/i2c/i2c-core.c 定义的接口i2c_master_recv 来接收一次数据。通过i2c_transfer调用数据包。int i2c_master_recv(struct i2c_client *client, char *buf ,int count) struct i2c_adapter *adap=client-adapter; / 获取adapter信息 struct i2c_msg msg; / 定义一个临时的数据包 int ret; msg.addr = client-addr; / 将从机地址写入数据包 msg.flags = client-flags & I2C_M_TEN; / 将从机标志并入数据包 msg.flags |= I2C_M_RD; / 将此次通信的标志并入数据包 msg.len = count; / 将此次接收的数据字节数写入数据包 msg.buf = buf; ret = i2c_transfer(adap, &msg, 1); / 调用平台接口接收数据 /* If everything went ok (i.e. 1 msg transmitted), return #bytes transmitted, else error code. */ return (ret = 1) ? count : ret; / 如果接收成功就返回字节数 EXPORT_SYMBOL(i2c_master_recv);参考驱动i2c_master_recv()函数封装属于自己用户态的接受函数。用户态是通过ioctl(handle-fd, I2C_RDWR, &data)函数与i2c从设备进行数据交互。主要有2个步骤:首先是写入需要读取的寄存器的地址,然后从寄存器中读取数据。需要2个数据包。如下:3.5、发送数据设备驱动中我们通常调用 /driver/i2c/i2c-core.c 定义的接口i2c_master_send来发送一次数据。通过i2c_transfer调用数据包int i2c_master_send(struct i2c_client *client,const char *buf ,int count) int ret; struct i2c_adapter *adap=client-adapter; / 获取adapter信息 struct i2c_msg msg; / 定义一个临时的数据包 msg.addr = client-addr; / 将从机地址写入数据包 msg.flags = client-flags & I2C_M_TEN; / 将从机标志并入数据包 msg.len = count; / 将此次发送的数据字节数写入数据包 msg.buf = (char *)buf; / 将发送数据写入数据包 ret = i2c_transfer(adap, &msg, 1); / 调用平台接口发送数据 /* If everything went ok (i.e. 1 msg transmitted), return #bytes transmitted, else error code. */ return (ret = 1) ? count : ret; / 如果发送成功就返回字节数 EXPORT_SYMBOL(i2c_master_send);参考驱动i2c_master_send()函数封装属于自己用户态的接受函数。用户态是通过ioctl(handle-fd, I2C_RDWR, &data)函数与i2c从设备进行数据交互。每次要写入两个字节数据主要包括写入的寄存器地址和要写入的数据。只需发送一次数据包。如下:3.6、使用案例1. #include2. #include3. #include4. #include5. #include6. #include7. #include8. #include9. #include10.11. #define I2C_FILE_NAME /dev/i2c-112. #define I2C_ADDR 0x4013.14. int fd;15.16. int i2c_open()17. 18. fd = open(I2C_FILE_NAME, O_RDWR);19. if(fd 0)20. perror(Unable to open i2c control file);21. return 1;22. 23. 24.25. int i2c_write(int fd, unsigned char dev_addr, unsigned char reg_addr, unsigned char val)26. 27. int ret;28. unsigned char buf2;29. struct i2c_rdwr_ioctl_data data;30. struct i2c_msg messages;31.32. buf0 = reg_addr;33. buf1 = val;34. messages.addr = dev_addr; /device address35. messages.flags = 0; /write36. messages.len = 2;37. messages.buf = buf; /data address38.39. data.msgs = &messages;40. data.nmsgs = 1;41. if(ioctl(fd, I2C_RDWR, &data) 0)42. printf(write ioctl errn);43. return 1;44. 45. usleep(1000);46.47. return 1;48. 49.50. int i2c_read(int fd, unsigned char addr, unsigned char reg, unsigned char *val)51. 52. int ret;53. struct i2c_rdwr_ioctl_data data;54. struct i2c_msg messages2;55.56. messages0.addr = addr; /device address57. messages0.flags = 0; /write58. messages0.len = sizeof(reg);59. messages0.buf = ® /data address60.61. messages1.addr = addr; /device address62. messages1.flags = I2C_M_RD; /read63. messages1.len = sizeof(val);64. messages1.buf = val; 65.66. data.msgs = messages;67. data.nmsgs = 2;68. if(ioctl(fd, I2C_RDWR, &data) 0)69. printf(read ioctl errn); 70. return 1;71. 72.73. return 0;74. 75.76. int main()77. 78. int i;79. unsigned char buf4;80. unsigned char val = 0x04, 0x05, 0x06, 0x07;81.82. i2c_open();83. 84.85. for(i =0; i 4; i+)86.

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论