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. android L 新控件侧滑菜单DrawerLayout 使用教程

    介绍 drawerLayout是Support Library包中实现了侧滑菜单效果的控件,可以说drawerLayout是因为第三方控件如MenuDrawer等的出现之后,google借鉴而出现的产 ...

  2. unity两点之间抛物线,完美金手指

    学校享受的日子一去不复还了,呜呜.话说面试了几个公司,真心没准备好就上了,结果当然是小悲催.还好有容身之处,就算是搬砖,也有可能为自己盖楼,吼吼. 好,下面我来分享一道有意思的面试题,说他有意思,是因 ...

  3. Jackson 框架,轻易转换JSON(转)

    Jackson可以轻松的将Java对象转换成json对象和xml文档,同样也可以将json.xml转换成Java对象. 相比json-lib框架,Jackson所依赖的jar包较少,简单易用并且性能也 ...

  4. 常见行为:仿真&重力&碰撞&捕捉

    一.UIDynamic是从iOS 7开始引入的一种新技术,隶属于UIKit框架.可以认为是一种物理引擎,能模拟和仿真现实生活中的物理现象,重力.弹性碰撞等,游戏开发中很常见,例如愤怒的小鸟. 二.UI ...

  5. c语言解数独

    来自:http://my.oschina.net/lovewxm/blog/288043?p=1 #include <stdio.h> #include <stdlib.h> ...

  6. LA3211 飞机调度 Now or later-二分法&TwoSet

    https://vjudge.net/problem/UVALive-3211 As you must have experienced, instead of landing immediately ...

  7. KMP算法模板

    不懂的话推荐看这篇博客,讲的很清楚 http://blog.csdn.net/v_july_v/article/details/7041827 #include<iostream> #in ...

  8. [JFinal 1] JFinal和SSH中使用拦截器的对比

    导读:先前在做.NET项目时,拦截的功能主要是依靠缓存session来实现.当需要跳转到某个页面,使用某个功能查询一些数据时,会根据session中的用户值来判断是否已经正常登录,如果没有,则重定向到 ...

  9. WP8_UTF8 to GB2312转码 (url网址中带中文字符的处理)

    直接使用例如:http://www.abc.php?name=中文符 ,客户端调用,在服务端修改后,会出现乱码, 而windows phone 又不能直接支持gb2312, 经过大量分析和验证,发现 ...

  10. Heavily reliance on forensic tools is risky

    We could take advantage of forensic tools to examine and analyze the evidence, but heavily reliance ...