当前位置:文档之家› Linux驱动之i2c用户态调用

Linux驱动之i2c用户态调用

一、概述I2C只有两条线,一条串行数据线:SDA,一条是时钟线SCL.正因为这样,它方便了工程人员的布线.二、用户态实现设备驱动在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.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.#define I2C_RETRIES0x0701/*设置收不到ACK时的重试次数*/2.#define I2C_TIMEOUT0x0702/*设置超时时限的jiffies*/3.#define I2C_SLAVE0x0703/*设置从机地址*/4.#define I2C_SLAVE_FORCE0x0706/*强制设置从机地址*/5.#define I2C_TENBIT0x0704/*选择地址位长:=0for7bit,!=0for10bit*/6.#define I2C_FUNCS0x0705/*获取适配器支持的功能*/7.#define I2C_RDWR0x0707/*Combin ed R/W transfer(one STOP only)*/8.#define I2C_PEC0x0708/* !=0to use PEC with SMBus*/9.#define I2C_SMBUS0x0720/*SMBus transfer*/例如: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 #bytestransmitted, 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 #bytestransmitted, 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.#include<stdio.h>2.#include<linux/i2c.h>3.#include<linux/i2c-dev.h>4.#include<fcntl.h>5.#include<stdio.h>6.#include<unistd.h>7.#include<sys/ioctl.h>8.#include<string.h>9.#include<stdlib.h>10.11.#define I2C_FILE_NAME "/dev/i2c-1"12.#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 buf[2];29.struct i2c_rdwr_ioctl_data data;30.struct i2c_msg messages;31.32.buf[0] = reg_addr;33.buf[1] = 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 err\n");43.return 1;44.}leep(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 messages[2];55.56.messages[0].addr = addr; //device address57.messages[0].flags = 0; //write58.messages[0].len = sizeof(reg);59.messages[0].buf = &reg; //data address60.61.messages[1].addr = addr; //device address62.messages[1].flags = I2C_M_RD; //read63.messages[1].len = sizeof(val);64.messages[1].buf = val;65.66.data.msgs = messages;67.data.nmsgs = 2;68.if(ioctl(fd, I2C_RDWR, &data) < 0){69.printf("read ioctl err\n");70.return 1;71.}72.73.return 0;74.}75.76.int main()77.{78.int i;79.unsigned char buf[4];80.unsigned char val[] = {0x04, 0x05, 0x06, 0x07};81.82.i2c_open();83.84.85.for(i =0; i< 4; i++)86.i2c_write(fd, I2C_ADDR, i, val[i]);87.88.89.memset(buf, 0x00, sizeof(buf));90.for(i = 0; i < sizeof(buf); i++){91.if(i2c_read(fd, I2C_ADDR, i, &buf[i])){92.printf("Unable to get register!\n");93.}94.}95.96.for(i=0; i< 4;i++)97.printf("buf[%d]=%d\n",i, buf[i]);98.99.}。

相关主题