3  i2c-dev

3.1 概述

之前在介绍I2C子系统时,提到过使用i2c-dev.c文件在应用程序中实现我们的I2C从设备驱动。不过,它实现的是一个虚拟,临时的i2c_client,随着设备文件的打开而产生,并随着设备文件的关闭而撤销。I2c-dev.c针对每个I2C适配器生成一个主设备号为89的设备文件,实现了i2c_driver的成员函数以及文件操作接口,所以i2c-dev.c的主题是”i2c_driver成员函数+字符设备驱动”。

3.2 i2c-dev.c源码分析

初始化模块

  1. static int __init i2c_dev_init(void)
  2. {
  3. res= register_chrdev(I2C_MAJOR, "i2c", &i2cdev_fops);
  4. i2c_dev_class= class_create(THIS_MODULE, "i2c-dev");
  5. /*Keep track of adapters which will be added or removed later */
  6. res= bus_register_notifier(&i2c_bus_type, &i2cdev_notifier);
  7. /*绑定已经存在的适配器 */
  8. i2c_for_each_dev(NULL,i2cdev_attach_adapter);
  9. }

I2c-dev初始化函数主要做了注册名为”i2c”的字符设备文件和”i2c-dev”的类

i2cdev_read和i2cdev_write

I2c-dev.c中实现的i2cdev_read和i2cdev_write函数不具有太强的通用性,只适合下面这种单开始信号情况:

而不适合多开始信号的情况:

所以我们经常会使用i2cdev_ioctl函数的I2C_RDWR,在分析i2cdev_ioctl函数之前,我们需要了解一个结构体:

  1. /* This is the structure as used in theI2C_RDWR ioctl call */
  2. struct i2c_rdwr_ioctl_data {
  3. structi2c_msg __user *msgs;         /* pointersto i2c_msgs */
  4. __u32nmsgs;                    /* number ofi2c_msgs */
  5. };

Msgs     表示单个开始信号传递的数据;

Nmsgs     表示有多少个msgs,比如上图,单开始信号时,nmsgs等于1;多开始信号时,nmsgs等于2

  1. struct i2c_msg {
  2. __u16addr;     /* slave address                         */
  3. __u16flags;  /* 默认为写入 */
  4. #define I2C_M_TEN                  0x0010     /*this is a ten bit chip address */
  5. #define I2C_M_RD           0x0001     /* read data,from slave to master */
  6. #define I2C_M_NOSTART                  0x4000     /* if I2C_FUNC_PROTOCOL_MANGLING */
  7. #define I2C_M_REV_DIR_ADDR     0x2000     /*if I2C_FUNC_PROTOCOL_MANGLING */
  8. #define I2C_M_IGNORE_NAK          0x1000     /*if I2C_FUNC_PROTOCOL_MANGLING */
  9. #define I2C_M_NO_RD_ACK           0x0800     /* if I2C_FUNC_PROTOCOL_MANGLING */
  10. #define I2C_M_RECV_LEN               0x0400     /* length will be first received byte */
  11. __u16len;                  /* msg length                              */
  12. __u8*buf;                 /* pointer to msgdata                       */
  13. };

3.3 eeprom实例

预备知识

使用的ok6410开发板,eeprom的地址为0x50,实验完成一个数据的读写,先看下读写时序

AT24C02任意地址字节写的时序:

AT24C02任意地址字节写的时序:

用户态驱动:

#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/ioctl.h>
#include <stdlib.h>
//#include <linux/types.h>

#define I2C_RDWR 0x0707

struct i2c_msg {
unsigned short addr; /* slave address */
unsigned short flags;
unsigned short len; /* msg length */
unsigned char *buf; /* pointer to msg data */
};

struct i2c_rdwr_ioctl_data {
struct i2c_msg *msgs; /* pointers to i2c_msgs */
unsigned int nmsgs; /* number of i2c_msgs */
};

int main()
{
int fd;
struct i2c_rdwr_ioctl_data e2prom_data;

//1. 打开通用设备文件
fd = open("/dev/i2c-0", O_RDWR);

//为i2c_rdwr_ioctl_data中的struct i2c_msg *分配空间
e2prom_data.msgs = (struct i2c_msg *)malloc(2*sizeof(struct i2c_msg)); // 构造两条消息

//2. 构造写数据到eeprom
e2prom_data.nmsgs = 1; // 只有一条消息
(e2prom_data.msgs[0]).len = 2; //长度等于2,第一个字节代表的是i2c设备的内部地址,第二个字节代表的是写入的数据
(e2prom_data.msgs[0]).addr = 0x50; // 从设备地址(e2prom的地址),注意这里是不带方向的!
(e2prom_data.msgs[0]).flags = 0; // 方向由flag标志位来指明,0代表了写,1代表了读
(e2prom_data.msgs[0]).buf = (unsigned char*)malloc(2); // 这里只分配两个字节(内部偏移地址一字节,数据1字节)
(e2prom_data.msgs[0]).buf[0] = 0x10; // 数据将写入e2prom中的内部0x10地址中
(e2prom_data.msgs[0]).buf[1] = 0x60; // 写入e2prom中内 部0x10地址中的数据位0x60

//3. 使用ioctl写入数据
ioctl(fd, I2C_RDWR, (unsigned long)&e2prom_data, I2C_RDWR); // 这里面的命令参数对应的是驱动内部的ioctl中的case语句中的参数

//4. 构造从eeprom读数据的消息
e2prom_data.nmsgs = 2; // 读数据需要两条消息
(e2prom_data.msgs[0]).len = 1; //长度为一个字节,代表的是i2c设备的内部地址
(e2prom_data.msgs[0]).addr = 0x50; // 从设备地址(e2prom的地址),注意这里是不带方向的!
(e2prom_data.msgs[0]).flags = 0; // 方向由flag标志位来指明,0代表了写,1代表了读
(e2prom_data.msgs[0]).buf[0] = 0x10; // 数据将写入e2prom中的内部0x10地址中
// 第二条消息(读数据)
(e2prom_data.msgs[1]).len = 1; //长度为一个字节,代表的是i2c设备的内部地址
(e2prom_data.msgs[1]).addr = 0x50; // 从设备地址(e2prom的地址),注意这里是不带方向的!
(e2prom_data.msgs[1]).flags = 1; // 方向由flag标志位来指明,0代表了写,1代表了读
(e2prom_data.msgs[1]).buf = (unsigned char*)malloc(2);
(e2prom_data.msgs[1]).buf[0] = 0; // 数据从e2prom中的内部0x10地址中读出

//5. 使用ioctl读出数据
ioctl(fd, I2C_RDWR, (unsigned long)&e2prom_data);
printf("buf[0] = %x\n", (e2prom_data.msgs[1]).buf[0]);

//6. 关闭设备
close(fd);
}

  

linuxok6410的I2C驱动分析---用户态驱动的更多相关文章

  1. [国嵌攻略][155][I2C用户态驱动设计]

    用户态驱动模型 用户态驱动模型首先是一个应用程序,其次是在这个用户程序中通过内核调用来驱动设备. IIC通用驱动代码 IIC通用驱动程序的代码在/drivers/i2c/i2c-dev.c中.一次读操 ...

  2. Linux I2C驱动--用户态驱动简单示例

    1. Linux内核支持I2C通用设备驱动(用户态驱动:由应用层实现对硬件的控制可以称之为用户态驱动),实现文件位于drivers/i2c/i2c-dev.c,设备文件为/dev/i2c-0 2. I ...

  3. I2C用户态驱动设计

    一.用户态驱动模型 1.1 I2C通用驱动代码 i2c_dev_init: static int __init i2c_dev_init(void) { int res; printk(KERN_IN ...

  4. Linux I2C驱动分析(三)----i2c_dev驱动和应用层分析 【转】

    本文转载自:http://blog.chinaunix.net/uid-21558711-id-3959287.html 分类: LINUX 原文地址:Linux I2C驱动分析(三)----i2c_ ...

  5. 聊聊Linux用户态驱动设计

    序言 设备驱动可以运行在内核态,也可以运行在用户态,用户态驱动的利弊网上有很多的讨论,而且有些还上升到政治性上,这里不再多做讨论.不管用户态驱动还是内核态驱动,他们都有各自的缺点.内核态驱动的问题是: ...

  6. Linux用户态驱动设计

    聊聊Linux用户态驱动设计   序言 设备驱动可以运行在内核态,也可以运行在用户态,用户态驱动的利弊网上有很多的讨论,而且有些还上升到政治性上,这里不再多做讨论.不管用户态驱动还是内核态驱动,他们都 ...

  7. v79.01 鸿蒙内核源码分析(用户态锁篇) | 如何使用快锁Futex(上) | 百篇博客分析OpenHarmony源码

    百篇博客分析|本篇为:(用户态锁篇) | 如何使用快锁Futex(上) 进程通讯相关篇为: v26.08 鸿蒙内核源码分析(自旋锁) | 当立贞节牌坊的好同志 v27.05 鸿蒙内核源码分析(互斥锁) ...

  8. 用户态驱动--UIO机制的实现【转】

    转自:https://blog.csdn.net/u013982161/article/details/51584900 1 uio理论部分   1.1为什么出现了UIO? 硬件设备可以根据功能分为网 ...

  9. tiny4412 串口驱动分析六 --- TTY驱动架构

    转载: http://www.linuxidc.com/Linux/2013-11/92639.htm 参考: http://blog.csdn.net/lamdoc/article/details/ ...

随机推荐

  1. NEERC 2013, Eastern subregional contest

    I. The old Padawan Time limit: 0.5 secondMemory limit: 64 MB Yoda: Use the Force. Yes. Now, the ston ...

  2. 用FireBreath制作浏览器插件

    参考: http://blog.csdn.net/z6482/article/details/7486921 1.下载firebreath, 安装cmake, python. 2.在FireBreat ...

  3. Python安装Selenium3

    概述 2016.10.13,Selenium3.0正式发布,官方说明如下: The major change in Selenium 3.0 is we're removing the origina ...

  4. github添加ssh认证

    总概:在使用git的时候,和目标仓库建立关系有两种方式https,ssh.一般用的是https认证(这样简单方便),但有个缺点,pull,push等操作需要频繁输入用户验证.虽然可以把用户验证账号密码 ...

  5. Ext.Net 破解

    在使用 Ext.Net 框架时,如果没有得到正版授权(安装密钥),在站点发布后,打开界面总是弹出一个窗口,提示没有授权,看着都头疼,难道一定要安装密钥吗?但还是有办法解决的,在研究时发现,页面中多了两 ...

  6. 【Hibernate 8】Hibernate的调优方法:抓取策略

    在上一篇博客中,介绍了Hibernate的缓存机制.合理的配置缓存,可以极大程度上优化Hibernate的性能.这篇博客,介绍另外一个调优方式:抓取策略. 一.什么是抓取策略 抓取策略(fetchin ...

  7. 【Python】django安装

    官方下载:https://www.djangoproject.com/download/ 报错 [root@test Django-]# python setup.py install Traceba ...

  8. Loadrunner:安装LR11时提示缺少vc2005_sp1_with_atl_fix_redist

    [问题现象] 安装LR11时提示缺少vc2005_sp1_with_atl_fix_redist: [解决办法] 手动安装缺少的组件,LR安装包中已自带该组件,为何不自动捕捉异常去获取该自带的组件去安 ...

  9. 七个你可能不了解的CSS单位

    我们很容易无法摆脱的使用我们所熟悉的CSS技术,当新的问题出现,这样会使我们处于不利的地位. 随着Web继续的发展,对新的解决方案的需求也会继续增大.因此,作为网页设计师和前端开发人员,我们别无选择, ...

  10. 如何使VS2008 调试网站的根目录和IIS调试的一致?

    用VS2008做asp.net网站调试时,经常会多出来一个目录,如http://localhost:1234/Foo/ , 由于一些图片的路径问题,我们不需要最后的/Foo/目录,而是像IIS调试那样 ...