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

Linux驱动开发(一)

linux驱动学习记录

一、背景

在开始学习我的linux驱动之旅之前,先提一下题外话,我是一个c语言应用层开发工作人员,在工作当中往往会和硬件直接进行数据的交互,往往遇到数据不通的情况,常常难以定位,而恰巧驱动是处于应用层和硬件之间的一个重要的数据传输媒介,也是在这个背景下,驱动了我对驱动的好奇心,当抱着了解的心态来的时候,却不知不觉地对驱动的执行方式产生了兴趣,由此也开始了我学习的征途…

二、驱动开发

驱动的作用是用来操作硬件工作,使其应用开发人员能间接实现对功能的开发。驱动函数都是注册在内核当中,应用层代码也都是通过直接调用系统的函数来实现对驱动的数据传递和接收,实现流程如下图所示:
在这里插入图片描述
1、驱动环境搭建(内核编译的方式)

  1. 从官网获取,将获取的压缩包解压后放入/usr/src中
  2. 使用apt指令获取
sudo apt-cache search linux-source
sudo apt-get install linux-source-<version> 
sudo tar jxvf linux-source-<version>.tar.bz2
  1. 编译源码
 cd /usr/src/linux-source-<version>
#以下三选一
sudo make config #逐一选择kernel的配置形式
#或
sudo make oldconfig #按原有kernel配置配置kernel
#或
sudo make menuconfig #使用可视化环境配置kernel#编译内核的过程
sudo make #编译内核,时间较长,可能会长达1小时及以上,如5.4.0用时约3-5个小时
#此时已完成了镜像编译和所有modules编译,如果不确定可以使用以下指令再编译
sudo make modules #编译内核驱动modules
sudo make bzImage #编译内核镜像#加载modules的过程
sudo make modules_install #安装内核模块
#检查安装
cd /lib/modules 
#如果安装正常这里面会出现以版本号为文件名的文件夹,就证明安装成功了,如5.4.xx或5.4.xx-xx-generic

2、驱动加载和卸载的指令

  • lsmod :list moduel 把我们机器上所有的驱动打印出来,
  • insmod:安装驱动
  • rmmod:删除驱动
  • modinfo:打印驱动信息

3、最简单的驱动开始示例

#include <linux/module.h>        // module_init  module_exit
#include <linux/init.h>            // __init   __exit// 模块安装函数
static int __init chrdev_init(void)
{    printk(KERN_INFO "chrdev_init helloworld init\n");return 0;
}// 模块下载函数
static void __exit chrdev_exit(void)
{printk(KERN_INFO "chrdev_exit helloworld exit\n");
}module_init(chrdev_init);
module_exit(chrdev_exit);// MODULE_xxx这种宏作用是用来添加模块描述信息
MODULE_LICENSE("GPL");                // 描述模块的许可证

注:
__init宏告知编译器,将变量或函数放在一个特殊的区域,这个区域定义在vmlinux.lds中。__init将函数放在".init.text"这个代码区中,__initdata将数据放在".init.data"这个数据区中。

标记为初始化的函数,表明该函数供在初始化期间使用。在模块装载之后,模块装载就会将初始化函数扔掉。这样可以将该函数占用的内存释放出来。

__exit宏告知编译器,将函数放在".exit.text"这个区域中。__exitdata宏则告知编译器将数据放在".exit.data"这个区域中。
exit.*区域仅仅对于模块是有用的:如果编译稳定的话,exit函数将永远不会被调用。只有当模块支持无效的时候,exit.*区域将被丢弃。这就是为什么定义中会出现ifdef。

printk是内核的日志函数,日志级别有八个0-7,通过读写/proc/sys/kernel/printk文件可以读取、修改控制台的日志级别,数字越小,级别越高,其实printk始终能输出信息,只不过是在
/var/log/messages文件中,可以通过dmesg来进行查看。

#define KERN_EMERG 0   
#define KERN_ALERT 1   
#define KERN_CRIT 2   
#define KERN_ERR 3   
#define KERN_WARNING 4   
#define KERN_NOTICE 5   
#define KERN_INFO 6   
#define KERN_DEBUG 7

4、驱动编译Makefile的编写

前提: 一定要把源码树目录中主Makefile中ARCH、cross_compile变量的值更改了;

#ubuntu的内核源码树,如果要编译在ubuntu中安装的模块就打开这2个
#KERN_VER = $(shell uname -r)
#KERN_DIR = /lib/modules/$(KERN_VER)/build# 开发板的linux内核的源码树目录
KERN_DIR = /root/driver/kernelobj-m    += module_test.oall:
make -C $(KERN_DIR) M=`pwd` modulescp:
cp *.ko /root/porting_x210/rootfs/rootfs/driver_test.PHONY: clean    
clean:
make -C $(KERN_DIR) M=`pwd` modules clean

注:

make -C $(KERN_DIR) M=`PWD` modules

这句话代码的作用就是到 KERN_DIR这个文件夹中 make modules

把当前目录赋值给M,M作为参数传到主目录的Makefile中,实际上是主目录的makefile中有目标modules,下面有一定的规则来编译驱动;

#KERN_VER = $(shell uname -r)#KERN_DIR = /lib/modules/$(KERN_VER)/build

我们在ubuntu中编译内核的时候用这两句代码,因为在ubuntu中为我们保留了一份linux内核的源码树,我们编译的时候直接调用那个源码树的主Makefile以及一些头文件、内核函数等;

了解规则以后,我们设置好KERN_DIR、obj-m这两个变量以后直接make就可以了;
在这里插入图片描述
这里注意vermagic 这个的1.8.0-41是你用的linux内核源码树的版本号,只有这个编译的版本号与运行的linux内核版本一致的时候,驱动程序才会被安装

注意license:GPL linux内核开元项目的许可证一般都是GPL这里尽量设置为GPL,否则有些情况下会出现错误;

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

相关文章:

  • Spring MVC 之返回数据(静态页面、非静态页面、JSON对象、请求转发与请求重定向)
  • leetcode-每日一题-2335(简单,贪心)
  • Verilog语法之数学函数
  • 【手撕面试题】JavaScript(高频知识点一)
  • 如何用PHP实现消息推送
  • 电子学会2020年6月青少年软件编程(图形化)等级考试试卷(四级)答案解析
  • DaVinci:调色版本
  • 【C++初阶】十二、STL---反向迭代器的实现
  • day 43|● 1049. 最后一块石头的重量 II ● 494. 目标和 ● 474.一和零
  • [SSD固态硬盘技术 0] SSD的结构和原理导论
  • Vue (3)
  • SQL语句,常用的DDL表操作语句
  • C 语言 宏定义 :字符串化 stringify 的应用
  • 代替swagger的api接口神器
  • 2月12日,30秒知全网,精选7个热点
  • HTML img和video object-fit 属性
  • Pascal版本的 - freopen
  • STM32单片机OLED显示
  • 备战金三银四,软件测试面试题(全)
  • 硬件篇-配置
  • 网页内容 中文乱码 解决办法
  • 【C++之容器篇】造轮子:模拟实现vector类
  • C++中的右值引用与移动构造函数
  • Swift如何使用依赖注入进行解藕
  • 合宙ESP32S3-CORE开发板|保姆级|Arduino IDE|windows11|esp32S3支持库|helloword例程:Arduino 环境搭建
  • CMake中target_precompile_headers的使用
  • SpringCloud和微服务介绍
  • Qt源码编译过程中配置文件中的选项说明
  • Mysql 增删改查(一) —— 查询(条件查询where、分页limits、排序order by、分组 group by)
  • VScode 结合clangd 构建linux源代码阅读环境