【linux】i2c使用分析&源码实战
前言
- 目前不涉及驱动源码
- 原文
1. 设备检查命令
1.1 查看I2C驱动
- 命令:
ls /sys/bus/i2c/devices用于查看系统上存在的 I2C 总线
1.2 i2c-tools
- i2c-tools,安装 i2c-tools 方便调试 i2c设备
1.2.1 I2C-detect安装
- 使用命令:
sudo apt install i2c-tools -y安装 i2c-tools - 安装后可以使用命令:i2cdetect、i2cdump、i2cset 和 i2cget
1.2.2 i2cdetect 命令
- i2cdetect
- 用于扫描 I2C 总线上的设备
- 语法
i2cdetect [-y] [-a] [-q|-r] i2cbus [first last]- 参数
- y:关闭交互模式,使用该参数时,不会提示警告信息。
- a:扫描总线上的所有设备
- q:使用SMBus的“quick write”命令进行检测,不建议使用该参数
- r:使用SMBus的“receive byte”命令进行检测,不建议使用该参数
- i2cbus:指定i2c总线的编号
- first、last:扫描的地址范围
- 返回值
- '-':表示该地址被检测,但是没有芯片应答
- 'UU':表示该地址当前由内核驱动程序使用
- '**':** 表示以16进制表示的设备地址编号,如“68”
- 例子:
i2cdetect -a 0- i2cdetect:i2cdetect命令
- -a:总线上所有设备
- 0:标号为 0 的 I2C,即是 I2C 1。

- 上图中扫描出存在设备地址为 0x1e 和 0x68 的设备。
i2cdetect -F i2cbus:查询 i2c 总线的功能,参数 i2cbus 表示 i2c 总线(看上)i2cdetect -V:打印软件的版本号i2cdetect -l:检测当前系统有几组 i2c 总线
1.2.3 i2cget 命令
- i2cget
- 用于读取 I2C 设备的某个寄存器的值
- 语法
i2cget [-f] [-y] i2cbus chip-address [data-address [mode]]- 参数
- f:强制访问
- y:关闭交互模式,使用该参数时,不会提示警告信息
- i2cbus:指定 I2C 总线的编号
- chip-address:I2C 设备地址
- data-address:I2C 寄存器地址
- mode:指定读取的大小, 可以是b, w, s或i,分别对应了字节,字,SMBus块, I2C块
1.2.4 i2cset 命令
- i2cset
- 写入指定 I2C 设备的某个寄存器的值
- 语法
i2cset [-f] [-y] [-m mask] [-r] i2cbus chip-address data-address [value] … [mode]- 参数
- f:强制访问
- y:关闭交互模式,使用该参数时,不会提示警告信息
- m:
- r:写入后立即回读寄存器值,并将结果与写入的值进行比较
- i2cbus:指定 I2C 总线的编号
- chip-address:I2C 设备地址
- data-address:I2C 寄存器地址
- value:要写入的值
- mode:指定读取的大小, 可以是b, w, s或i,分别对应了字节,字,SMBus块, I2C块
1.2.5 i2cdump 命令
- i2cdump
- 读取指定设备的全部寄存器的值
- 语法
i2cdump [-f] [-r first-last] [-y] i2cbus address [mode [bank [bankreg]]]- 参数
- r:指定寄存器范围,只能扫描从 first 到 last 区域
- f:强制访问设备
- y:关闭人机交互模式
- i2cbus:指定 I2C 总线的编号
- address:指定设备地址
- mode:指定读取的大小, 可以是b, w, s或i,分别对应了字节,字,SMBus块, I2C块
- 例子
i2cdump -V:打印软件的版本号
2. 源码实战
- 采用MPU6050设备进行实验
- 步骤:
- 先编写基础的 I2C 基础函数
- 编写 MPU6050 初始化函数和关闭设备文件函数
- 编写获取 MPU6050 数据函数
- 编写业务函数
2.1 编写 bsp_mpu6050.h 文件
- 编写好 MPU6050 需要的宏
- extern 外部函数
/** @file bsp_mpu6050.h
* @brief 简要说明
* @details 详细说明
* @author lzm
* @date 2020-11-28 19:22:20
* @version v1.0
* @copyright Copyright By lizhuming, All Rights Reserved
*
**********************************************************
* @LOG 修改日志:
**********************************************************
*/
#ifndef _BSP_MPU6050_H_
#define _BSP_MPU6050_H_
/* 宏 */
#define SMPLRT_DIV 0x19
#define CONFIG 0x1A
#define GYRO_CONFIG 0x1B
#define ACCEL_CONFIG 0x1C
#define ACCEL_XOUT_H 0x3B
#define ACCEL_XOUT_L 0x3C
#define ACCEL_YOUT_H 0x3D
#define ACCEL_YOUT_L 0x3f
#define ACCEL_ZOUT_H 0x3F
#define ACCEL_ZOUT_L 0x40
#define TEMP_OUT_H 0x41
#define TEMP_OUT_L 0x42
#define GYRO_XOUT_H 0x43
#define GYRO_XOUT_L 0x44
#define GYRO_YOUT_H 0x45
#define GYRO_YOUT_L 0x46
#define GYRO_ZOUT_H 0x47
#define GYRO_ZOUT_L 0x48
#define PWR_MGMT_1 0x6B
#define WHO_AM_I 0x75
#define SlaveAddress 0xD0
#define Address 0x68 //MPU6050地址
#define I2C_RETRIES 0x0701
#define I2C_TIMEOUT 0x0702
#define I2C_SLAVE 0x0703 //IIC从器件的地址设置
#define I2C_BUS_MODE 0x0780
/* 类型转换 */
typedef unsigned char uint8_t;
/* 函数 */
uint8_t mpu6050_init(char * i2cDev);
void mpu6050_close(void);
short getData(unsigned char regAddr);
#endif /* #define _BSP_MPU6050_H_ */
2.2 编写 bsp_mpu6050.c 文件
- bsp_mpu6050.c
- 编写 I2C 读写函数
- 编写 MPU6050 设备初始化函数及关闭文件函数
- 编写获取 MPU6050 设备寄存器数据函数
/** @file bsp_mpu6050.c
* @brief 简要说明
* @details 详细说明
* @author lzm
* @date 2020-11-28 19:20:20
* @version v1.0
* @copyright Copyright By lizhuming, All Rights Reserved
*
**********************************************************
* @LOG 修改日志:
**********************************************************
*/
/* 头文件 */
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/ioctl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<sys/select.h>
#include<sys/time.h>
#include "bsp_mpu6050.h"
/* 设备句柄 */
int i2cFd; // i2c设备句柄
/**
* @brief i2c 写
* @param fd:i2c设备句柄
* @param regAddr:寄存器地址
* @param val:需要写入的值
* @retval 0:写入成功
* @retval -1:写入失败
* @author lzm
*/
static uint8_t i2c_write(int fd, uint8_t regAddr, uint8_t val)
{
int cnt; // 写入失败后,重复写入的次数
uint8_t data[2]; // data[0]为寄存器地址,data[1]为需要写入的值
data[0] = regAddr;
data[1] = val;
for(cnt=5; cnt>0; cnt--)
{
if(write(fd, data, 2) == 2)
return 0; // 写入成功
}
return -1; // 写入失败
}
/**
* @brief i2c 读
* @param fd:i2c设备句柄
* @param regAddr:寄存器地址
* @param val:读取到数据保存的地方
* @retval 0:读取成功
* @retval -1:读取失败
* @author lzm
*/
static uint8_t i2c_read(int fd, uint8_t regAddr, uint8_t * val)
{
int cnt; // 读取失败后,重新读取的次数
for(cnt=5; cnt>0; cnt--)
{
if(write(fd, ®Addr, 1) == 1)
{
if(read(fd, val, 1) == 1)
return 0;
}
}
return -1;
}
/**
* @brief mpu6050初始化
* @param i2cDev
* @retval 1:初始化成功
* @retval 0:初始化失败
* @author lzm
*/
uint8_t mpu6050_init(char * i2cDev)
{
i2cFd = open(i2cDev, O_RDWR); // 打开i2c设备文件
if(i2cFd < 0)
{
printf("Can't open %s!\n", i2cDev);
exit(1);
}
printf("Open %s success!", i2cDev);
if(ioctl(i2cFd, I2C_SLAVE, Address) < 0)
{
printf("fail to set i2c device slave address!");
close(i2cFd); // 关闭i2c设备文件
return -1;
}
printf("set slave address to 0x%x success!", Address);
i2c_write(i2cFd, PWR_MGMT_1, 0X00);
i2c_write(i2cFd, SMPLRT_DIV, 0X07);
i2c_write(i2cFd, CONFIG, 0X06);
i2c_write(i2cFd, ACCEL_CONFIG, 0X01);
return 1;
}
/**
* @brief 关闭设备文件
* @param
* @retval
* @author lzm
*/
void mpu6050_close(void)
{
close(i2cFd);
}
/**
* @brief 获取数据
* @param regAddr:寄存器地址
* @retval 获取到的数据
* @author lzm
*/
short getData(unsigned char regAddr)
{
char chH; // 高字节
char chL; // 低字节
i2c_read(i2cFd, regAddr, &chH);
usleep(1000);
i2c_read(i2cFd, regAddr, &chL);
return ((chH << 8) + chL);
}
2.3 编写 main.c 文件
- 编写业务函数
/** @file main.c
* @brief 简要说明
* @details 详细说明
* @author lzm
* @date 2020-11-28 19:18:20
* @version v1.0
* @copyright Copyright By lizhuming, All Rights Reserved
*
**********************************************************
* @LOG 修改日志:
**********************************************************
*/
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/ioctl.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<sys/select.h>
#include<sys/time.h>
#include "mpu6050/bsp_mpu6050.h"
int main(int argc, char * argv[])
{
/* 程序开始标语 */
printf("This is a mpu6050 test!\n");
/* 初始化 MCPU6050 */
mpu6050_init("/dev/i2c-0");
sleep(1);
/* mpu6050 应用测试 */
while(1)
{
printf("get mpu6050 data!\n");
usleep(1000 * 100);
printf("ACCE_X:%6d\n ", getData(ACCEL_XOUT_H));
usleep(1000 * 100);
printf("ACCE_Y:%6d\n ", getData(ACCEL_YOUT_H));
usleep(1000 * 100);
printf("ACCE_Z:%6d\n ", getData(ACCEL_ZOUT_H));
usleep(1000 * 100);
printf("GYRO_X:%6d\n ", getData(GYRO_XOUT_H));
usleep(1000 * 100);
printf("GYRO_Y:%6d\n ", getData(GYRO_YOUT_H));
usleep(1000 * 100);
printf("GYRO_Z:%6d\n\n ", getData(GYRO_ZOUT_H));
sleep(1);
}
/* 退出 mpu6050 */
mpu6050_close();
}
相关链接
【linux】i2c使用分析&源码实战的更多相关文章
- 【linux】系统调用版串口分析&源码实战
目录 前言 参考 1. 实战分析 1.1 开发步骤 1.1.1 获取串口设备路径 1.1.2 打开设备文件 1.1.3 配置串口 termios 结构体 1. c_iflag 输入模式标志 2. c_ ...
- 【芯片手册开发】Sil9136音频开发详细分析+源码实战
目录 前言 参考 手册使用+实战 配置 Configuring Audio Using I2S 总结实现 前言 默认在开发了视频方面后 这方面的工作本来可以找技术支持拿个例程参考下,很快就可以的写出来 ...
- Linux内核分析(一)---linux体系简介|内核源码简介|内核配置编译安装
原文:Linux内核分析(一)---linux体系简介|内核源码简介|内核配置编译安装 Linux内核分析(一) 从本篇博文开始我将对linux内核进行学习和分析,整个过程必将十分艰辛,但我会坚持到底 ...
- Linux 内核调度器源码分析 - 初始化
导语 上篇系列文 混部之殇-论云原生资源隔离技术之CPU隔离(一) 介绍了云原生混部场景中CPU资源隔离核心技术:内核调度器,本系列文章<Linux内核调度器源码分析>将从源码的角度剖析内 ...
- Netty4.x 源码实战系列(一): 深入理解ServerBootstrap 与 Bootstrap
转载自:https://www.cnblogs.com/itdriver/p/8149913.html 从Java1.4开始, Java引入了non-blocking IO,简称NIO.NIO与传统s ...
- 转载~Linux 平台下阅读源码的工具
Linux 平台下阅读源码的工具 前言 看源代码是一个程序员必须经历的事情,也是可以提升能力的一个捷径.个人认为: 要完全掌握一个软件的方法只有阅读源码在Windows下有sourceinsight这 ...
- Linux 平台下阅读源码的工具链
原文:http://blog.jobbole.com/101322/ 前言 看源代码是一个程序员必须经历的事情,也是可以提升能力的一个捷径.个人认为: 要完全掌握一个软件的方法只有阅读源码. 在Win ...
- 鸿蒙内核源码分析(源码注释篇) | 鸿蒙必定成功,也必然成功 | 百篇博客分析OpenHarmony源码 | v13.02
百篇博客系列篇.本篇为: v13.xx 鸿蒙内核源码分析(源码注释篇) | 鸿蒙必定成功,也必然成功 | 51.c.h .o 几点说明 kernel_liteos_a_note | 中文注解鸿蒙内核 ...
- Netty4.x 源码实战系列(一): 深入理解ServerBootstrap 与 Bootstrap (1)
从Java1.4开始, Java引入了non-blocking IO,简称NIO.NIO与传统socket最大的不同就是引入了Channel和多路复用selector的概念.传统的socket是基于s ...
随机推荐
- 学习写简单Spring源码demo
最近在研究怎么实现简单的Spring的源码,通过注解的方式来实现对bean的加载管理. 首先先来看下我的工程结构: (1)spring-common:定义了常用的枚举常量,工具类(如FileUtils ...
- hello world .net core 微服务框架 Viper
1.Viper是什么? Viper 是.NET平台下的Anno微服务框架的一个示例项目.入门简单.安全.稳定.高可用.全平台可监控.底层通讯可以随意切换thrift grpc. 自带服务发现.调用链追 ...
- 团灭 LeetCode 股票买卖问题
很多读者抱怨 LeetCode 的股票系列问题奇技淫巧太多,如果面试真的遇到这类问题,基本不会想到那些巧妙的办法,怎么办?所以本文拒绝奇技淫巧,而是稳扎稳打,只用一种通用方法解决所用问题,以不变应万变 ...
- JavaMail 发送邮件出现 Connection reset 问题
问题描述 使用 java mail 发送邮件的时候,申请的 163 邮箱作为发件箱,然无论如何配置,均出现 Connection reset,无法正常发送邮件. Exception in thread ...
- 知识补充之Django缓存
缓存 简单概括就是将对数据库操作查询所得到的数据放入另外一台机器上(缓存)中,当用户再次请求时,直接去缓存中拿,避免对数据库的频繁操作,加快数据的显示时间,需要知道的是,缓存里面的数据一般都设置有超时 ...
- https中引入http资源资源所导致的问题
问题描述 因为公司要求所有生产环境为了安全性需求,全部都走https, 并且在Nginx里面加入了Content-Security-Policy "upgrade-insecure-requ ...
- F1分数
分类的常用指标有: accuracy:准确率 recall:召回率 precison:精确率 f1score:f1分数,是recall和precison的调和均值. 准确率什么情况下失效? 在正负样本 ...
- SpringBoot第九集:整合JSP和模板引擎Freemarker/Thymeleaf(2020最新最易懂)
SpringBoot第九集:整合JSP和模板引擎(2020最新最易懂) 当客户通过前端页面提交请求后,我们以前是怎么做的?后端接收请求数据,处理请求,把响应结果交给模板引擎JSP,最后将渲染后的JSP ...
- SSM工作流程与原理详解
自学ssm->springboot->springcloud,所以很多东西会用但理解较浅,所以现在从最开始的ssm开始进行对原理以及运行过程的整理知识归纳,若有错误感谢指正. Spring ...
- python像matlab类似的符号函数绘图
matplotlib 绘图之前需要先定义数据范围,python 有一个 sympy 包,类似 matlab 里面的符号函数,可以用来进行二维和三维的图像绘制, from sympy.plotting ...