ARM 裸机与 Linux 驱动对比及 Linux 内核入门

目录

ARM裸机代码和驱动的区别

Linux系统组成

内核五大功能

设备驱动分类

内核类型

驱动模块

驱动模块示例

Makefile配置

命令

编码辅助工具

内核中的打印函数

printk 函数

修改打印级别

​编辑

打印级别含义

 驱动多文件编译

示例

 模块传递参数

命令行传递参数

支持的数据类型

module_param 函数

MODULE_PARM_DESC 函数

示例

module_param_array 函数

字符设备驱动

Linux系统中一切皆文件

字符设备驱动步骤

字符设备驱动的注册

​编辑

字符设备驱动的注销


以下是本人学习时的一些笔记,对初入门的驱动可能会有一些帮助,希望可以帮到大家~

ARM裸机代码和驱动的区别

共同点

  • 都能够操作硬件。

不同点

  • 裸机编程是直接用C语言给寄存器写值。
  • 驱动编程遵循一定的框架和规范,通过往寄存器写值来控制硬件。
  • 裸机代码独立编译和执行,而驱动依赖于内核进行编译和执行。
  • 裸机程序一次只能执行一个任务,而驱动可以支持并发执行多个任务。
  • 裸机程序只需一个main函数即可,而驱动需要遵循内核的框架和流程。

Linux系统组成

  • 用户空间(0-3G):每个进程独占0-3G的虚拟地址空间。

  • 内核空间(3-4G):所有进程共享3-4G的虚拟地址空间。
  • 系统调用:应用程序通过系统调用(软中断SWI)与内核交互。

内核五大功能

  • 进程管理:负责进程的创建、销毁和调度。
  • 文件管理:通过文件系统(如ext2/ext3/ext4/YAFFS/JFFS等)来管理文件。
  • 网络管理:通过网络协议栈(如OSI/TCP/IP)处理数据包的封装和拆解。
  • 内存管理:负责用户空间和内核空间内存的分配和回收。
  • 设备管理:管理设备驱动,如字符设备、块设备和网络设备。

设备驱动分类

  • 字符设备驱动:如LED、鼠标、键盘、LCD、触摸屏等。
    • 按照字节为单位访问,支持顺序访问。
    • 创建设备文件,通过openreadwriteclose等操作访问。
  • 块设备驱动:如摄像头、U盘、eMMC等。
    • 按照块(通常是512字节)访问,支持顺序和随机访问。
    • 创建设备文件,通过openreadwriteclose等操作访问。
  • 网络设备驱动:如网卡。
    • 按照网络数据包进行收发。

内核类型

  • 宏内核:将主要功能集成在一个内核中。
    • 优点:运行效率高。
    • 缺点:任何一个部分出错都可能导致整个内核崩溃。
    • 示例:Ubuntu, Android
  • 微内核:只包含最基本的功能,其他功能通过服务的形式在用户空间实现。
    • 优点:更高的稳定性和安全性。
    • 缺点:相对较低的运行效率。
    • 示例:HarmonyOS, QNX

驱动模块

  • 三要素:入口、出口、许可证。
  • 入口:资源的申请。
  • 出口:资源的释放。
  • 许可证:通常使用GPL许可。

__init可以不指定,及可以不写,但是正常是写的)

驱动模块示例

#include <linux/init.h>
#include <linux/module.h>//__init将hello_init放到.init.text段中
static int __init hello_init(void) {// 初始化函数return 0;
}//__exit将hello_exit放到.exit.text段中
static void __exit hello_exit(void) {// 清理函数
}//告诉内核驱动的入口地址(函数名为函数首地址)
module_init(hello_init);//告诉内核驱动的出口地址
module_exit(hello_exit);//许可证
MODULE_LICENSE("GPL");

Makefile配置

KERNELDIR := /lib/modules/$(shell uname -r)/build/
PWD := $(shell pwd)all:make -C $(KERNELDIR) M=$(PWD) modulesclean:make -C $(KERNELDIR) M=$(PWD) cleanobj-m += hello.o

命令

  • 安装驱动模块sudo insmod hello.ko
  • 卸载驱动模块sudo rmmod hello

  • 查看已加载的模块lsmod
  • 查看内核消息dmesg
  • 清空内核消息sudo dmesg -C 或 sudo dmesg -c
  • 持续查看内核消息:sudo dmesg -w

编码辅助工具

  • 创建索引文件ctags -R
  • 在vi中跳转至标签ctrl + ] 和 ctrl + t

Ubuntu内核所对应的内核路径

内核中的打印函数

printk 函数

  • 函数原型

    printk(打印级别 "内容")
  • 示例

    printk(KERN_ERR "Fail%d", a);
    printk(KERN_ERR "%s:%s:%d\n", __FILE__, __func__, __LINE__);
    printk("%s:%s:%d\n", __FILE__, __func__, __LINE__);
  • 查看内核打印级别

    vi -t KERN_ERR
  • 内核打印级别定义

    #define KERN_EMERG  "<0>"   /* system is unusable        */
    #define KERN_ALERT  "<1>"   /* action must be taken immediately */
    #define KERN_CRIT   "<2>"   /* critical conditions          */
    #define KERN_ERR    "<3>"   /* error conditions         */
    #define KERN_WARNING    "<4>"   /* warning conditions           */
    #define KERN_NOTICE "<5>"   /* normal but significant condition */
    #define KERN_INFO   "<6>"   /* informational            */
    #define KERN_DEBUG  "<7>"   /* debug-level messages         */

  • 打印级别范围

    • 从 <0> 到 <7><0> 为最高级别,<7> 为最低级别。
修改打印级别
  • 查看当前打印级别

    cat /proc/sys/kernel/printk
  • 打印级别的含义

    • 第一个数字:终端的级别。
    • 第二个数字:消息的默认级别。
    • 第三个数字:终端的最大级别。
    • 第四个数字:终端的最小级别。
  • 修改系统默认的级别

    su root
    echo 4 3 1 7 > /proc/sys/kernel/printk
  • 添加修改级别命令

echo 4 3 1 7 > /proc/sys/kernel/printk
打印级别含义
  • 终端的级别:只有当消息的级别大于或等于终端级别时,消息才会在终端上显示。
  • 消息的默认级别:如果没有特别指定,消息将采用此级别。
  • 终端的最大级别:终端可以显示的最高级别。

安装驱动和卸载驱动时,消息会打印。

 驱动多文件编译

示例
  • 文件列表

    • hello.c
    • add.c
  • Makefile

    obj-m := demo.o
    demo-y += hello.o add.o
  • 说明

    • -y 作用:将 hello.o 和 add.o 文件合并到 demo.o 中。
    • 最终生成demo.ko 文件。

 模块传递参数

命令行传递参数
  • 命令示例
    sudo insmod demo.ko hello world
支持的数据类型
  • 标准类型
    • byteshortushortintuintlongulong
    • charp: 字符串指针
    • bool: 布尔值,接受 0/1y/nY/N
    • invbool: 布尔值,接受 0/1y/nY/N,但意义相反(N 表示真)

module_param 函数
  • 函数原型
    module_param(name, type, perm);
  • 参数
    • name: 变量的名字。
    • type: 变量的类型。
    • perm: 权限,如 06640775

MODULE_PARM_DESC 函数
  • 函数原型
    MODULE_PARM_DESC(_parm, desc);
  • 参数
    • _parm: 变量。
    • desc: 描述字段。
示例
  • 命令行参数

    sudo insmod hello.ko a=20 b=30 c=65 p="hello_world"
  • 注意事项

    • 传递字符时使用 ASCII 码值。
    • 传递字符串时不能包含空格。
module_param_array 函数
  • 函数原型
    module_param_array(name, type, nump, perm);
  • 参数
    • name: 数组名。
    • type: 数组的类型。
    • nump: 参数的个数,变量的地址。
    • perm: 权限。

练习:

        1.byte类型如何使用 (传递参数用ascii)

        2.如何给一个指针传递一个字符串

  •  命令行参数
sudo insmod hello.ko a=121 b=10 c=65 p="hello" ww=1,2,3,4,5

传参成功~

字符设备驱动

Linux系统中一切皆文件
  • 应用层

    fd = open("led驱动的文件", O_RDWR);
    read(fd);
    write(fd);
    close(fd);
  • 内核层

    • 驱动文件led_driver.c
    • 驱动函数
      driver_open();
      driver_read();
      driver_write();
      driver_close();
  • 结构体定义

    struct file_operations {int (*open)(struct inode *, struct file *);ssize_t (*read)(struct file *, char __user *, size_t, loff_t *);ssize_t (*write)(struct file *, const char __user *, size_t, loff_t *);int (*release)(struct inode *, struct file *);
    ;
  • 设备号

    • 32位无符号数字
      • 高12位:主设备号,用于区分设备类别。
      • 低20位:次设备号,用于区分同一类别中的不同设备。
  • 硬件层

    • LEDUARTADCPWM 等设备。
字符设备驱动步骤
  1. 注册字符设备驱动 - 得到一个字符设备驱动的框架,并获得设备号。
  2. 确定操作的硬件设备 - 如 LED 灯(初始化灯)。
  3. 初始化灯 - 建立灯实际物理地址和虚拟地址之间的映射。
  4. 用户空间与内核空间数据交互 - 当用户使用时,驱动会被真正运行,涉及数据交互。
  5. 在应用层创建设备文件(设备节点)。
字符设备驱动的注册
  • 函数原型

    int register_chrdev(unsigned int major, const char *name,const struct file_operations *fops);
  • 参数

    • major:主设备号。
      • 如果填写的值大于0,它认为这个就是主设备号。
      • 如果填写的值为0,操作系统会分配一个主设备号。
    • name:设备名称。
    • fops:操作方法结构体。
  • 返回值

    • major > 0:成功返回0,失败返回错误码(负数)。
    • major = 0:成功返回分配的主设备号,失败返回错误码(负数)。
  • 查看设备信息

    cat /proc/devices
字符设备驱动的注销
  • 函数原型

    void unregister_chrdev(unsigned int major, const char *name);
  • 参数

    • major:主设备号。
    • name:设备名称。
  • 返回值:无。

就分享到这,希望可以帮到你吧~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/406025.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

jmeter简单发送接口

一、安装jmeter 拥有java环境&#xff0c;再下载jmeter 安装之后解压到本地&#xff0c;jmeter中的bin目录配置到环境变量中 之后可以通过cmd中 jmeter.bat命令运行 二、利用jmeter发送接口请求 1、添加线程组 添加->线程->线程组 2、添加http请求 添加->取样器-&g…

利用Matlab求解高阶微分方程(ode45)

1、高阶微分方程的基本概念 二阶以及二阶以上的微分方程称之为高阶微分方程&#xff0c;一般来说&#xff0c;微分方程的阶数越高&#xff0c;求解的难度也就越大。求高阶方程的一个常用方法就是降低阶数。对二阶方程 &#xff0c;如果能用变量代换把它化成一阶方程&#xff0c…

学习记录——day33 HTTP

目录 一、HTTP相关概念 二、客服端请求 1、请求首部 2、 响应首部 三、线程实现HTTP并发服务器 一、HTTP相关概念 1、HTTP&#xff0c;全称Hyper Text Transfer Protocol&#xff0c;用于万维网&#xff08;world wide web&#xff09;进行超文本学习的传输协议 2、HTTP属…

计算xpclr

1.conda安装xpclr 首先安装流程很轻松 conda create -n xpclr -c bioconda xpclr conda activate xpclr xpclr -h 2.按照要求准备文件 XPCLR - 简书 (jianshu.com) 根据教程准备文件&#xff0c;vcf&#xff0c;计算好的map&#xff0c;以及样本文件txt 其实官网也有介绍…

Docker基础概述、Docker安装、Docker镜像加速、Docker镜像指令

1.为什么学docker 开发环境与测试环境不同&#xff0c;导致错误 因此docker提供解决方法———系统平滑移植&#xff0c;容器虚拟化技术 将代码与软件与配置文件 打包成一个镜像 2.docker的历练 创建一个开发环境内成为镜像文件再用docker使用镜像 3.什么是docker Docke…

MySQL5.7数据库---入门教程(小白教程)

一、MySQL安装 本文以MySQL5.7安装为例。在设置完root密码和添加一个用户后&#xff0c;一路默认。 1、 2、通过点击红圈里的箭头选择对应的版本。 3、 4、端口&#xff08;Port&#xff09;一般默认不需要更改。 5、 二、配置环境变量 配置环境变量可以方便在win系统中cmd…

流媒体服务器二 3学习 librtmp 库的配置使用

librtmp 库是个啥&#xff1f; librtmp是一个开源的基于C语言的库&#xff0c;提供了一个连接RTMP服务器&#xff0c;发送和接收RTMP流的API。 它可以用来开发流媒体播放器&#xff0c;网络直播等应用。它的主要特点是快速、稳定和低延迟。 librtmp支持RTMP&#xff0c;RTMPS…

Qt第十七章 多线程

文章目录 多线程1. 线程概念的起源2. 三种方式创建线程3. 启动线程前的准备工作4. 启动线程/退出线程5. 操作运行中的线程6. 为每个线程提供独立数据7.子线程不能操作ui解决方案 多线程 1. 线程概念的起源 单核CPU 早期还没有线程的概念&#xff0c;如何保证2个进程同时进行呢…

基于Java爬取微博数据(四) 获取 图片 or 视频

基于Java爬取微博数据四 获取 图片 or 视频 图片 or 视频转存 图片 or 视频注意点 前面已经讲述了基于 Java 爬取微博正文列表内容&#xff0c;微博用户主页内容以及导出爬取到的微博数据等操作&#xff0c;那么下面讲述一下如何处理微博正文中的图片/视频等内容。 图片 or 视…

(转载)使用zed相机录制视频

参照下面这个链接 https://blog.csdn.net/peng_258/article/details/127457199?ops_request_misc&request_id&biz_id102&utm_termzed2%E5%BD%95%E5%88%B6%E6%95%B0%E6%8D%AE%E9%9B%86&utm_mediumdistribute.pc_search_result.none-task-blog-2~all~sobaiduweb…

代码复现改进

代码复现&#xff0c;文献复现&#xff0c;文章复现&#xff0c; 算法复现&#xff0c;科研复现 Matlab,Python中英文均可 保证质量&#xff0c;加快你的研究速度 代码改进跑通&#xff0c;模型优化改进

三种相机模型总结(针孔、鱼眼、全景)

相机标定 文章目录 相机标定前言 前言 我们最常见的投影模型Perspective Projection Model描述的就是针孔相机的成像原理。从上面的图根据相似三角形可以得出 参考链接 https://zhuanlan.zhihu.com/p/540969207 相机标定之张正友标定法数学原理详解&#xff08;含python源码&a…

鹭鹰优化算法SBOA优化RBF神经网络的扩散速度实现多数入多输出数据预测,可以更改数据集(MATLAB代码)

一、鹭鹰优化算法介绍 鹭鹰优化算法&#xff08;Secretary Bird Optimization Algorithm, SBOA&#xff09;是一种新型的元启发式算法&#xff0c;它于2024年4月由Youfa Fu等人提出&#xff0c;并发表在SCI人工智能二区顶刊《Artificial Intelligence Review》上。该算法的灵感…

uniapp h5手机如何打开本地跑的前端项目进行本地调试

本地调试使用 vConsole是一个轻量级的移动端调试工具&#xff0c;可以在iOS设备上直接调试Uniapp H5应用。下面是具体的步骤&#xff1a; 在Uniapp项目中安装vConsole依赖&#xff1a;npm install vconsole。 在项目的main.js文件中引入vConsole库&#xff1a;import VConso…

将iso格式的镜像文件转化成云平台能安装的镜像格式(raw/vhd/QCOW2/VMDK )亲测--图文详解

1.首先,你将你的iso的文件按照正常的流程和需求安装到你的虚拟机中,我这里使用的是vmware,安装完成之后,关机。再次点开你安装好的那台虚拟机的窗口,如下图 选中要导出的镜像,镜像需要关机 2.点击工具栏的文件------选择 导出 整个工程到 ovf 格式—这里你可以选择你要导…

思科设备静态路由实验

拓扑及需求 网络拓扑及 IP 编址如图所示&#xff1b;PC1 及 PC2 使用路由器模拟&#xff1b;在 R1、R2、R3 上配置静态路由&#xff0c;保证全网可达&#xff1b;在 R1、R3 上删掉上一步配置的静态路由&#xff0c;改用默认路由&#xff0c;仍然要求全网可达。 各设备具体配置…

【数模修炼之旅】05 拟合模型 深度解析(教程+代码)

【数模修炼之旅】05 拟合模型 深度解析&#xff08;教程代码&#xff09; 接下来 C君将会用至少30个小节来为大家深度解析数模领域常用的算法&#xff0c;大家可以关注这个专栏&#xff0c;持续学习哦&#xff0c;对于大家的能力提高会有极大的帮助。 1 拟合模型介绍及应用 …

C++ | Leetcode C++题解之第357题统计各位数字都不同的数字个数

题目&#xff1a; 题解&#xff1a; class Solution { public:int countNumbersWithUniqueDigits(int n) {if (n 0) {return 1;}if (n 1) {return 10;}int ans 10, cur 9;for (int i 0; i < n - 1; i) {cur * 9 - i;ans cur;}return ans;} };

91.WEB渗透测试-信息收集-Google语法(5)

免责声明&#xff1a;内容仅供学习参考&#xff0c;请合法利用知识&#xff0c;禁止进行违法犯罪活动&#xff01; 内容参考于&#xff1a; 易锦网校会员专享课 上一个内容&#xff1a;90.WEB渗透测试-信息收集-Google语法&#xff08;4&#xff09; 怎样判断哪些漏洞有什么样…

数据结构——顺序栈和链式栈

目录 引言 栈的定义 栈的分类 栈的功能 栈的声明 1.顺序栈 2.链式栈 栈的功能实现 1.栈的初始化 (1)顺序栈 (2)链式栈 (3)复杂度分析 2.判断栈是否为空 (1)顺序栈 (2)链式栈 (3)复杂度分析 3.返回栈顶元素 (1)顺序栈 (2)链式栈 (3)复杂度分析 4.返回栈的大…