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) modules

clean:
    make -C $(KERNELDIR) M=$(PWD) clean

obj-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.mfbz.cn/a/872049.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【机器学习】经典CNN架构

第一章:引言 1.1 研究背景 机器学习的发展历程 机器学习作为人工智能的重要分支,其发展历程可以追溯到20世纪50年代。初期的机器学习研究主要集中在规则系统和基本的统计学习方法上。随着计算能力的提升和数据的积累,机器学习逐渐发展出更为复杂的算法和模型。20世纪80年…

7.添加购物车以及完善导航条跳转

添加购物车 一、引言二、前端2.1引入轻组件2.2为图片添加点击事件2.3创建addShoppingCart函数 三、后端四、运行效果五、导航条跳转 一、引言 在前几篇文章中&#xff0c;我们一步一步慢慢的实现了项目的建立&#xff0c;从数据库获取数据显示在页面&#xff0c;商品大图查看&…

C++ JAVA源码 HMAC计算 openssl 消息认证码计算 https消息防篡改 通信安全

签名和验签 把所有消息按顺序合并成一条信息&#xff0c;对这个信息用密钥进行签名。 签名信息通过 HTTP 头 Sign 传递&#xff0c;没有携带签名或者签名验证不通过的请求&#xff0c;将会被认为异常请求&#xff0c;并返回相应 code 码。 校验方法&#xff1a;根据 http请求…

国产3A游戏《黑神话悟空》中AI绘画技术的运用与探索

导语&#xff1a;近年来&#xff0c;我国游戏产业不断发展&#xff0c;越来越多的国产游戏开始尝试运用AI技术&#xff0c;以提升游戏品质。其中&#xff0c;国产3A游戏《黑神话悟空》便在原画设计过程中&#xff0c;巧妙地运用了AI绘画技术。本文将带你了解《黑神话悟空》如何…

【计算机三级-数据库技术】操作题大题(第六套)

第六套操作题 第46题 假定要建立一个学校科研项目管理的信息系统&#xff0c;需要管理如下信息&#xff1a; 教师&#xff1a;教师编号、教师姓名&#xff1b; 项目&#xff1a;项目编号、项目名称、资助额&#xff1a; 学生&#xff1a;学生编号、学生姓名、学位&#xff0c…

Vodafone 推出了与 Wi-Fi 竞争的基于树莓派私人5G技术

随着全球5G网络的逐步推出&#xff0c;在其过程中遇到了可预见的起起伏伏&#xff0c;并且蜂窝技术也开始进入另一个无线技术 Wi-Fi &#xff0c;并且已经占据的市场。私有5G网络&#xff08;即个人或公司建立自己的全设施蜂窝网络&#xff09;如今正在寻找曾经属于Wi-Fi的唯一…

AR 眼镜之-系统应用音效-实现方案

目录 &#x1f4c2; 前言 AR 眼镜系统版本 系统应用音效 1. &#x1f531; 技术方案 1.1 技术方案概述 1.2 实现方案 1&#xff09;初始化 2&#xff09;播放音效 3&#xff09;释放资源 2. &#x1f4a0; 播放音效 2.1 静音不播放 2.2 获取音效默认音量 3. ⚛️ …

QTCreator学习

1.新建程序 2. 设置项目名称 3. Build System选择qmake,若选择cmake则只会产生CmakeLists文件&#xff0c;不会产生pro文件。 4.Base class选择QDialog,表示该类继承于QDialog类 5.套件选择MinGW 32bit,取消掉其他的。 6. 双击ui文件&#xff0c;拖动可添加工具。 7.点击左…

C++模拟实现priority_queue(优先级队列)

一、priority_queue的函数接口 从上图我们可以看出&#xff0c; priority_queue也是一个容器适配器&#xff0c;我们使用vector容器来模拟实现priority_queue。 namespace bit{#include<vector>#include<functional>template <class T, class Container vector…

iOS App上架审核被拒——2.3.3 - Performance - Accurate Metadata

iOS上架审核被拒——Guideline 2.3.3 - Performance - Accurate Metadata 噢&#xff0c;又被拒了… 文章目录 iOS上架审核被拒——Guideline 2.3.3 - Performance - Accurate Metadata被拒原因解决 被拒原因 大概翻译了下&#xff1a;预览图问题&#xff0c;只因某张预览图加了…

正则表达式备查

一、常用 符号内容\将下一字符标记为特殊字符、文本、反向引用或八进制转义符。例如&#xff0c;“n”匹配字符“n”。“\n”匹配换行符。序列“\”匹配“\”&#xff0c;“(”匹配“(”。^匹配输入字符串开始的位置。如果设置了 RegExp 对象的 Multiline 属性&#xff0c;^ 还…

CSS知识点详解:display+float

display&#xff1a;浮动 1.block&#xff1a;使元素呈现为块级元素&#xff0c;可设置宽高 display: block; 特点&#xff1a;使元素呈现为块级元素&#xff0c;即该元素会以新行开始&#xff0c;占据整行的宽度&#xff0c;即使其宽度未满。 例子&#xff1a; 2.inline&a…

答题小程序的轮播图管理与接入获取展示实现

实现了答题小程序的轮播图管理&#xff0c;包括上传图片、设置轮播图、操作上下线等功能&#xff0c;可用于管理各类答题小程序的轮播图。 轮播图前端接入代码 答题小程序内使用以下代码接入轮播图&#xff1a; WXML&#xff1a; <view style"width: 100%"> …

继承(下)【C++】

文章目录 子类继承父类之后&#xff0c;子类的默认成员函数的变化构造函数编译器自动生成的构造函数程序员手动写的构造函数 拷贝构造编译器自动生成的拷贝构造函数程序员手动写的拷贝构造函数 赋值重载编译器自动生成的赋值重载程序员手动写的赋值重载 析构函数 继承与友元菱形…

ISO 26262中的失效率计算:IEC 61709-Clause 17_Switches and push-buttons

概要 IEC 61709是国际电工委员会&#xff08;IEC&#xff09;制定的一个标准&#xff0c;即“电子元器件 可靠性 失效率的基准条件和失效率转换的应力模型”。主要涉及电学元器件的可靠性&#xff0c;包括失效率的基准条件和失效率转换的应力模型。本文介绍IEC 61709第十七章&…

Linux安装并配置Hadoop

目录 一、安装并配置JDK二、安装并配置Hadoop三、安装过程中遇到的问题总结 一、安装并配置JDK Linux上一般会安装Open JDK,关于OpenJDK和JDK的区别&#xff1a;http://www.cnblogs.com/sxdcgaq8080/p/7487369.html 准备Open JDK 1.8 查询可安装的java版本 yum -y list jav…

C语言第17篇

1.在C语言中,全局变量的存储类别是_________. A) static B) extern C) void D) register 提示&#xff1a;extern adj.外来的 register n.登记表&#xff0c;v.登记 提示与本题无关 2.在一个C源程序文件中,要定义一个只允许本源文件中所有函数使用的全局变…

【经典算法】BFS_最短路问题

目录 1. 最短路问题介绍2. 算法原理和代码实现(含题目链接)1926.迷宫中离入口最近的出口433.最小基因变化127.单词接龙675.为高尔夫比赛砍树 3. 算法总结 1. 最短路问题介绍 最短路径问题是图论中的一类十分重要的问题。本篇文章只介绍边权为1(或边权相同)的最简单的最短路径问…

ant design pro 中用户的表单如何控制多个角色

ant design pro 如何去保存颜色ant design pro v6 如何做好角色管理ant design 的 tree 如何作为角色中的权限选择之一ant design 的 tree 如何作为角色中的权限选择之二ant design pro access.ts 是如何控制多角色的权限的 看上面的图片 当创建或编辑一个用户时&#xff0c;…

自带灭火电池?深蓝SL03托底事故揭秘

近日&#xff0c;网络上的一段热传视频&#xff0c;让不少网友看得先是惊心动魄&#xff0c;然后却又啧啧称奇。 该视频显示&#xff0c;8月18日晚上19点28分&#xff0c;一辆深蓝SL03在行驶中意外遭遇严重托底事故&#xff0c;车辆瞬间腾空跳跃&#xff0c;紧接着底盘出现明火…