zoukankan      html  css  js  c++  java
  • 制作第一个Linux驱动程序

    学习目标:

    • 编写一个简单的Linux驱动程序,实现应用程序能通过系统调用,调用相应驱动函数,驱动程序中仅打印信息,不实现一些特定功能

     写第一个驱动程序需要以下几个步骤:

    1)写出open、read等系统调用在进入内核空间中调用的对应函数xxx_open、xxx_read

    2)定义file_operations类型结构体,用写好的xxx_open、xxx_read函数填充file_operations结构体成员

    3)写一个xxx_init函数调用register_chardev函数注册填充好的file_operations类型结构体,目的时当调用它时将相关信息告诉内核

    4)通过module_init()来修饰xxx_init入口函数

    5)写驱动的xxx_exit出口函数,调用这个unregister_chrdev()函数卸载挂入内核中的file_operations类型结构体

    6)通过module_exit()来修饰出口函数

    7)模块许可证声明, 最常见的是以MODULE_LICENSE( "GPL v2" )来声明


    1、创建一个驱动源文件first_drv.c

    代码如下:

    #include <linux/module.h>
    #include <linux/kernel.h>
    #include <linux/fs.h>
    #include <linux/init.h>
    #include <linux/delay.h>
    #include <linux/device.h>
    #include <asm/uaccess.h>
    #include <asm/irq.h>
    #include <asm/io.h>
    //#include <asm/arch/regs-gpio.h>
    //#include <asm/hardware.h>
    
    int major;
    static int first_drv_open(struct inode *inode, struct file *file); static ssize_t first_drv_read(struct file *file, char __user *buf, size_t count, loff_t *ppos); static ssize_t first_drv_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos); struct file_operations first_drv_fileop = { //---->① .owner = THIS_MODULE, /* 这是一个宏,推向编译模块时自动创建的__this_module变量 */ .open = first_drv_open, .read = first_drv_read, .write = first_drv_write, };
    /*inode表示具体文件,file结构用来追踪运行时的信息 */
    static int first_drv_open(struct inode *inode, struct file *file) { printk("first_drv_open "); /*内核的打印用printk,而不是printf函数) return 0; } static ssize_t first_drv_read(struct file *file, char __user *buf, size_t count, loff_t *ppos) { printk("first_drv_read "); return 0; } static ssize_t first_drv_write(struct file *file, const char __user *buf, size_t count, loff_t *ppos) { printk("first_drv_read "); return 0; } static int first_drv_init(void) { major = register_chrdev(0, "first_drv", &first_drv_fileop); //---->② return 0; } static void first_drv_exit(void) { unregister_chrdev(major, "first_drv"); //---->③ } module_init(first_drv_init); //---->④ module_exit(first_drv_exit); //---->⑤ MODULE_LICENSE("GPL");

    ①定义file_operations结构体,并填充相关成员函数,如open时,通过系统调用,最终会调用到.open函数指针指向函数

    ②注册file_operation结构体,将相关信息告诉内核,只有注册后file_operations内部成员才有效。register_chrdev函数第一项参数设置为0,让系统自动分配主设备号

    ③将相关信息从内核中去除

    ④修饰函数,当执行insmod时,将调用first_drv_init函数(此例中,即调用register_chrdev函数)

    ⑤修饰函数,当执行rmmod时,将调用first_drv_exit函数(此例中,即调用unregister_chrdev函数)

    2、创建一个编译驱动的Makefile

    KERN_DIR = /home/book/self_learn/01_linux_develop/02_embeded_dir/02_kernel/linux-3.4.2 #----->①
    
    all:
        make -C $(KERN_DIR) M=`pwd` modules #---->②
    
    clean:
        make -C $(KERN_DIR) M=`pwd` modules clean
        rm -rf modules.order
    
    obj-m    += first_drv.o #----->③

    ①编译驱动时依赖内核存放的目录,内核必须是已经编译好的

    ②make -C切换到内核目录,调用内核目录的Makefile;M='pwd',当用户需要以内核为基础编译一个外部模块,需要将M='pwd'加入其中,表示到驱动的目录中查找编译源代码,modules是一个参数,就是告诉内核在编译是将驱动编译成模块,不编译进内核

    ③表示将first_drv.o编译成模块

    3、执行make,编译生成first_drv.ko

    4、将first_drv.ko拷贝到跟文件系统并加载驱动模块

    执行insmod命令将first_drv.ko加载到内核

    5、手动创建设备节点

    在注册file_operations结构体时,我们选择了让系统自动分配主设备号,在创建设备节点之前,应先获取系统自动分配的主设备号。执行cat /proc/devices 命令可以看出系统自动分配主设备号

     

    系统自动分配的主设备号时252,后面跟的是设备名称,设备名称是由register_chrdev传入的第二个参数决定的。获取主设备号之后,执行 mknod /dev/first c 252 0 命令在dev目录中创建设备节点

     

    6、编写应用程序,进行驱动测试

    应用代码first_drv_test.c:

    #include <sys/types.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    
    #include <stdio.h>
    #include <stdlib.h>
    
    int main(int argc, char **argv)
    {
        int fd;
        int val = 1;
        
        fd = open("/dev/first", O_RDWR);
        if(fd == -1)
        {
            printf("can't open...
    ");
            exit(EXIT_FAILURE);
        }
        
        read(fd, &val, sizeof(val));
        
        exit(EXIT_SUCCESS);
    }

    执行arm-linux-gcc -o first_drv_test first_drv_test.c编译测试应该程序,并将编译成功的应用程序拷贝到网络文件系统中,执行应用程序

    运行成功,由打印结果可以看出,应用程序中的open函数和read函数调用了对应驱动函数中的first_open和first_read,第一个驱动程序测试成功!

    7、改进first_drv.c驱动程序

    由上述操作结果可以看出,每加载一次驱动程序都要手动在/dev目录中创将相应设备节点,这样做太麻烦了。可以使用自动创建设备节点,Linux有udev、mdev的机制,而我们的ARM开发板上移植的busybox有mdev机制,然后mdev机制会通过/sys目录中class类来找到相应类的驱动设备来自动创建设备节点 (前提需要有mdev)

    1)首先创建一个class设备类,class是C语言中的一个结构体,是软件开发的一个设备的高级视图,它抽象出低级的实现细节,然后在class类下,创建一个class_device,即类下面创建类的设备:

    static struct class *first_drv_class;
    static struct class_device    *first_drv_class_dev;

    2)在first_drv_init后面添加如下代码

    //创建类,它会在sys/class目录下创建firstdrv_class这个类
    firstdrv_class= class_create(THIS_MODULE,"firstdrv"); //在该类下创建设备 firstdrv_class_devs=class_device_create(firstdrv_class,NULL,MKDEV(major,0),NULL,"xyz");

    注意:在高版本的linux内核中class_device_create用device_create函数代替

    3)同理,在first_drv_exit中添加卸载类和设备的函数,在卸载驱动程序后,自动删除创建的设备节点

     class_device_unregister(firstdrv_class_devs);      //注销类设备,与class_device_create对应
     class_destroy(firstdrv_class);                    //注销类,与class_create对应

    4)按照上述过程,重新编译驱动程序,并将编译好的驱动程序加载到内核(注意在加载新驱动程序时,先卸载之前驱动程序,并删除手动创建的设备节点)

    由此可见,第一个简单驱动程序创建成功,值得注意的是,自动创建设备节点,必须在mdev机制使能时才能正常使用。

  • 相关阅读:
    CF1260F
    牛客挑战赛34 A~E
    CSP-S2019游记&拆塔记
    6424. 【NOIP2019模拟2019.11.13】我的订书机之恋
    CF1257E/F
    6423. 【NOIP2019模拟11.11】画
    1222/2516. Kup
    Comet OJ
    浅析CSS定位
    css文字颜色渐变的3种实现
  • 原文地址:https://www.cnblogs.com/053179hu/p/13336279.html
Copyright © 2011-2022 走看看