Linux驱动开发实战(一):LED控制驱动详解

Linux驱动开发野火实战(一):LED控制驱动详解


文章目录

  • Linux驱动开发野火实战(一):LED控制驱动详解
  • 引言
  • 一、基础知识
    • 1.1 什么是字符设备驱动
    • 1.2 重要的数据结构
      • read 函数
      • write 函数
      • open 函数
      • release 函数
  • 二、 驱动程序实现
    • 2.1 完整的驱动代码示例
    • 2.2 整体流程(图解)
    • 2.3 用户空间与内核空间交互(图解)
    • 2.4 驱动模块初始化
      • 虚拟地址映射
    • 2.5 拷贝数据
    • 2.6 控制GPIO输出的LED开关状态
    • 2.7 LED驱动程序的退出函数
  • 三、实验过程
    • 项目编译
    • 连接开发板
      • 挂载NFS文件系统
    • 加载驱动(点灯!)
  • 总结


引言

在Linux设备驱动开发中,字符设备驱动是最基础也是最常见的驱动类型。本文将从理论到实践,详细讲解字符设备驱动的开发流程,帮助读者掌握驱动开发的核心知识


一、基础知识

1.1 什么是字符设备驱动

字符设备(Character Device)是Linux中最基本的设备类型之一,它的特点是数据以字符流的方式被访问,像串口、键盘、LED等都属于字符设备。与块设备不同,字符设备不能随机访问,只能顺序读写。

1.2 重要的数据结构

在开发字符设备驱动之前,我们需要了解几个关键的数据结构:

struct file_operations {struct module *owner;ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);int (*open) (struct inode *, struct file *);int (*release) (struct inode *, struct file *);...
};

这个结构体定义了驱动程序需要实现的接口函数。

read 函数

static ssize_t led_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)

作用: 响应用户空间的读取请求
参数:

  • filp:文件结构体指针
  • buf:用户空间缓冲区指针
  • cnt:要读取的字节数
  • offt:文件位置指针
    返回值:
  • 成功返回实际读取的字节数
  • 失败返回负错误码
  • 使用场景:
  • 读取设备状态
  • 获取设备数据
  • 将数据从内核空间复制到用户空间

write 函数

static ssize_t led_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)

作用: 响应用户空间的写入请求
参数:

  • filp:文件结构体指针
  • buf:用户空间数据缓冲区指针
  • cnt:要写入的字节数
  • offt:文件位置指针
    返回值:
  • 成功返回实际写入的字节数
  • 失败返回负错误码
    使用场景:
  • 向设备发送控制命令
  • 更新设备状态
  • 将数据从用户空间复制到内核空间

open 函数

static int led_open(struct inode *inode, struct file *filp)

作用: 当用户空间调用 open() 打开设备文件时被调用
参数:

  • inode:包含文件系统信息的结构体,如设备号等
  • filp:文件结构体,包含文件操作方法、私有数据等
    返回值:
  • 成功返回0
  • 失败返回负错误码
    使用场景:
  • 初始化设备
  • 检查设备状态
  • 分配资源
  • 增加使用计数

release 函数

static int led_release(struct inode *inode, struct file *filp)

作用: 当最后一个打开的文件被关闭时调用
参数:

  • inode:索引节点结构体指针
  • filp:文件结构体指针
    返回值:
  • 成功返回0
  • 失败返回负错误码
    使用场景:
  • 释放资源
  • 清理设备状态
  • 减少使用计数

二、 驱动程序实现

2.1 完整的驱动代码示例

代码如下:

#include <linux/module.h>
#include <linux/init.h>
#include <linux/kernel.h>#include <linux/fs.h>
#include <linux/uaccess.h>
#include <asm/io.h>#define DEV_MAJOR 0		   /* 动态申请主设备号 */
#define DEV_NAME "red_led" /*led设备名字 *//* GPIO虚拟地址指针 */
static void __iomem *IMX6U_CCM_CCGR1;
static void __iomem *SW_MUX_GPIO1_IO04;
static void __iomem *SW_PAD_GPIO1_IO04;
static void __iomem *GPIO1_DR;
static void __iomem *GPIO1_GDIR;static int led_open(struct inode *inode, struct file *filp)
{return 0;
}static ssize_t led_read(struct file *filp, char __user *buf, size_t cnt, loff_t *offt)
{return -EFAULT;
}static ssize_t led_write(struct file *filp, const char __user *buf, size_t cnt, loff_t *offt)
{unsigned char databuf[10];if (cnt > 10)cnt = 10;/*从用户空间拷贝数据到内核空间*/if (copy_from_user(databuf, buf, cnt)){return -EIO;}if (!memcmp(databuf, "on", 2)){iowrite32(0 << 4, GPIO1_DR);}else if (!memcmp(databuf, "off", 3)){iowrite32(1 << 4, GPIO1_DR);}/*写成功后,返回写入的字数*/return cnt;
}static int led_release(struct inode *inode, struct file *filp)
{return 0;
}/* 自定义led的file_operations 接口*/
static struct file_operations led_fops = {.owner = THIS_MODULE,.open = led_open,.read = led_read,.write = led_write,.release = led_release,
};int major = 0;
static int __init led_init(void)
{/* GPIO相关寄存器映射 */IMX6U_CCM_CCGR1 = ioremap(0x20c406c, 4);SW_MUX_GPIO1_IO04 = ioremap(0x20e006c, 4);SW_PAD_GPIO1_IO04 = ioremap(0x20e02f8, 4);GPIO1_GDIR = ioremap(0x0209c004, 4);GPIO1_DR = ioremap(0x0209c000, 4);/* 使能GPIO1时钟 */iowrite32(0xffffffff, IMX6U_CCM_CCGR1);/* 设置GPIO1_IO04复用为普通GPIO*/iowrite32(5, SW_MUX_GPIO1_IO04);/*设置GPIO属性*/iowrite32(0x10B0, SW_PAD_GPIO1_IO04);/* 设置GPIO1_IO04为输出功能 */iowrite32(1 << 4, GPIO1_GDIR);/* LED输出高电平 */iowrite32(1 << 4, GPIO1_DR);/* 注册字符设备驱动 */major = register_chrdev(DEV_MAJOR, DEV_NAME, &led_fops);printk(KERN_ALERT "led major:%d\n", major);return 0;
}static void __exit led_exit(void)
{/* 取消映射 */iounmap(IMX6U_CCM_CCGR1);iounmap(SW_MUX_GPIO1_IO04);iounmap(SW_PAD_GPIO1_IO04);iounmap(GPIO1_DR);iounmap(GPIO1_GDIR);/* 注销字符设备驱动 */unregister_chrdev(major, DEV_NAME);
}module_init(led_init);
module_exit(led_exit);MODULE_LICENSE("GPL2");
MODULE_AUTHOR("embedfire ");
MODULE_DESCRIPTION("led_module");
MODULE_ALIAS("led_module");

2.2 整体流程(图解)

在这里插入图片描述

2.3 用户空间与内核空间交互(图解)

在这里插入图片描述

2.4 驱动模块初始化

虚拟地址映射

  1. ioremap 函数
void __iomem *ioremap(unsigned long phys_addr, unsigned long size);

作用: 将物理地址映射到虚拟地址空间
参数:

  • phys_addr:物理地址
  • size:映射的大小(字节数)
    返回值: 映射后的虚拟地址指针
  • void * 类型的指针,指向被映射的虚拟地址
  • __iomem 主要是用于编译器的检查地址在内核空间的有效性
    为什么要用: Linux内核出于安全考虑,不允许直接访问物理地址,必须先映射到虚拟地址

GPIO相关寄存器映射

	IMX6U_CCM_CCGR1 = ioremap(0x20c406c, 4);SW_MUX_GPIO1_IO04 = ioremap(0x20e006c, 4);SW_PAD_GPIO1_IO04 = ioremap(0x20e02f8, 4);GPIO1_GDIR = ioremap(0x0209c004, 4);GPIO1_DR = ioremap(0x0209c000, 4);
  1. 虚拟地址读写
void iowrite32(u32 b, void __iomem *addr)   //写入一个双字(32bit)unsigned int ioread32(void __iomem *addr)   //读取一个双字(32bit)
/* 使能GPIO1时钟 */iowrite32(0xffffffff, IMX6U_CCM_CCGR1);/* 设置GPIO1_IO04复用为普通GPIO*/iowrite32(5, SW_MUX_GPIO1_IO04);/*设置GPIO属性*/iowrite32(0x10B0, SW_PAD_GPIO1_IO04);/* 设置GPIO1_IO04为输出功能 */iowrite32(1 << 4, GPIO1_GDIR);/* LED输出高电平 */iowrite32(1 << 4, GPIO1_DR);
  1. register_chrdev 函数
int register_chrdev(unsigned int major, const char *name, const struct file_operations *fops);

作用: 注册字符设备驱动
参数:

major:主设备号(0表示动态分配)
name:设备名称
fops:文件操作结构体
次设备号为0,次设备号数量为256
返回值: 成功返回主设备号,失败返回负值
为什么要用: 向Linux系统注册一个字符设备,使系统能够识别和管理该设备

在这里插入图片描述

2.5 拷贝数据

copy_from_user函数

unsigned long copy_from_user(void *to, const void __user *from, unsigned long n);

作用: 将数据从用户空间复制到内核空间
参数:

  • to:内核空间目标地址
  • from:用户空间源地址
  • n:复制的字节数
    返回值: 成功返回0,失败返回未复制的字节数
    为什么要用: 内核空间和用户空间是隔离的,需要专门的函数来安全地传输数据,要是有野指针会导致整个系统的崩溃,所以是起到一个安全的作用。

2.6 控制GPIO输出的LED开关状态

if (!memcmp(databuf, "on", 2))  // 比较是否接收到"on"命令
{iowrite32(0 << 4, GPIO1_DR); // GPIO1_4输出低电平,LED亮
}
else if (!memcmp(databuf, "off", 3)) // 比较是否接收到"off"命令
{iowrite32(1 << 4, GPIO1_DR); // GPIO1_4输出高电平,LED灭
}
  • memcmp()函数:
int memcmp(const void *str1, const void *str2, size_t n)
// 比较两个内存区域的前n个字节
// 返回0表示相等

2.7 LED驱动程序的退出函数

static void __exit led_exit(void)
{// 1. 取消IO内存映射iounmap(IMX6U_CCM_CCGR1);iounmap(SW_MUX_GPIO1_IO04);iounmap(SW_PAD_GPIO1_IO04);iounmap(GPIO1_DR);iounmap(GPIO1_GDIR);// 2. 注销字符设备unregister_chrdev(major, DEV_NAME);
}
  • iounmap 函数
void iounmap(void __iomem *addr);

作用: 解除I/O内存映射
参数:

addr: 要解除映射的虚拟地址
为什么要用: 释放ioremap占用的资源,防止内存泄漏

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

作用: 注销字符设备驱动
参数:

major:设备的主设备号
name:设备名称
为什么要用: 在驱动卸载时清理系统资源

三、实验过程

项目编译

在这里插入图片描述
然后make
在这里插入图片描述

连接开发板

打开手机热点并连上
让电脑跟手机在同一个局域网内

  • ubuntu端
    在这里插入图片描述

  • 开发板端
    在这里插入图片描述

挂载NFS文件系统

sudo mount -t nfs ”NFS服务端IP”:/home/embedfire/workdir /mnt

我们ubuntu的IP为192.168.46.118
所以为

sudo mount -t nfs 192.168.46.118:/home/embedfire/workdir /mnt

在这里插入图片描述
挂载成功后进入共享文件夹查看

ubuntu把ko文件复制到共享文件夹中
在这里插入图片描述

我们到共享文件夹ls查看
在这里插入图片描述

加载驱动(点灯!)

在这里插入图片描述
244是设备号(动态分配)
0是次设备号
为什么是0
因为在register_chrdev函数定义了
次设备号在(0~256之间随便选)
ebf-buster-linux/include/linux/fs.h

static inline int register_chrdev(unsigned int major, const char *name,const struct file_operations *fops)
{return __register_chrdev(major, 0, 256, name, fops);
}

创建设备文件
在这里插入图片描述
利用echo应用打开灯的命令
在这里插入图片描述
请添加图片描述

利用echo应用关灯的命令
在这里插入图片描述
卸载模块
在这里插入图片描述


总结

本文详细介绍了Linux字符设备驱动的开发流程,包括:

  • 基本概念和原理
  • 完整的代码实现
  • 详细的流程图解
  • 实际操作
    通过本文的学习,大家应该能够掌握字符设备驱动的开发方法,并能够开发简单的字符设备驱动程序。

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

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

相关文章

Linux上用C++和GCC开发程序实现不同MySQL实例下单个Schema之间的稳定高效的数据迁移

设计一个在Linux上运行的GCC C程序&#xff0c;同时连接两个不同的MySQL实例&#xff0c;两个实例中分别有两个Schema的表结构完全相同&#xff0c;复制一个实例中一个Schema里的所有表的数据到另一个实例中一个Schema里&#xff0c;使用以下快速高效的方法&#xff0c;加入异常…

Redis除了做缓存还能做什么?

Redis 除了作为高性能缓存外&#xff0c;还因其丰富的数据结构和功能&#xff0c;广泛应用于多种场景。以下是 Redis 的十大核心用途及具体示例&#xff1a; 1. 分布式会话存储 用途&#xff1a;存储用户会话信息&#xff08;如登录状态&#xff09;&#xff0c;实现多服务间共…

JBoltAI_SpringBoot如何区分DeepSeek R1深度思考和具体回答的内容(基于Ollama)?

当我们用Ollama运行DeepSeek R1模型&#xff0c;向它提问时&#xff0c;会发现它的回答里是有think标签的 如果我们直接将Ollama的回复用于生产环境&#xff0c;肯定是不行的&#xff0c;对于不同的场景&#xff0c;前面输出的一堆内容&#xff0c;可能并不需要在客户端展示&a…

MySQL 使用 `WHERE` 子句时 `COUNT(*)`、`COUNT(1)` 和 `COUNT(column)` 的区别解析

文章目录 1. COUNT() 函数的基本作用2. COUNT(*)、COUNT(1) 和 COUNT(column) 的详细对比2.1 COUNT(*) —— 统计所有符合条件的行2.2 COUNT(1) —— 统计所有符合条件的行2.3 COUNT(column) —— 统计某一列非 NULL 的记录数 3. 性能对比3.1 EXPLAIN 分析 4. 哪种方式更好&…

将DeepSeek接入vscode的N种方法

接入deepseek方法一:cline 步骤1:安装 Visual Studio Code 后,左侧导航栏上点击扩展。 步骤2:搜索 cline,找到插件后点击安装。 步骤3:在大模型下拉菜单中找到deep seek,然后下面的输入框输入你在deepseek申请的api key,就可以用了 让deepseek给我写了一首关于天气的…

AndroidManifest.xml文件的作用

AndroidManifest.xml文件在Android应用程序中扮演着至关重要的角色。它是应用程序的全局配置文件&#xff0c;提供了关于应用程序的所有必要信息&#xff0c;这些信息对于Android系统来说是至关重要的&#xff0c;因为它决定了应用程序的运行方式和权限要求&#xff0c;确保了应…

Mac本地部署Deep Seek R1

Mac本地部署Deep Seek R1 1.安装本地部署大型语言模型的工具 ollama 官网&#xff1a;https://ollama.com/ 2.下载Deepseek R1模型 网址&#xff1a;https://ollama.com/library/deepseek-r1 根据电脑配置&#xff0c;选择模型。 我的电脑&#xff1a;Mac M3 24G内存。 这…

React进阶之前端业务Hooks库(五)

前端业务Hooks库 Hooks原理useStateuseEffect上述问题useState,useEffect 复用的能力练习:怎样实现一套React过程中的hooks状态 & 副作用Hooks原理 不能在循环中、条件判断、子函数中调用,只能在函数最外层去调用useEffect 中,deps 为空,执行一次useState 使用: imp…

从像素到光线:现代Shader开发的范式演进与性能优化实践

引言 在实时图形渲染领域&#xff0c;Shader作为GPU程序的核心载体&#xff0c;其开发范式已从早期的固定功能管线演进为高度可编程的计算单元。本文通过解析关键技术案例&#xff0c;结合现代图形API&#xff08;如Vulkan、Metal&#xff09;的特性&#xff0c;深入探讨Shade…

(七)消息队列-Kafka 序列化avro(传递)

&#xff08;七&#xff09;消息队列-Kafka 序列化avro&#xff08;传递&#xff09; 客从远方来&#xff0c;遗我双鲤鱼。呼儿烹鲤鱼&#xff0c;中有尺素书。 ——佚名《饮马长城窟行》 本文已同步CSDN、掘金平台、知乎等多个平台&#xff0c;图片依然保持最初发布的水印&…

PXE批量网络装机与Kickstart自动化安装工具

目录 一、系统装机的原理 1.1、系统装机方式 1.2、系统安装过程 二、PXE批量网络装机 2.1、PXE实现原理 2.2、搭建PXE实际案例 2.2.1、安装必要软件 2.2.2、搭建DHCP服务器 2.2.3、搭建TFTP服务器 2.2.4、挂载镜像并拷贝引导文件到tftp服务启动引导文件夹下 2.2.5、编…

【全栈开发】从0开始搭建一个图书管理系统【一】框架搭建

【全栈开发】从0开始搭建一个图书管理系统【一】框架搭建 前言 现在流行降本增笑&#xff0c;也就是不但每个人都要有事干不能闲着&#xff0c;更重要的是每个人都要通过报功的方式做到平日的各项工作异常饱和&#xff0c;实现1.5人的支出干2人的活计。单纯的数据库开发【肤浅…

部署Flink1.20.1

1、设置环境变量 export JAVA_HOME/cluster/jdk export CLASSPATH.:$JAVA_HOME/lib/tools.jar:$JAVA_HOME/lib/dt.jarp #export HIVE_HOME/cluster/hive export MYSQL_HOME/cluster/mysql export HADOOP_HOME/cluster/hadoop3 export HADOOP_CONF_DIR$HADOOP_HOME/etc/hadoop …

【超详细】神经网络的可视化解释

《------往期经典推荐------》 一、AI应用软件开发实战专栏【链接】 项目名称项目名称1.【人脸识别与管理系统开发】2.【车牌识别与自动收费管理系统开发】3.【手势识别系统开发】4.【人脸面部活体检测系统开发】5.【图片风格快速迁移软件开发】6.【人脸表表情识别系统】7.【…

深入了解 Python 中的 MRO(方法解析顺序)

文章目录 深入了解 Python 中的 MRO&#xff08;方法解析顺序&#xff09;什么是 MRO&#xff1f;如何计算 MRO&#xff1f;C3 算法的合并规则C3 算法的合并步骤示例&#xff1a;合并过程解析 MRO 解析失败的场景使用 mro() 方法查看 MRO示例 1&#xff1a;基本用法 菱形继承与…

数字化赋能:制造业如何突破低效生产的瓶颈?

随着全球经济的快速发展与市场需求的变化&#xff0c;制造业面临着前所未有的压力与挑战。生产效率、资源管理、品质控制、成本控制等方面的问题日益突出&#xff0c;尤其是低效生产成为了许多制造企业亟待解决的瓶颈。在这种背景下&#xff0c;数字化转型成为提升制造业效率的…

Element-Plus,使用 El-form中 的 scroll-to-error 没有效果问题记录

因业务需要表单组件中嵌套着表格列表&#xff0c;内容比较多&#xff1b; 所以需要表单校验不通过时&#xff0c;自动定位到不通过的节点&#xff1b; 但发现这个像是没有起到效果一样&#xff0c;后面就是排查的思路了&#xff1a; 容器高度问题&#xff1a;如果表单容器的高度…

基于Javase的停车场收费管理系统

基于Javase的停车场收费管理系统 停车场管理系统开发文档 项目概述 1.1 项目背景 随着现代化城市的不断发展&#xff0c;车辆数量不断增加&#xff0c;停车难问题也日益突出。为了更好地管理停车场资 源&#xff0c;提升停车效率&#xff0c;需要一个基于Java SE的停车场管理…

网络协议 HTTP、HTTPS、HTTP/1.1、HTTP/2 对比分析

1. 基本定义 HTTP&#xff08;HyperText Transfer Protocol&#xff09; 应用层协议&#xff0c;用于客户端与服务器之间的数据传输&#xff08;默认端口 80&#xff09;。 HTTP/1.0&#xff1a;早期版本&#xff0c;每个请求需单独建立 TCP 连接&#xff0c;效率低。HTTP/1.1&…

DeepSeek掘金——调用DeepSeek API接口 实现智能数据挖掘与分析

调用DeepSeek API接口:实现智能数据挖掘与分析 在当今数据驱动的时代,企业和开发者越来越依赖高效的数据挖掘与分析工具来获取有价值的洞察。DeepSeek作为一款先进的智能数据挖掘平台,提供了强大的API接口,帮助用户轻松集成其功能到自己的应用中。本文将详细介绍如何调用D…