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

Linux字符设备驱动开发的详细步骤

1. 确定主设备号​

  • ​手动指定​​:明确设备号时,使用register_chrdev_region()静态申请(需确保未被占用)。
  • ​动态分配​​:通过alloc_chrdev_region()由内核自动分配主设备号(更灵活,推荐)。
    dev_t dev; 
    alloc_chrdev_region(&dev, 0, 1, "mydevice"); // 动态分配主设备号

​2. 定义file_operations结构体​

  • ​结构体作用​​:关联用户空间系统调用与驱动函数(如open/read/write)。
  • ​关键成员​​:
    static struct file_operations fops = {.owner   = THIS_MODULE,  // 模块所有者标识.open    = drv_open,     // 设备打开函数.read    = drv_read,     // 数据读取函数.write   = drv_write,    // 数据写入函数.release = drv_release,  // 设备关闭函数
    };

​3. 实现驱动函数​

需实现drv_opendrv_readdrv_write等函数,并处理硬件交互逻辑:

  • ​示例:open函数​​(初始化硬件或分配资源):
    static int drv_open(struct inode *inode, struct file *filp) {printk(KERN_INFO "Device opened\n");return 0;
    }
  • ​数据传输函数​​:需通过copy_from_user()copy_to_user()实现用户空间与内核空间的数据交换。

​4. 注册驱动到内核​

  • ​入口函数​​:通过module_init()指定驱动加载时的初始化函数:
    static int __init mydrv_init(void) {// 注册字符设备,主设备号设为0表示自动分配int ret = register_chrdev(0, "mydevice", &fops);if (ret < 0) {printk(KERN_ERR "Register failed\n");return ret;}// 自动创建设备节点(后续补充)return 0;
    }
    module_init(mydrv_init);
  • ​出口函数​​:通过module_exit()指定驱动卸载时的清理函数:
    static void __exit mydrv_exit(void) {unregister_chrdev(major, "mydevice");
    }
    module_exit(mydrv_exit);

​5. 自动创建设备节点​

  • ​使用class_createdevice_create​:
    static struct class *dev_class;
    static dev_t dev;static int __init mydrv_init(void) {// 创建设备类dev_class = class_create(THIS_MODULE, "mydevice_class");// 创建设备节点(/dev/mydevice)device_create(dev_class, NULL, dev, NULL, "mydevice");return 0;
    }static void __exit mydrv_exit(void) {device_destroy(dev_class, dev); // 销毁节点class_destroy(dev_class);        // 销毁类
    }
    • ​作用​​:用户空间可通过/dev/mydevice直接访问设备,无需手动mknod

​6. 其他完善步骤​

  • ​错误处理​​:检查register_chrdevclass_create等函数的返回值,避免资源泄漏。
  • ​资源释放​​:在出口函数中释放所有申请的资源(如设备号、内存)。
  • ​兼容性​​:确保驱动代码与内核版本匹配,遵循内核编码规范。

​总结流程图​

驱动初始化(入口函数)
├─ 分配设备号(动态/静态)
├─ 初始化file_operations结构体
├─ 注册字符设备(register_chrdev)
├─ 创建设备类(class_create)
└─ 创建设备节点(device_create)驱动卸载(出口函数)
├─ 注销字符设备(unregister_chrdev)
├─ 销毁设备节点(device_destroy)
└─ 销毁设备类(class_destroy)

通过以上步骤,可完成一个完整的Linux字符设备驱动开发流程。具体实现需结合硬件特性调整函数逻辑(如中断处理、DMA操作)

http://www.dtcms.com/a/154616.html

相关文章:

  • Vue3实现高仿word自定义颜色选择器组件(支持 v-model)
  • 矩阵运营:抢占市场与流量的利器
  • 如何避免IDEA每次打开新项目都重复配置Maven?
  • 【Java并发】【LinkedBlockingQueue】适合初学体质的LinkedBlockingQueue入门
  • C++ 基础知识 指针和函数(利用指针作函数的参数修改实参的值)
  • 代码随想录打卡|Day27(合并区间、单调递增的数字、监控二叉树)
  • 河道流量监测:雷达流量计与多普勒流量计的优劣势与技术特点分析
  • 【C语言】全局变量、静态本地变量
  • 常见的几种分块策略,每种策略都有适用场景和优缺点
  • 华为L410上制作内网镜像模板:搭建Apache服务器并上传离线WeChat包
  • Linux-05 半个月崩了三次 ubuntu 系统记录
  • 基于STM32、HAL库的ADS1256IDBR模数转换器ADC驱动程序设计
  • 常见的6种外挂获取执行时机方法介绍
  • 安全编排自动化与响应(SOAR):从事件响应到智能编排的技术实践
  • 常见游戏引擎介绍与对比
  • Linux的基础指令
  • Exposure Adjusted Incidence Rate (EAIR) 暴露调整发病率:精准量化疾病风险
  • 关于hbaseRegion和hbaseRowKey的一些处理
  • linux ptrace 图文详解(七) gdb、strace跟踪系统调用
  • Python解析地址中省市区街道
  • Codeforces Round 1020 (Div. 3) A-D
  • 【高频考点精讲】ES6 String的新增方法,处理字符串更方便了
  • Unreal制作角色冲锋时冲击波效果
  • 论文学习:《聚类矩阵正则化指导的层次图池化》
  • 数学基础 -- 欧拉恒等式的魅力:让复数旋转起来!
  • 通信算法之273 : 循环自相关函数和自相关函数
  • LeNet5 神经网络的参数解析和图片尺寸解析
  • uniapp 常用开发技巧与实战指南
  • unity 读取csv
  • 消息队列mq在Mlivus Cloud向量数据库中的关键配置与最佳实践