【Linux开发】linux设备驱动归纳总结(十):1.udev&misc

178 篇文章 37 订阅 ¥39.90 ¥99.00
linux 设备驱动归纳总结(十): 1.udev&misc


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

不知不觉我的总结已经写得七七八八了,这一章节只是补充一下两个知识点:动态创建设备节点和杂设备类的注册。

xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


一、动态创建设备节点——udev


之前加载字符型设备后是通过命令mknod来创建设备节点的。在2.6内核中,有一个名叫udev的后台程序,它通过读取/sys/class的信息,一旦添加的新的设备,该后台程序就会自动创建设备节点。


一、要使用动态创建设备节点,首先要运行udev后台程序。


在嵌入式下有两种方法:

1、移植udev到嵌入式系统中。

2、在编译busybox时加入mdev

mdev可以理解是udev的精简版,在这里我就不介绍如何加入mdev,仅仅介绍udev的移植。

首先:进入到udev的目录,我在附件中提供了udev的源文件:

root@xiaobai-laptop:/nfsroot/review_driver/10th_udev_misc/udev/tools/udev-096# pwd

/nfsroot/review_driver/10th_udev_misc/udev/tools/udev-096

接着修改Makefile的两处:

line 96 prefix = /nfsroot //指定udev安装路径,安装到我们的跟文件系统中

linr 114 CROSS_COMPILE = arm-linux- //指定编译工具

然后编译并安装:

make

make install

最后在跟文件系统的/nfsroot/etc/init.d/rcS加上三句话,让系统启动后运行udev后台程序:

mount -t tmpfs tmpfs /dev

/sbin/udevd -d

/sbin/udevstart

移植udev后,系统启动是就会自己启动udev程序。


二、接着就是在代码中使用动态创建函数了。


udev的动态创建是通过读取/sys/class的信息来实现的。所以,首先是要创建一个类:

struct class *class_create(struct module *owner, const char *name)

owner用于指定该类的所属,一般填写THIS_MODULE

name是指定该类该/sys/class/目录下的目录名字。

同样的,该操作有可能出错,需要检验返回值。

类创建成功后会在/sys/class创建一个以name命名的目录。


类的注销使用以下函数:

void class_destroy(struct class *cls)


创建类后,就可以调用以下函数注册类设备,将设备加入到指定的类中,这样udev就可以才class目录下读取信息后动态创建设备节点了:

/*drivers/base/core.c*/

1386 struct device *device_create(struct class *class, struct device *parent,

1387 dev_t devt, void *drvdata, const char *fmt, ...)

class是用于指定所属的class

parent用于指定设备的父设备,一般填NULL就可以了。

Devt用于指定该设备的设备号。

drvdata用于指定class下的数据,一般也不用传。

fmt就是动态创建的设备文件的名字,一般格式:"%s", "test_led"

设个函数成功调用会在/sys/devices/virtual的对应总线下创建一个test_led的目录,并且软连接到/sys/class对应的总线目录下。


使用以下函数注销类设备:

void device_destroy(struct class *class, dev_t devt)


注意:上面的两步并没有真正创建设备文件,只有在udev读取/sys/class信息后发现tesl_led,才会真正创建设备文件。


先来来个代码,这是我之前写的简洁版led驱动,添加了动态创建设备文件功能:

/*10th_udev_misc/udev/1st_udev/test.c*/

7 #include //需要包含该头文件

10 struct class *my_class;

。。。。。

62 int test_init(void)

63 {

64      //devmodel class create

65      my_class = class_create(THIS_MODULE, "test_class"); //创建class

66      if(IS_ERR(my_class)){

67           printk("create cleaa wrong!\n");

68           return -1;

69      }

70      device_create(my_class, NULL, MKDEV(253, 0), NULL, "%s", "test_led"); //根据class和设备号创建设备

71

72      register_chrdev(253, "udev led", &led_fops);

73      led_init(); //ioremap的地址映射

74      led_config(); //gpio_led的配置

75

76      return 0;

77 }

78 void test_exit(void)

79 {

80      iounmap((void *)virt); //注销虚拟地址的映射

81      unregister_chrdev(253, "udev led"); //注销设备节点

82      device_destroy(my_class, MKDEV(253, 0)); //注销类设备

83      class_destroy(my_class); //注销类

84      printk("bye\n");

85 }


接下来验证一下:

[root: 1st_udev]# insmod test.ko //加载驱动

[root: 1st_udev]# ./app on //灯亮

[root: 1st_udev]# ./app off //灯灭

[root: 1st_udev]# cat /proc/devices //查看一下

Character devices:

1 mem

。。。。。

136 pts

204 s3c2410_serial

253 udev led //设备被注册了

254 rtc

[root: 1st_udev]# cd /

[root: /]# find -name "*test_led*" //查找test_led相关文件

./sys/devices/virtual/test class/test_led ///device/virtule目录下有test_led目录

./sys/class/test class/test_led //并且有类test_class

./dev/test_led //动态创建了test_led设备文件

[root: 1st_udev]# ls -l /sys/class/test\ class/test_led

lrwxrwxrwx 1 root root 0 Jan 1 08:29 /sys/class/test class/test_led -> ../../devices/virtual/test class/test_led


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


二、杂设备——misc


简单的说,杂设备就是内核自动帮你分配设备号并且自动创建设备文件。

1、自动分配设备号,是指所有注册为杂设备的设备的主设备号为10,而次设备号内核自动分配。

2、自动创建设备文件是指,内核会使用udev(前提是你已经移植udev),动态创建设备节点。


方法很简单:需要包含头文件:linux/miscdevice.h
1
、定义杂设备结构体:

36 struct miscdevice {

37      int minor;

38      const char *name; //设备文件的名字

39      const struct file_operations *fops; //指定该设备的fops结构体

40      struct list_head list;

41      struct device *parent;

42      struct device *this_device;

43 };

红色标记的内容是我们必须要自己填写的,其他部分可以由内核自己分配。


2、定义结构体后使用使用一下函数注册和注销:

int misc_register(struct miscdevice * misc); //注册

int misc_deregister(struct miscdevice *misc); //注销

注册操作会失败,建议检查返回值。


看代码:

/*10th_udev_misc/misc/1st_misc/test.c*/

58 static struct file_operations led_fops = {

59      .ioctl = led_ioctl,

60 };

61

62 static struct miscdevice misc_led = { //杂设备

63      .name = "test_led",

64      .fops = &led_fops,

65 };

66

67 int test_init(void)

68 {

69      misc_register(&misc_led); //注册杂设备

70

71      led_init(); //ioremap的地址映射

72      led_config(); //gpio_led的配置

73

74      return 0;

75 }

76 void test_exit(void)

77 {

78      misc_deregister(&misc_led); //注销杂设备

79      iounmap((void *)virt); //注销虚拟地址的映射

80      printk("bye\n");

81 }

再看效果

[root: 1st_misc]# insmod test.ko

[root: 1st_misc]# cat /proc/devices

Character devices:

1 mem

4 /dev/vc/0

4 tty

5 /dev/tty

5 /dev/console

5 /dev/ptmx

7 vcs

10 misc //创建的设备在misc

[root: 1st_misc]# lsmod

test 2212 0 - Live 0xbf000000

[root: 1st_misc]# ls -l /dev/test_led

crw-rw---- 1 root root 10, 0 Jan 1 08:05 /dev/test_led //自动创建的设备节点主设备号为10

[root: 1st_misc]# ./app on //亮灯

[root: 1st_misc]# ./app off //灭灯

[root: 1st_misc]# cd / //查看test_led相关文件

[root: /]# find -name "*test_led*"

./sys/devices/virtual/misc/test_led

./sys/class/misc/test_led //发现注册到misc类中

./dev/test_led


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx


三、总结


这节介绍了如何动态创建设备文件和如果创建杂设备类驱动。


xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx

源代码: 10th_udev_misc.rar   

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Linux字符设备驱动开发是指在Linux系统中编写驱动程序,使得用户可以通过字符设备接口来访问硬件设备。这种驱动程序通常用于控制串口、并口、USB设备等。开发Linux字符设备驱动需要掌握Linux内核的基本知识,包括进程管理、内存管理、中断处理、设备驱动等方面。此外,还需要了解字符设备驱动的编写流程、驱动程序的结构和接口等。开发Linux字符设备驱动需要使用C语言和Linux内核编程接口。 ### 回答2: Linux字符设备驱动开发Linux系统中的一部分,它允许开发人员在Linux系统上使用字符设备,这些字符设备可以包括串口、USB口、网卡等。Linux字符设备驱动开发可帮助开发人员实现各种各样的设备驱动,从而增强Linux系统的功能。 在Linux字符设备驱动开发过程中,需要注意以下几点: 1. 实现设备驱动的一个基本框架,包括注册设备设备的初始化,以及对设备进行读写操作等。 2. 开发人员不仅需要熟悉驱动程序开发技术,还需要了解Linux内核系统的相关知识,例如进程、中断、内存管理等。 3. 应该在代码注释中提供详细的文档,以方便其他开发人员进行维护和修改。 4. 在实现字符设备驱动过程中,必须保证安全性和可靠性,防止设备出现故障或者损坏用户的数据。 5. 在测试和维护设备驱动时,需要使用一些常见的工具和技术,例如devfs、udev等。 总之,Linux字符设备驱动开发是一个需要熟练技能和丰富经验的过程。开发人员需要有足够的专业知识和经验来确保设备驱动的高效和稳定性。通过精心设计和开发Linux字符设备驱动可以提供高性能、高可靠性、易于使用的设备驱动,从而大大增强了Linux系统的功能和灵活性。 ### 回答3: Linux字符设备驱动开发Linux系统中的一个重要领域。其主要任务是开发一些支持字符设备驱动程序,从而使用户能够在Linux系统中使用各种不同类型的字符设备,例如串口、打印机、读卡器和磁盘等。同时,这些驱动程序还要保证设备完全可靠和高效地工作,确保系统的安全性和性能。 Linux字符设备驱动开发需要掌握以下基本知识: 1.了解Linux系统体系结构和内核架构 Linux系统由内核和用户空间组成,内核作为系统的核心组件,是实现系统功能的主要部分,因此了解内核体系结构和架构是开发Linux字符设备驱动所必须掌握的知识。 2.熟悉字符设备的相关知识 字符设备Linux系统中的一种重要的设备类型,它与其他类型设备不同之处在于它只能逐个字符地进行读写操作。因此需要深入了解字符设备的相关知识,例如驱动的主要功能、驱动程序与设备的交互方式、设备控制结构等。 3.熟练掌握C语言及Linux内核编程技术 编写Linux字符设备驱动程序需要掌握良好的C语言编程知识以及熟练的Linux内核编程技术,包括内存管理、进程管理、文件系统、中断处理等。同时,还需要了解Linux内核代码的结构和代码的编写规范,以便于编写出符合内核标准的驱动程序。 4.掌握Linux驱动框架的使用方法 为了简化Linux驱动开发流程,Linux提供了一些驱动框架,这些框架定义了一些驱动程序中常用的接口和函数,能够方便驱动程序的开发和调试。因此,Linux字符设备驱动开发者需要掌握其中的一些驱动框架,如字符驱动框架。 5.熟悉Linux字符设备驱动开发过程 Linux字符设备驱动开发过程主要包括驱动程序的初始化、驱动程序的主要功能实现、驱动程序的卸载等环节。在开发过程中,需要合理使用系统提供的工具和调试手段,如gdb、strace、make等,以便于分析和排查驱动程序出现的问题,确保驱动程序的稳定和可靠性。 总之,在Linux字符设备驱动开发过程中,开发者需要掌握相关的知识和技能,以实现对字符设备的编程和调试,开发出满足用户需求的高质量驱动程序。同时,Linux字符设备驱动开发也是一项长期持续的工作,开发者需要时刻关注最新的技术发展和硬件设备变化,才能更好地适应市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值