Linux内核驱动开发-字符设备驱动框架

ops/2024/9/24 13:13:57/

1前置条件

(1)【linux】内核编译结束
(2)【linux】目录配置跳转文件:

在这里插入图片描述

补充:配置的跳转文件只能在【linux】目录下使用,子目录无法使用

2驱动框架

2.1编写驱动程序

#include <linux/init.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/fs.h>#define DEVICE_MAJOR 200
#define DEVICE_NAME  "first_device"int first_driver_open(struct inode *node, struct file *fp)
{printk("first_driver_open\n");return 0;
}ssize_t first_driver_read(struct file *fp, char __user *user_buffer, size_t len, loff_t *offset)
{printk("first_driver_read\n");return 0;
}ssize_t first_driver_write(struct file *fp, const char __user *user_buffer, size_t len, loff_t *offset)
{printk("first_driver_write\n");return 0;
}int first_driver_close(struct inode *node, struct file *fp)
{printk("first_driver_close\n");return 0;
}static struct file_operations fops = 
{.owner = THIS_MODULE,.open = first_driver_open,.read = first_driver_read,.write = first_driver_write,.release = first_driver_close
};static int __init first_driver_init(void)
{	register_chrdev(DEVICE_MAJOR, DEVICE_NAME, &fops);printk("first_driver_init OK\n");return 0;
}static void __exit first_driver_exit(void)
{	unregister_chrdev(DEVICE_MAJOR, DEVICE_NAME);printk("first_driver_exit OK\n");
}module_init(first_driver_init);
module_exit(first_driver_exit);MODULE_LICENSE("GPL");
MODULE_AUTHOR("XXX");

2.2 编译驱动程序

2.2.1编译方式1

【step1】:将编写完成的驱动程序放至【linux/drivers/char】目录下
【cp ./file_name.c linux/drivers/char】
【step2】:修改内核配置文件
【vi linux/drivers/char/Kconfig】

在这里插入图片描述

【step3】:修改编译规则,添加工程文件
【vim linux/drivers/char/Makefile】

在这里插入图片描述

在这里插入图片描述

【step4】:菜单配置
【make menuconfig】

在这里插入图片描述

在这里插入图片描述

修改为<M>模式

在这里插入图片描述
附:3种模式讲解

当然,也可以使用【help】选项查看相关信息

在这里插入图片描述
在这里插入图片描述

【step5】:调用编译命令,进行驱动程序的编译
【make modules】	

在这里插入图片描述

至此,采用方式1编译完成。当然,也可以查看编译后文件是否存在:

在这里插入图片描述

这是一个需要手动添加到操作系统的文件。

2.2.2编译方式2

【step1】:创建一个文件夹:是linux目录下的任意目录
【mkdir my_drivers】

在这里插入图片描述

【step2】:将驱动程序源码放入该目录

在这里插入图片描述

【step3】:该目录下创建【Makefile】文件
【vim Makefile】
KERNELDIR := /home/linux/s3c2440/linux-2.6.32.2/
CURRENT_PATH := $(shell pwd)
obj-m := first_driver.obuild: kernel_moduleskernel_modules:$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) modulesclean:$(MAKE) -C $(KERNELDIR) M=$(CURRENT_PATH) clean
【step4】:编译
【make】

在这里插入图片描述

至此,采用方式2编译完成。当然,也可以查看编译后文件是否存在:

在这里插入图片描述

这是一个需要手动添加到操作系统的文件。

2.3移植驱动程序

(1)将编译产生的【filename.ko】文件移植到开发板
【cp drivers/char/first_driver.ko ~/nfs/rootfs】

在这里插入图片描述

在开发板种可以看到文件移植成功:

在这里插入图片描述

2.4加载驱动程序

(1)加载被移植到开发板的【filename.ko】文件
【insmod filename.ko】

在这里插入图片描述

补充:
查看加载成功的驱动:【lsmod】

2.5卸载驱动程序

(1)卸载被移植到开发板的【filename.ko】文件【rmmod filename】

2.6创建设备文件

2.6.1创建设备文件方式1

手动创建设备文件
【mknod[options] name type major minor】
【options】:
【name】:【/dev】下的设备文件名
【type】:设备型号,【c】
【major】:主设备号
【minor】:子设备号
eg:【mknod /dev/first_driver c 200 0】

在这里插入图片描述

设备创建完毕,可以使用【ls】命令查看是否成功创建设备文件。

在这里插入图片描述

补充:
应用程序就是以【name】为入口点来调用驱动程序的。
【mknod ?】:查看命令帮助
【cat /proc/devices/】:查看设备
【ls /dev】:查看设备文件删除设备文件:

在这里插入图片描述

2.7调用驱动程序

【step1】:其实就是在开发板上编写一个应用程序(主要是指main函数)
编写应用程序:打开设备文件->调用文件I/O(系统I/O)->
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>int main(void)
{int fd=0;fd=open("dev/first_device",O_RDWR);if(fd<0){printf("opening is error\n");return -1;}while(1){int n=123;read(fd,&n,4);sleep(1);write(fd,&n,4);sleep(1);}return 0;
}
【step2】:编译应用程序

在这里插入图片描述

【step3】:关闭开发板原有的设备驱动程序
通过菜单配置命令:【make menuconfig】
将对应的驱动模块的模式【<>】改为图中所标记的模式。

在这里插入图片描述

2.8编译并移植内核文件

【make uImage】

在这里插入图片描述

编译完成之后,将产生的镜像文件移植到开发板:
【cp arch/arm/boot/uImage ~/tftpboot/】

在这里插入图片描述

2.9运行操作系统内核

接下来,将操作系统运行在开发板上:
【tftp 0x30008000 uImage】
【bootm 0x30008000】

2.10开发板文件配置

当新的内核文件在开发板上成功运行之后,接下来需要对开发板上的一个文件进行配置:

在这里插入图片描述

关闭led相关配置:

在这里插入图片描述

补充:
(1)【reboot】:该命令将重启开发板,让开发板回到初始状态,从运行内核文件之后的所有操作需要重新配置。

在这里插入图片描述

(2)通过配置环境变量可以让开发板上电以后自动运行操作系统并挂载根目录:
【setenv bootcmd 'tftp 0x30008000; bootm 0x30008000'】
注意:环境变量配置完成一定要进行保存操作!!!

在这里插入图片描述

(3)环境变量配置完成之后可以重启开发板:可以看到开发板将自动运行操作系统并挂载根文件系统。
【reset】:重启开发板

在这里插入图片描述

3板载驱动程序示例

路径:【linux-2.6.32.2/drivers/char/mini2440_hello_module.c 】

#include <linux/kernel.h>
#include <linux/module.h>static int __init mini2440_hello_module_init(void)
{printk("Hello, Mini2440 module is installed !\n");return 0;
}static void __exit mini2440_hello_module_cleanup(void)
{printk("Good-bye, Mini2440 module was removed!\n");
}module_init(mini2440_hello_module_init);
module_exit(mini2440_hello_module_cleanup);
MODULE_LICENSE("GPL");

http://www.ppmy.cn/ops/13103.html

相关文章

JPA Example 默认 join

起因&#xff1a; 由于同事增加了一个对象关联。并且采用了Example查询&#xff0c;导致了一个Null的问题。 主表&#xff1a;BoBookingorder 关联表&#xff1a;BoJobcontainerinfo 一 关联如下&#xff1a; Entity Table(name "bo_bookingorder",catalog &quo…

阿斯达年代记三强争霸账号怎么注册 游戏账号注册教程分享

即将于4月24日隆重推出的《阿斯达时代&#xff1a;三巨头对决》这款大规模多人在线角色扮演游戏巨制&#xff0c;是由Netmarble公司携手STUDIO DRAGON联手创作。游戏中&#xff0c;围绕着阿斯大陆的主权掌控&#xff0c;三个主要阵营——阿斯达、亚高及非法者组织之间展开了扣人…

比赛记录:Codeforces Round 940 (Div. 2) and CodeCraft-23 A~E

传送门:CF [前题提要]:感觉这场题目其实都很经典.遗憾的是赛时C题答案统计看成了不同的下棋的方案数,然后以为刚开始不能放不是一种答案(直接特判输出了0),卡了一场比赛.幸好最后15min险过CD,不然掉大分了 A. Stickogon 难点在于读懂题意. 读懂题意之后不难发现每根木棒都形成…

【机器学习】科学库使用第5篇:Matplotlib,学习目标【附代码文档】

机器学习&#xff08;科学计算库&#xff09;完整教程&#xff08;附代码资料&#xff09;主要内容讲述&#xff1a;机器学习&#xff08;常用科学计算库的使用&#xff09;基础定位、目标&#xff0c;机器学习概述定位,目标,学习目标,学习目标,1 人工智能应用场景,2 人工智能小…

python Django中分配库存给用户包括定义库存模型、用户模型、以及一个用于分配库存的逻辑

在Django中分配库存给用户通常涉及几个步骤&#xff0c;包括定义库存模型、用户模型、以及一个用于分配库存的逻辑。以下是一个基本的示例来说明如何执行这个过程&#xff1a; 1. 定义模型 首先&#xff0c;你需要定义两个模型&#xff1a;一个是User模型&#xff08;可以使用…

(delphi11最新学习资料) Object Pascal 学习笔记---第10章第1节(属性的代码补全)

10.1.3 属性的代码补全 ​ 给类添加属性是一项繁琐的工作&#xff0c;IDE的编辑器可以让你在编写属性声明的初始部分&#xff08;在类内部&#xff09;时轻松自动完成属性&#xff0c;如下所示&#xff1a; typeTMyClass classpublicproperty Month : Integer;end;​ 在光标…

selenium_鼠标_点击、移动、输入

selenium_鼠标_点击、移动、输入 # 导包from selenium import webdriver from time import sleep# 获取浏览器驱动对象 from selenium.webdriver.common.action_chains import ActionChains from selenium.webdriver.common.by import By from selenium.webdriver.common.keys…

【Leetcode每日一题】 动态规划 - 地下城游戏(难度⭐⭐⭐)(61)

1. 题目解析 题目链接&#xff1a;174. 地下城游戏 这个问题的理解其实相当简单&#xff0c;只需看一下示例&#xff0c;基本就能明白其含义了。 2.算法原理 一、状态表定义 在解决地下城游戏问题时&#xff0c;我们首先需要对状态进行恰当的定义。一个直观的想法是&#x…