当前位置:首页 > 嵌入式 > 嵌入式云IOT技术圈
[导读]从这篇博文开始,我将开始手把手教会大家写Linux设备驱动程序 这是开篇,如何来写第一个字符设备驱动程序。 首先,写一个最简单的字符设备驱动程序需要什么?或者说我们需要了解什么? 1、每一个字符设备至少需要有一个设备号 2、设备号 = 主设备号 + 次设备

从这篇博文开始,我将开始手把手教会大家写Linux设备驱动程序

这是开篇,如何来写第一个字符设备驱动程序。


首先,写一个最简单的字符设备驱动程序需要什么?或者说我们需要了解什么?

1、每一个字符设备至少需要有一个设备号

2、设备号 = 主设备号 + 次设备号

3、同一类设备的主设备号一般是相同的,但不是绝对的。


那么,写一个简单的字符设备驱动程序,我们需要内核里的这几个头文件,因为我们需要调用一个基本的宏和一些基本的函数来给我们使用。

#include <linux/cdev.h>
#include <linux/kdev_t.h>
#include <linux/fs.h>

打开linux内核源代码,进入include/linux/,找到cdev.h,打开,我们会看到这个结构体:

struct cdev {

//设备模型相关的

struct kobject kobj;

//所属于哪个模块--->THIS MODULE

struct module *owner;

//利用file_operations跟用户态进行操作--->有open , read , write 等方法

const struct file_operations *ops;

//链表,将设备插入到一条链表里去

struct list_head list;

//通过设备号匹配对应的驱动

dev_t dev;

//要注册字符设备的个数

unsigned int count;

};

还会看到以下的函数:

void cdev_init(struct cdev *, const struct file_operations *);

struct cdev *cdev_alloc(void);

void cdev_put(struct cdev *p);

int cdev_add(struct cdev *, dev_t, unsigned);

void cdev_del(struct cdev *);

void cd_forget(struct inode *);

这里我们需要的就是以上的这个结构体,还有cdev_init,cdev_add,cdev_del这三个函数,其余的暂时用不着。本节暂时不会用到以上的函数,下节将会使用。


然后看到#include <linux/kdev_t.h>这个头文件,这里面有我们需要的东西:

#define MINORBITS 20

#define MINORMASK ((1U << MINORBITS) - 1)

//从设备号中取出主设备号

#define MAJOR(dev) ((unsigned int) ((dev) >> MINORBITS))

//从设备号中取出次设备号

#define MINOR(dev) ((unsigned int) ((dev) & MINORMASK))

//创建一个设备号

#define MKDEV(ma,mi) (((ma) << MINORBITS) | (mi))

我们在接下来写的这个字符设备就需要创建一个设备号,所以我们需要MKDEV这个宏,第一个参数表示主设备号,第二个参数表示次设备号。


我们知道如何去创建一个设备号,那么创建了设备号,还没有对这个设备进行注册,这时候就需要#include <linux/fs.h>这个头文件里的一个函数:

extern int register_chrdev_region(dev_t, unsigned, const char *);

既然有注册,当然就有释放,所以还需要:

extern void unregister_chrdev_region(dev_t, unsigned);

好了,有了这些基本知识,可以开始我们的第一个字符设备驱动程序的编写。


编写这个简单的字符设备需要以下步骤:

1、创建设备号

2、注册设备号

3、如何驱动模块退出的时候,我们需要注销设备的操作。


好了,开始写代码:


#include <linux/init.h>

#include <linux/module.h>

#include <linux/sched.h>

#include <linux/kernel.h>

#include <linux/cdev.h>

#include <linux/kdev_t.h>

#include <linux/fs.h>

//定义一个结构体变量,用来表示设备号--->cdev.h--->dev_t

dev_t dev_no ;

static int __init  cdev_test_init(void)

{

int ret ;

printk("HELLO KERNEL FOR CDEV!\n");

//1、创建设备号-->第一个是主设备号,第二个是次设备号

//主设备号可以通过cat /proc/devices查看,如果设备号已经被占用,则需要使用没有使用过的设备号

dev_no  = MKDEV(222,2);

//2、注册设备号

//count表示要分配多少个设备号

ret = register_chrdev_region(dev_no,1,"my_dev");

if(ret < 0){

//如果注册失败,跳转到对应的位置。

goto register_error ;

}

return 0 ;

register_error:

return ret ;

}

static int __exit cdev_test_exit(void)

{

//注销驱动-->后面写1表示从dev_no开始连续一个设备

unregister_chrdev_region(dev_no,1);

return 0 ;

}


module_init(cdev_test_init);

module_exit(cdev_test_exit);

MODULE_LICENSE("GPL");

再和以前一样,写一个Kconfig和Makefile

Kconfig

menu "4412_CDEV_DRV"

     config CDEV_TEST

          bool "cdev_test"

          default n

   help

 if you select , you can use it


endmenu

Makefile

obj-y += cdev_test.o

再到上层的driver目录下Kconfig和Makefile中添加相应的语句,跟以往一样这里是在driver目录下创建了一个4char_dev的目录。
接下来在内核根目录下make menuconfig配置相应的驱动:





将编译生成的zImage下载至开发板,打开串口调试,会看到以下log,说明驱动已经开始运行了:


接下来通过adb shell进入安卓系统的根目录下:

cat /proc/devices

我们成功的看到主设备号222的字符设备驱动my_dev已经成功装载了。






免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭