最近需要操作24C02,封装了一下函数方便以后操作。

参考链接:

  https://my.oschina.net/handawei/blog/68526

  http://blog.csdn.net/onetwothreef/article/details/49488443

源码:

#include <stdio.h>
#include <linux/i2c.h>
#include <linux/i2c-dev.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h> #define I2C_DEFAULT_TIMEOUT 1
#define I2C_DEFAULT_RETRY 3 /*
* fd : 文件描述符
* timeout : 发送超时时间
* retry : 重复发送次数
*/
//重复发送次数可以设多一点,在调试的时候,只设置了一次,导致有时候发送会失败。
int i2c_set(int fd, unsigned int timeout, unsigned int retry)
{
if (fd == 0 )
return -1; if (ioctl(fd, I2C_TIMEOUT, timeout ? timeout : I2C_DEFAULT_TIMEOUT) < 0)
return -1;
if (ioctl(fd, I2C_RETRIES, retry ? retry : I2C_DEFAULT_RETRY) < 0)
return -1; return 0;
}
/*
* fd : 文件描述符
* addr : i2c的设备地址
* reg : 寄存器地址
* val : 要写的数据
* 描述 :从指定地址写数据
*/
int i2c_byte_write(int fd, unsigned char addr, unsigned char reg, unsigned char val)
{
int ret = 0;
unsigned char outbuf[2];
struct i2c_rdwr_ioctl_data packets;
struct i2c_msg messages; packets.nmsgs = 1;
packets.msgs = &messages; //发送要读取的寄存器地址
messages.addr = addr;
messages.flags = 0;
messages.len = 2; //寄存器地址加数据,共发送2个字节
messages.buf = outbuf;
outbuf[0] = reg;
outbuf[1] = val; ret = ioctl(fd, I2C_RDWR, (unsigned long)&packets); //读出来
if (ret < 0)
ret = -1; return ret;
} /*
* fd : 文件描述符
* addr : i2c的设备地址
* reg : 寄存器地址
* val : 要写的数据
* len : 数据长度
* 描述 :从指定地址写数据
* 24c02以8字节为1个page,如果在一个page里面写,写的字节长度超过这个page的末尾,
* 就会从page的开头写,覆盖开头的内容
*/
int i2c_nbytes_write(int fd, unsigned char addr, unsigned char reg, unsigned char *val, int len)
{
int ret = 0;
struct i2c_rdwr_ioctl_data packets;
struct i2c_msg messages;
int i; packets.nmsgs = 1;
packets.msgs = &messages; //发送要读取的寄存器地址
messages.addr = addr;
messages.flags = 0; //write
messages.len = len + 1; //数据长度
//发送数据
messages.buf = (unsigned char *)malloc(len+1);
if (NULL == messages.buf)
{
ret = -1;
goto err;
} messages.buf[0] = reg;
for (i = 0; i < len; i++)
{
messages.buf[1+i] = val[i];
} ret = ioctl(fd, I2C_RDWR, (unsigned long)&packets);//读出来
if (ret < 0){
printf("write error!\n");
return -1;
} err:
free(messages.buf); return ret;
} /*
* fd : 文件描述符
* addr : i2c的设备地址
* val : 保存读取数据
* 描述 :从当前地址读取一个字节数据
*/
int i2c_byte_read(int fd, unsigned char addr, unsigned char *val)
{
int ret = 0;
struct i2c_rdwr_ioctl_data packets;
struct i2c_msg messages; packets.nmsgs = 1; //数据帧类型只有一种,读操作,只需要发送一个起始信号,因此是1
packets.msgs = &messages; //发送要读取的寄存器地址
messages.addr = addr; //i2c设备地址
messages.flags = I2C_M_RD; //读操作
messages.len = 1; //数据长度
messages.buf = val; //读取的数据保存在val ret = ioctl (fd, I2C_RDWR, (unsigned long)&packets); //发送数据帧
if (ret < 0)
ret = -1; return ret;
} /*
* fd : 文件描述符
* addr : i2c的设备地址
* reg : 寄存器地址
* val : 保存读取的数据
* len : 读取数据的长度
* 描述 :读取达到eeprom的末尾时,会读取最开头的字节
*/
int i2c_nbytes_read(int fd, unsigned char addr, unsigned char reg, unsigned char *val, int len)
{
int ret = 0;
unsigned char outbuf;
struct i2c_rdwr_ioctl_data packets;
struct i2c_msg messages[2]; /* 数据帧类型有2种
* 写要发送起始信号,进行写寄存器操作,再发送起始信号,进行读操作,
* 有2个起始信号,因此需要分开来操作。
*/
packets.nmsgs = 2;
//发送要读取的寄存器地址
messages[0].addr = addr;
messages[0].flags = 0; //write
messages[0].len = 1; //数据长度
messages[0].buf = &outbuf; //发送寄存器地址
outbuf = reg;
//读取数据
messages[1].len = len; //读取数据长度
messages[1].addr = addr; //设备地址
messages[1].flags = I2C_M_RD; //read
messages[1].buf = val; packets.msgs = messages; ret = ioctl(fd, I2C_RDWR, (unsigned long)&packets); //发送i2c,进行读取操作
if (ret < 0)
ret = -1; return ret;
}

Tony Liu

2016-9-23, Shenzhen

linux i2c 设备节点读写的更多相关文章

  1. linux spi 设备节点 读写

    本文记录spi设备节点的操作方法. SPI总线设备文件名通常为/dev/spidevN.P(N=0.1.2--,P=0.1.2--), 其中N表示第几路SPI总线,而P表示在该路SPI总线中使用哪个C ...

  2. Linux I2C设备驱动编写(二)

    在(一)中简述了Linux I2C子系统的三个主要成员i2c_adapter.i2c_driver.i2c_client.三者的关系也在上一节进行了描述.应该已经算是对Linux I2C子系统有了初步 ...

  3. 【转】Linux I2C设备驱动编写(二)

    原文网址:http://www.cnblogs.com/biglucky/p/4059582.html 在(一)中简述了Linux I2C子系统的三个主要成员i2c_adapter.i2c_drive ...

  4. Android和Linux下设备节点的创建笔记

    1. Linux kernel创建的/dev/下的设备节点是不对的, 其实是kernel仅负责在/sys/(基于内存的虚拟文件系统)创建一大堆下目录和文件,而真正的设备节点是在用户空间程序创建的,应该 ...

  5. Linux I2C设备驱动编写(三)-实例分析AM3359

    TI-AM3359 I2C适配器实例分析 I2C Spec简述 特性: 兼容飞利浦I2C 2.1版本规格 支持标准模式(100K bits/s)和快速模式(400K bits/s) 多路接收.发送模式 ...

  6. 【转】Linux I2C设备驱动编写(三)-实例分析AM3359

    原文网址:http://www.cnblogs.com/biglucky/p/4059586.html TI-AM3359 I2C适配器实例分析 I2C Spec简述 特性: 兼容飞利浦I2C 2.1 ...

  7. Linux I2C设备驱动

    i2c设备:ts.camera.audio.gsensor.e2prom I2C基本协议: 写:开始 -> 设备地址 -> 写标志 -> 应答 -> 内部地址 -> 应答 ...

  8. Linux I2C设备驱动编写(一)

    在Linux驱动中I2C系统中主要包含以下几个成员: I2C adapter 即I2C适配器 I2C driver 某个I2C设备的设备驱动,可以以driver理解. I2C client 某个I2C ...

  9. 【转】Linux I2C设备驱动编写(一)

    原文网址:http://www.cnblogs.com/biglucky/p/4059576.html 在Linux驱动中I2C系统中主要包含以下几个成员: I2C adapter 即I2C适配器 I ...

随机推荐

  1. 【BZOJ】1098: [POI2007]办公楼biu(补图+bfs+链表)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1098 显然答案是补图连通块..... 想到用并查集...可是连补图的边都已经...n^2了...怎么 ...

  2. 最大权闭合图 && 【BZOJ】1497: [NOI2006]最大获利

    http://www.lydsy.com/JudgeOnline/problem.php?id=1497 最大权闭合图详细请看胡伯涛论文<最小割模型在信息学竞赛中的应用>,我在这里截图它的 ...

  3. oracle系列--第六篇 Oracle上面小试牛刀

    现在我们可以在oracle上面进行创建表,向表中插入数据,修改表中数据,删除数据,甚至删除表等一系列操作. 即我们所说的CRUD操作. --create a table which name is t ...

  4. elasticsearch2.2 集群搭建各种坑

        目前生产环境的es版本是1.0版本,需要升级到最新的2.2版本,于是在测试环境进行部署集群测试,在测试过程中遇到的坑相当多,下面详细介绍下.       1. 版本升级到2.2后,必须建一个单 ...

  5. [转] - Linux网络编程 -- 网络知识介绍

    (一)Linux网络编程--网络知识介绍 Linux网络编程--网络知识介绍客户端和服务端         网络程序和普通的程序有一个最大的区别是网络程序是由两个部分组成的--客户端和服务器端. 客户 ...

  6. Shell下的正则表达式 (鸟哥私房菜)

    "Open Source" is a good mechanism to develop programs.$ apple is my favorite food.$ Footba ...

  7. 【iCore3双核心板】扩展引脚分布

    PDF 版下载: http://files.cnblogs.com/files/xiaomagee/iCore3%E6%89%A9%E5%B1%95%E5%BC%95%E8%84%9A%E5%88%8 ...

  8. 移动Web应用开发入门指南——视觉篇

    视觉篇 智能移动设备由于发展历史短,但更新速度快,从而导致移动设备的物理属性差异巨大,其中一部分物理属性影响视觉,另一部分影响到交互.兼容或性能.对人类来说,至少有80%以上的外界信息通过视觉获得,视 ...

  9. C++的ORM 开源框架

    C++的ORM 开源框架 介绍一个C++的ORM工具ODB SOCI.LiteSQL.POCO数据库访问类库对比

  10. A trip through the Graphics Pipeline 2011_04

    Welcome back. Last part was about vertex shaders, with some coverage of GPU shader units in general. ...