开干:
1、闲言碎语
这个驱动,越写觉的越简单,入门难,入门之后感觉还好。Linux开发还是比较友好的。
2、编写MPU6050带字符驱动的i2c从设备驱动
要实现的功能就是,将MPU6050作为字符驱动,在应用层,对其进行读数据。实现简单的功能。在前面的分析和实践中,可以看到实现字符驱动主要是实现file_operation中的方法,注册初始化cdev,让cdev和file_opration产生联系,字符驱动的初始化通过module_init来声明。实现i2c从设备驱动,主要是i2c_client和i2c_driver通过名字匹配,然后调用probe函数对设备进行初始化。那么,实现字符驱动的i2c从设备驱动,其实就是在i2c从设备驱动中添加字符驱动操作方法,从而在/dev中产生设备节点,让用户可以通过Linux
application的API对其进行操作。
本文实现了带字符驱动的i2c从设备驱动,然后编写了用户层测试程序,测试结果正确。下面对部分代码进行分析。
定义i2c_driver,file_operations,
static struct i2c_driver mpu6xxx_driver = {
.driver = {
.name = "mpu6xxx",
.owner = THIS_MODULE,
},
.class = I2C_CLASS_HWMON,
.id_table = mpu6xxx_ids,
.probe = mpu6xxx_probe,
.remove = mpu6xxx_remove,
};
主要实现其中的probe函数。
struct file_operations mpu6xxx_fops = {
owner : THIS_MODULE,
unlocked_ioctl : mpu6xxx_ioctl,
open : mpu6xxx_open,
release : mpu6xxx_release,
};
主要实现其中的ioctl函数。
probe函数实现:
static int mpu6xxx_probe(struct i2c_client *client, const struct i2c_device_id *id){
u16 version;
int result,retval = -1;
struct mpu6xxx_data *mpu6xxx;
dev_t dev;
printk(KERN_DEBUG "mpu6xxx driver probe... \n");
if(!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)){
retval = -EINVAL;
goto failed;
} //核查i2c驱动
mpu6xxx = kzalloc(sizeof(struct mpu6xxx_data),GFP_KERNEL);
//分配用户数据
if(!mpu6xxx)
{
retval = -ENOMEM;
goto failed;
}
memset(&mpu6xxx->gyro,0,sizeof(struct sub_sensor));
memset(&mpu6xxx->accel,0,sizeof(struct sub_sensor));
mutex_init(&mpu6xxx->lock);
mpu6xxx->client = client;
i2c_set_clientdata(client,mpu6xxx); //用来在驱动中获取数据
this_client = client;
version = i2c_smbus_read_byte_data(client,MPU6050_REG_WHO_AM_I);
if(version != 0x70)
{
printk("error retval %d , version %.2x \n",retval , version);
retval = -2;
goto failed;
}
//检查version
result = alloc_chrdev_region(&dev,this_major,1,"mpu6xxx");
if(result < 0)
{
retval = result;
goto failed;
}
this_major = MAJOR(dev);
cdev_init(&mpu6xxx->cdev,&mpu6xxx_fops);
mpu6xxx->cdev.owner = THIS_MODULE;
result = cdev_add(&mpu6xxx->cdev,dev,1);
if(result)
{
retval = result;
goto failed;
}
mpu_cls = class_create(THIS_MODULE,"mpu6xxx");
if(IS_ERR(mpu_cls))
{
retval = -3;
goto failed;
}
mpu_device = device_create(mpu_cls,NULL,dev,NULL,"mpu6xxx");
if(IS_ERR(mpu_device))
{
class_destroy(mpu_cls);
}
//注册cdev,以及在/dev中产生节点
mpu6xxx_reset(mpu6xxx);
mpu6xxx_disable(mpu6xxx);
retval = 0;
printk(KERN_DEBUG "mpuxxx probe succeed...\n");
return retval;
failed:
return retval;
};
ioctl实现:
static int mpu6xxx_ioctl(struct file *file, unsigned int cmd,unsigned long arg)
{
int re = -1;
void __user *argp = (void __user *)arg;
struct mpu6xxx_data *mpu6xxx = i2c_get_clientdata(this_client);
printk("ioctl ... cmd %d mpucmd %d BUFFERSIZE %d\n",cmd,MPU_IOCTL_GETGYRO,BUFFERSIZE);
switch(cmd)
{
case MPU_IOCTL_GETGYRO:
mutex_lock(&(mpu6xxx->lock) ); //通过互斥操作来避免频繁读写
re = mpu6xxx_read_data(mpu6xxx,0);
if(re != 0) return re;
printk("x : %d \n", mpu6xxx->gyro.x.value);
if(copy_to_user(argp,&mpu6xxx->gyro,sizeof(struct sub_sensor))) //将数据拷贝到用户空间
{
printk("failed copy data .. \n");
mutex_unlock(&mpu6xxx->lock);
return -EFAULT;
}
printk("ioctl succed...\n");
mutex_unlock(&mpu6xxx->lock);
break ;
case MPU_IOCTL_GETACCEL:
mutex_lock(&(mpu6xxx->lock) );
re = mpu6xxx_read_data(mpu6xxx,1);
if(re != 0) return re;
if(copy_to_user(argp,&mpu6xxx->accel,sizeof(struct sub_sensor)))
{
mutex_unlock(&(mpu6xxx->lock) );
printk("failed copy data .. \n");
return -EFAULT;
}
mutex_unlock(&(mpu6xxx->lock) );
break;
default: break;
}
return 0;
}
驱动加载结果:
应用层代码结果:
在动6050时,数据会变动。并且数据大小正常。
- MPU6050带字符驱动的i2c从设备驱动2
#include <linux/kernel.h> #include <linux/module.h> #include <linux/init.h> #inclu ...
- 【Linux高级驱动】linux设备驱动模型之平台设备驱动机制
[1:引言: linux字符设备驱动的基本编程流程] 1.实现模块加载函数 a.申请主设备号 register_chrdev(major,name,file_operations); b.创 ...
- Linux中总线设备驱动模型及平台设备驱动实例
本文将简要地介绍Linux总线设备驱动模型及其实现方式,并不会过多地涉及其在内核中的具体实现,最后,本文将会以平台总线为例介绍设备和驱动程序的实现过程. 目录: 一.总线设备驱动模型总体介绍及其实现方 ...
- 【linux驱动分析】misc设备驱动
misc设备驱动.又称混杂设备驱动. misc设备驱动共享一个设备驱动号MISC_MAJOR.它在include\linux\major.h中定义: #define MISC_MAJO ...
- Linux驱动之I2C总线设备以及驱动
[ 导读] 本文通过阅读内核代码,来梳理一下I2C子系统的整体视图.在开发I2C设备驱动程序时,往往缺乏对于系统整体的认识,导致没有一个清晰的思路.所以从高层级来分析一下I2C系统的设计思路,将有助于 ...
- Linux 驱动框架---cdev字符设备驱动和misc杂项设备驱动
字符设备 Linux中设备常见分类是字符设备,块设备.网络设备,其中字符设备也是Linux驱动中最常用的设备类型.因此开发Linux设备驱动肯定是要先学习一下字符设备的抽象的.在内核中使用struct ...
- Linux设备驱动编程之复杂设备驱动
这里所说的复杂设备驱动涉及到PCI.USB.网络设备.块设备等(严格意义而言,这些设备在概念上并不并列,例如与块设备并列的是字符设备,而PCI.USB设备等都可能属于字符设备),这些设备的驱动中又涉及 ...
- Linux驱动编写(块设备驱动代码)
[ 声明:版权所有,欢迎转载,请勿用于商业用途. 联系信箱:feixiaoxing @163.com] 按照ldd的说法,linux的设备驱动包括了char,block,net三种设备.char设备 ...
- Linux gadget驱动分析3------复合设备驱动
windows上面对usb复合设备的识别需要下面条件. “ 如果设备满足下列要求,则总线驱动程序还会报告 USB\COMPOSITE 的兼容标识符: 设备描述符的设备类字段 (bDeviceClass ...
随机推荐
- AE视频制作 参考 资源 科幻科技风格 开场 公司企业宣传
== 人类已经堕落,机器军团大举入侵,开始抹除人类,净化地球!-原创-高清正版视频在线观看–爱奇艺 这个视频片头不错.展示效果和音乐. == 外星人入侵地球,只为收集人类大脑!速看科幻电影<天际 ...
- 潭州课堂25班:Ph201805201 django 项目 第二课 git 版本控制 (课堂笔记)
安装 git sudo apt-get install git 查看版本信息: git --version 演示: 创建个项目 创建文件夹 如果要对这个文件夹进行版本控制 先进到这个文件夹中, 命令查 ...
- React-Native-Storage使用介绍
react-native-storage 这是一个本地持久存储的封装,可以同时支持react-native(AsyncStorage)和浏览器(localStorage).ES6语法,promise异 ...
- SSD固态硬盘测试工具收集(持续更新)
https://www.crsky.com/zhuanti/gutaiyingpanceshi.html https://www.crsky.com/zhuanti/ssdjiance.html ht ...
- 旋转矩阵(Rotation Matrix)的推导及其应用
向量的平移,比较简单. 缩放也较为简单 矩阵如何进行计算呢?之前的文章中有简介一种方法,把行旋转一下,然后与右侧对应相乘.在谷歌图片搜索旋转矩阵时,看到这张动图,觉得表述的很清晰了. 稍微复杂一点的是 ...
- 调用 LoadLibraryEx 失败,在 ISAPI 筛选器 "C:\Windows\Microsoft.NET\Framework\v4.0.30319\\aspnet_filter.dll" 上
开始 -> 运行 -> inetmgr -> 应用程序池 -> 找到 我的网站对象的 程序池 -> 右键 -> 高级设置 -> 启用32位应用程序 由 fal ...
- javap 指令集
栈和局部变量操作将常量压入栈的指令aconst_null 将null对象引用压入栈iconst_m1 将int类型常量-1压入栈iconst_0 将int类型常量0压入栈iconst_1 将int类型 ...
- Node.js模板引擎的深入探讨
每次当我想用 node.js 来写一个 web 相关项目的时候.我总是会陷入无比的纠结.原因是 JavaScript 生态圈里的模板引擎实在太多了,但那么多却实在找不出一个接近完美的,所谓完美的概念就 ...
- Deep Learning.ai学习笔记_第一门课_神经网络和深度学习
目录 前言 第一周(深度学习引言) 第二周(神经网络的编程基础) 第三周(浅层神经网络) 第四周(深层神经网络) 前言 目标: 掌握神经网络的基本概念, 学习如何建立神经网络(包含一个深度神经网络), ...
- 每天一个linux命令(8):rm
1.命令简介 rm(Remove file 删除目录或文件)删除文件,对于链接文件,只是删除整个链接文件,而原有文件保持不变. 2.用法 rm [选项]... 文件.. 3.选项 -f, –force ...