当前位置: 首页 > news >正文

Linux驱动开发笔记(四):设备驱动介绍、熟悉杂项设备驱动和ubuntu开发杂项设备Demo

若该文为原创文章,转载请注明原文出处
本文章博客地址:https://hpzwl.blog.csdn.net/article/details/134533533

红胖子网络科技博文大全:开发技术集合(包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软硬结合等等)持续更新中…

Linux系统移植和驱动开发专栏

上一篇:《Linux驱动开发笔记(三):基于ubuntu的helloworld驱动源码编写、makefile编写以及驱动编译加载流程测试》
下一篇:敬请期待…


前言

  驱动的开发需要先熟悉基本概念类型,本篇讲解linux杂项设备基础,还是基于虚拟机ubuntu去制作驱动,只需要虚拟机就可以尝试编写注册杂项设备的基本流程。


linux三大设备驱动

  • 字符设备:IO的传输过程是以字符为单位的,没有缓冲,比如I2C(SDA、SCL),SPI(MISO、MOSI、SCLK、CS)。
  • 块设备:IO的传输过程是以块为单位的,跟存储相关的都属于块设备,比如tf卡,sd卡。
  • 网络设备:IO的传输以socket套接字来访问的。

杂项设备

  • 杂项设备是属于字符设备,可以自动生成设备节点,设备节点位于/dev/目录下,是设备名称,如/dev/ttyS9等。
  • 主设备号相同,统一为10,次设备号不同,主设备相同可以节省内核资源。
    通过下列指令,可以查看系统杂项设备
cat /proc/misc

  在虚拟机上测试,查看杂项:
  在这里插入图片描述

  • 设备号分为主设备号和次设备号,主设备号是唯一的,次设备号不一定唯一。
    通过下列指令,可以查看系统主设备号:
cat /proc/devices

  在这里插入图片描述

杂项设备描述结构体

  ubuntu来说,自带的/usr/src下的就是内核的头文件。

cd /usr/src/linux-headers-4.18.0-15
vi include/linux/miscdevice.h

  定位到之前ubuntu自带的内核头文件下:
  在这里插入图片描述
  在这里插入图片描述

  查看到杂项设备的结构体:

struct miscdevice  {int minor;  // 次设备号const char *name;  // 设备节点名称(如/dev/ttyS8,则ttyS是名称)const struct file_operations *fops; // 文件操作集(非常重要)struct list_head list; struct device *parent;struct device *this_device;const struct attribute_group **groups; const char *nodename; umode_t mode;
};

  (注意:没打注释的,一般不管)

杂项设备文件操作集

cd /usr/src/linux-headers-4.18.0-15
vi include/linux/fs.h

  搜索到(vi则直接使用“/”):
  在这里插入图片描述

struct file_operations {struct module *owner;loff_t (*llseek) (struct file *, loff_t, int);ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);ssize_t (*read_iter) (struct kiocb *, struct iov_iter *);ssize_t (*write_iter) (struct kiocb *, struct iov_iter *);int (*iterate) (struct file *, struct dir_context *);int (*iterate_shared) (struct file *, struct dir_context *);__poll_t (*poll) (struct file *, struct poll_table_struct *);long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);long (*compat_ioctl) (struct file *, unsigned int, unsigned long);int (*mmap) (struct file *, struct vm_area_struct *);unsigned long mmap_supported_flags;int (*open) (struct inode *, struct file *);int (*flush) (struct file *, fl_owner_t id);int (*release) (struct inode *, struct file *);int (*fsync) (struct file *, loff_t, loff_t, int datasync);int (*fasync) (int, struct file *, int);int (*lock) (struct file *, int, struct file_lock *);ssize_t (*sendpage) (struct file *, struct page *, int, size_t, loff_t *, int);unsigned long (*get_unmapped_area)(struct file *, unsigned long, unsigned long, unsigned long, unsigned long);int (*check_flags)(int);int (*setfl)(struct file *, unsigned long);int (*flock) (struct file *, int, struct file_lock *);ssize_t (*splice_write)(struct pipe_inode_info *, struct file *, loff_t *, size_t, unsigned int);ssize_t (*splice_read)(struct file *, loff_t *, struct pipe_inode_info *, size_t, unsigned int);int (*setlease)(struct file *, long, struct file_lock **, void **);long (*fallocate)(struct file *file, int mode, loff_t offset,loff_t len);void (*show_fdinfo)(struct seq_file *m, struct file *f);
#ifndef CONFIG_MMUunsigned (*mmap_capabilities)(struct file *);
#endifssize_t (*copy_file_range)(struct file *, loff_t, struct file *,loff_t, size_t, unsigned int);int (*clone_file_range)(struct file *, loff_t, struct file *, loff_t,u64);ssize_t (*dedupe_file_range)(struct file *, u64, u64, struct file *,u64);
} __randomize_layout;

  例如read函数,那么就是打开驱动使用系统read,打开这个设备驱动的句柄,那么久会调用read函数,其他的以此类推,还比较好理解。
  以我们一个registerHelloWorld为例子,来简单说明。


驱动编写空模板准备

  首先复制之前的hello world的驱动,改个名字为:registerMiscDev:

cd ~/work/drive
cp -arf hellowolrd registerMiscDev

  在这里插入图片描述

cd registerMiscDev/
rm *.ko *.o *.order *.symvers

  这里删除起来麻烦,修改makefile,添加clean:
  在这里插入图片描述

  然后测试一下:
  在这里插入图片描述

  继续修改源码文件名称:

mv helloworld.c registerMiscDev.c

  修改完如下:
  在这里插入图片描述

  然后修改makefile里面的(obj-m模块名称改下),模板准备好了
  在这里插入图片描述

  下面基于registerMiscDev.c文件进行注册杂项设备,在修改.c文件:
  在这里插入图片描述

#include <linux/init.h>
#include <linux/module.h>static int registerMiscDev_init(void)
{ // 在内核里面无法使用基础c库printf,需要使用内核库printkprintk("Hello, I’m hongPangZi, registerMiscDev_init\n");	return 0;
}
static void registerMiscDev_exit(void)
{printk("bye-bye!!!\n");
}
MODULE_LICENSE("GPL");
module_init(registerMiscDev_init);module_exit(registerMiscDev_exit); 

杂项设备注册流程Demo

步骤一:填充miscdevice结构体

  在编写驱动的时候,代码中填充信息结构体。
  添加头文件miscdevice.h

#include <linux/miscdevice.h>
#include <linux/fs.h>

  在这里插入图片描述

  然后填充杂项设备结构体:
  在这里插入图片描述

  (注意:开始为“.”,结束为“,”,最后一行习惯加“,”了,这样可以全部统一复制啥的,省的加没加的)

struct miscdevice misc_dev {.minor = MISC_DYNAMIC_MINRO, // 这个宏是动态分配次设备号,避免冲突.name = "register_hongPangZi_misc,  // 设备节点名称.fops = misc_fops, // 这个变量记住,自己起的,步骤二使用
}

  在这里插入图片描述

步骤二:填充file_operations结构体

  在编写驱动的时候,代码中填充文件操作结构体。
  在这里插入图片描述

struct file_operations misc_fops {.owner = THIS_MODULE
}

  在这里插入图片描述

步骤三:注册杂项设备并生成设备节点

  注册到内核:

static int registerMiscDev_init(void)
{ // 在内核里面无法使用基础c库printf,需要使用内核库printkprintk("Hello, I’m hongPangZi, registerMiscDev_init\n");	int ret = 0;ret = misc_register(misc_dev);if(ret < 0){printk("Failed to misc_register(misc_dev)\n");	return -1;} return 0;
}

  在这里插入图片描述

  有注册就有注销:

static int registerMiscDev_init(void)
{ // 在内核里面无法使用基础c库printf,需要使用内核库printkprintk("Hello, I’m hongPangZi, registerMiscDev_init\n");	int ret = 0;ret = misc_register(&misc_dev);if(ret < 0){printk("Failed to misc_register(misc_dev)\n");	return -1;} return 0;
}

  在这里插入图片描述

  完整的文件源码:

#include <linux/init.h>
#include <linux/module.h>#include <linux/miscdevice.h>
#include <linux/fs.h>struct file_operations misc_fops = {.owner = THIS_MODULE,
};struct miscdevice misc_dev = {.minor = MISC_DYNAMIC_MINOR, // 这个宏是动态分配次设备号,避免冲突.name = "register_hongPangZi_misc", // 设备节点名称.fops = &misc_fops,  // 这个变量记住,自己起的,步骤二使用
};static int registerMiscDev_init(void)
{ // 在内核里面无法使用基础c库printf,需要使用内核库printkprintk("Hello, I’m hongPangZi, registerMiscDev_init\n");	int ret = 0;ret = misc_register(&misc_dev);if(ret < 0){printk("Failed to misc_register(&misc_dev)\n");	return -1;} return 0;
}static void registerMiscDev_exit(void)
{misc_deregister(&misc_dev);printk("bye-bye!!!\n");
}MODULE_LICENSE("GPL");
module_init(registerMiscDev_init);
module_exit(registerMiscDev_exit);

步骤四:编译make

make

  直接在驱动工程目录编译:
  在这里插入图片描述

  下面这个警告,实际上定义要在任何使用函数之前:
  在这里插入图片描述

  修改下:
  在这里插入图片描述

  在这里插入图片描述

  编译成功
  在这里插入图片描述

步骤五:加载卸载驱动测试

  将驱动拷贝到开发板或者目标系统,然后使用加载指令:

sudo insmod registerMiscDev.ko

  会打印入口加载的printk输出。
  在这里插入图片描述

  出现问题可能原因一是内核编译使用的编译器和模块使用的编译器版本不一致。ubuntu中printk终端打入内核日志消息了,可以使用dmesg进行查看:

dmesg

  在这里插入图片描述

  然后查看是否加入了杂项设备节点:
  在这里插入图片描述

  然后注销:

sudo rmmod registerMiscDev.ko

  在这里插入图片描述

  跟随着,结点消失了:
  在这里插入图片描述


入坑

入坑一:编译报错,结构体之后未加分号

问题

  编译错误,结构体后面加分号

解决

  加分号,脑袋有点蒙
  在这里插入图片描述

入坑二:编译错误,文件操作指针问题

问题

  在这里插入图片描述

解决

  这是写错了,是指针,需要加取地址&。


上一篇:《Linux驱动开发笔记(三):基于ubuntu的helloworld驱动源码编写、makefile编写以及驱动编译加载流程测试》
下一篇:敬请期待…


若该文为原创文章,转载请注明原文出处
本文章博客地址:https://hpzwl.blog.csdn.net/article/details/134533533

http://www.lryc.cn/news/240639.html

相关文章:

  • C/C++ 实现Windows注册表操作
  • nginx 模块相关配置及结构理解
  • 同时创建多个websoket(初始化多个连接、断开的重连、每个连接定时发消息、每个连接存储接收的数据(vuex或者pinia))
  • C语言——指针(一)
  • 学习量化交易如何入门?
  • CSS3动画
  • 黑马点评-10实现用户点赞和点赞排行榜功能
  • Spring配置其他注解Spring注解的解析原理
  • TypeScript 学习笔记 第一部分 语法基础
  • 【element优化经验】怎么让element-ui中表单多语言切换排版不乱
  • 软件设计中如何画各类图之一实体关系图(ER图):数据库设计与分析的核心工具
  • 【神印王座】龙皓晨美妆胜过月夜,魔神皇识破无视,撮合月夜阿宝
  • 汽车级全保护型六路半桥驱动器NCV7708FDWR2G 原理、参数及应用
  • 【小技巧】MyBatis 中 SQL 写法技巧小总结
  • C#编程题分享(4)
  • CTF靶场搭建及Web赛题制作与终端docker环境部署
  • nodejs express vue uniapp新闻发布系统源码
  • FastText模型文本分类
  • CentOS 7 使用Fmt库
  • 如何通过宝塔面板搭建一个本地MySQL数据库服务并实现远程访问
  • 普通话考试相关(一文读懂)
  • 深度学习动物识别 - 卷积神经网络 机器视觉 图像识别 计算机竞赛
  • 【Redisson】基于自定义注解的Redisson分布式锁实现
  • QT中样式表常见属性与颜色的设置与应用
  • OpenCvSharp从入门到实践-(02)图像处理的基本操作
  • Spring Boot 升级3.x 指南
  • 使用支付宝的沙箱环境在本地配置模拟支付并发布至公网调试
  • python-opencv划痕检测
  • 微服务学习|Gateway网关:网关作用、快速入门、路由断言工厂、路由过滤器配置、全局过滤器、过滤器执行顺序、跨域问题处理
  • 七、通过libfdk_aac编解码器实现aac音频和pcm的编解码