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

两段驱动代码的区别

1

#include <linux/module.h> #include <linux/kernel.h> static int __init helloworld_init(void) //驱动入口函数 { printk(KERN_EMERG "helloworld_init\r\n");//注意: 内核打印用 printk 而不是 printf return 0; } static void __exit helloworld_exit(void) //驱动出口函数 { printk(KERN_EMERG "helloworld_exit\r\n"); } module_init(helloworld_init); //告诉linux模块入口函数,加载模块代码到操作系统 module_exit(helloworld_exit); //卸载 MODULE_LICENSE("GPL v2"); //同意 GPL 开源协议 MODULE_VERSION("1.0"); //驱动的版本 MODULE_AUTHOR("Cavium Inc"); MODULE_DESCRIPTION("helloworld Driver"); //lsmod

2

#include <linux/init.h> #include <linux/module.h> #include <linux/fs.h> #include <linux/kdev_t.h> #include <linux/cdev.h> static dev_t dev_num;//定义dev_t类型(32位大小)的变量dev_num,用来存放设备号 static struct cdev cdev_test;//定义cdev结构体类型的变量cdev_test static struct file_operations cdev_test_ops = { .owner=THIS_MODULE,//将owner字段指向本模块,可以避免在模块的操作正在被使用时卸载该模块 };//定义file_operations结构体类型的变量cdev_test_ops static int __init module_cdev_init(void)//驱动入口函数 { int ret;//定义int类型变量ret,进行函数返回值判断 int major,minor;//定义int类型的主设备号major和次设备号minor ret = alloc_chrdev_region(&dev_num,0,1,"chrdev_name");//自动获取设备号,设备名为chrdev_name if (ret < 0){ printk("alloc_chrdev_region is error\n"); } printk("alloc_register_region is ok\n"); major = MAJOR(dev_num);//使用MAJOR()函数获取主设备号 minor = MINOR(dev_num);//使用MINOR()函数获取次设备号 printk("major is %d\n",major); printk("minor is %d\n",minor); cdev_init(&cdev_test,&cdev_test_ops);//使用cdev_init()函数初始化cdev_test结构体,并链接到cdev_test_ops结构体 cdev_test.owner = THIS_MODULE;//将owner字段指向本模块,可以避免在模块的操作正在被使用时卸载该模块 ret = cdev_add(&cdev_test,dev_num,1);//使用cdev_add()函数进行字符设备的添加 if(ret < 0 ){ printk("cdev_add is error\n"); } printk("cdev_add is ok\n"); return 0; } static void __exit module_cdev_exit(void)//驱动出口函数 { cdev_del(&cdev_test);//使用cdev_del()函数进行字符设备的删除 unregister_chrdev_region(dev_num,1);//释放字符驱动设备号 printk("module exit \n"); } module_init(module_cdev_init);//注册入口函数 module_exit(module_cdev_exit);//注册出口函数 MODULE_LICENSE("GPL v2");//同意GPL开源协议

一、先分别明确两段代码的核心功能

1. 第一段(helloworld 驱动):最基础的 “空” 内核模块

这段代码是 Linux 内核模块的入门模板,没有任何实际的设备操作功能,仅完成 “模块加载” 和 “模块卸载” 的基础流程:

  • 加载模块时:通过printk输出日志(helloworld_init
  • 卸载模块时:通过printk输出日志(helloworld_exit
  • 额外补充了模块信息(版本、作者、描述),但这些信息仅用于标识模块,无实际功能。

2. 第二段(字符设备驱动):实用设备驱动的 “雏形”

这段代码是字符设备驱动的核心框架,在 helloworld 基础上,新增了 “字符设备注册 / 管理” 的核心逻辑,目的是为用户空间程序提供 “访问硬件设备” 的内核层入口,具备了实际设备驱动的基础骨架。

二、核心区别对比

对比维度第一段(helloworld 驱动)第二段(字符设备驱动)
功能定位内核模块入门演示,无实际功能字符设备驱动核心框架,为硬件操作提供内核入口
复杂程度极简(仅包含模块入口 / 出口 + 日志输出)中等(新增设备号、cdev、文件操作集等核心组件)
涉及内核组件仅基础模块框架(module_init/module_exit1. 设备号管理(dev_t/alloc_chrdev_region)2. 字符设备管理(cdev/cdev_init/cdev_add)3. 文件操作集(file_operations
实际用途验证内核模块编译、加载、卸载的基本流程作为实际字符设备(如串口、LED、按键)驱动的基础,后续可扩展读写、控制等设备操作
关键函数(独有)无独有核心函数(仅基础日志输出)alloc_chrdev_region(分配设备号)、cdev_init(初始化字符设备)、cdev_add(注册字符设备)、unregister_chrdev_region(释放设备号)、cdev_del(删除字符设备)
与硬件 / 用户空间的关联无任何关联(既不操作硬件,也不提供用户空间访问接口)搭建了 “用户空间→内核空间” 的访问桥梁(通过设备号和设备节点),后续可扩展硬件操作逻辑
日志输出目的仅验证模块是否成功加载 / 卸载除验证流程外,还排查设备号分配、字符设备注册的错误(如alloc_chrdev_region is error

三、关键差异的通俗解释

1. 核心差异:是否 “关联设备”

  • 第一段代码就像一个 “空盒子”:你把它放进内核(加载模块),它只告诉你 “我进来了”,把它拿出来(卸载模块),它只告诉你 “我走了”,盒子里没有任何东西。
  • 第二段代码就像一个 “带接口的设备盒子”:它放进内核后,会先向内核 “申请编号”(设备号),再把 “盒子接口”(cdev 字符设备)注册到内核,用户空间程序可以通过这个 “编号” 和 “接口” 访问盒子内部(硬件设备),卸载时还会归还 “编号” 并销毁 “接口”。

2. 组件差异:新增的核心结构体 / 函数的意义

  • dev_t dev_num:设备的 “身份证号”(主设备号 + 次设备号),用户空间通过这个编号找到对应的设备。
  • struct cdev:字符设备的 “管理结构体”,相当于设备的 “管家”,负责统筹设备的操作逻辑。
  • struct file_operations:设备的 “操作手册”,定义了用户空间可以对设备执行的操作(如读、写),第二段代码中仅初始化了owner字段,后续可添加read/write等函数实现实际操作。
  • alloc_chrdev_region/unregister_chrdev_region:向内核 “申请 / 归还” 设备号,避免设备号冲突。
  • cdev_init/cdev_add/cdev_del:初始化、注册、销毁字符设备,让内核识别并管理该设备。

3. 递进关系

第一段代码是第二段代码的基础,第二段代码是在第一段代码的框架上,新增了 “字符设备管理” 的核心逻辑,是从 “入门演示” 到 “实际驱动开发” 的关键递进。

简单说:helloworld 驱动是 “热身”,字符设备驱动是 “正式开始干活”

总结

  1. 功能定位不同:helloworld 驱动是无实际功能的入门模板,字符设备驱动是具备设备管理能力的实用驱动雏形。
  2. 组件复杂度不同:字符设备驱动新增了设备号、cdev 结构体、文件操作集等核心组件,是实际硬件驱动的基础框架。
  3. 用途不同:helloworld 仅验证模块加载 / 卸载流程,字符设备驱动为用户空间访问硬件提供了内核层入口,可直接扩展为实际设备驱动。
http://www.jsqmd.com/news/126470/

相关文章:

  • LangFlow中的数据清洗节点:预处理原始文本的有效方法
  • 设置中心-Cordovaopenharmony统一配置入口
  • LangFlow中的广告文案生成:高转化率内容批量产出
  • 从零实现干净驱动环境:DDU完整指南
  • N_m3u8DL-RE终极VR视频下载指南:快速获取360°全景内容
  • Keil4安装环境搭建:从零开始
  • WeChatExtension-ForMac:打造专业级Mac微信增强体验
  • LangFlow与语法纠错工具集成:提升文本专业度
  • 终极指南:5步轻松掌握虚拟机检测工具VMDE
  • LangFlow与Redis集成:实现高速数据缓存与共享
  • LangFlow与翻译API集成:构建多语言内容处理管道
  • LangFlow与冥想引导结合:心理健康辅助工具
  • 74HC74 D触发器电路图连接方法图解说明
  • Topit终极Mac窗口置顶工具:彻底告别窗口遮挡烦恼
  • 《C++初阶之类和对象》【类的六大默认成员函数】
  • LangFlow与股票行情接口结合:金融信息实时推送
  • LangFlow与命名实体识别(NER)结合:信息抽取利器
  • 【C++】简单介绍lambda表达式
  • 工厂数字孪生解决方案提供商深度盘点:技术路径/应用实践/市场份额全面对比分析
  • LangFlow与简历筛选结合:HR招聘流程智能化
  • LangFlow中的异步任务处理:提升整体执行效率
  • OrCAD原理图驱动Allegro布局布线的系统学习
  • 通俗解释ModbusRTU功能码与数据格式
  • 全面讲解ESP32连接阿里云MQTT准备工作
  • LangFlow中的版权检测器:识别潜在侵权内容
  • DA-03 双声道I2S数字音频转模拟音频模组,让每一段数字信号都焕发真实听觉生命力!
  • LangFlow中的SEO标题优化器:提升搜索引擎排名
  • Multisim示波器时间基准调节:实战案例演示
  • 大盘风险控制策略分析报告 - 2025年12月23日
  • 提醒列表模块 Cordova 与 OpenHarmony 混合开发实战