• mini2440的LEDS驱动程序和测试程序详解


    mini2440的LEDS驱动程序和测试程序详解

     

    http://hi.baidu.com/760159/blog/item/75c225f3dea26d19b17ec525.html

    一 leds的驱动程序

    位置:linux 2.6.29/drivers/char/mini2440_leds.c

    #include <linux/miscdevice.h>
    #include <linux/delay.h>

    #include <asm/irq.h>
    #include <mach/regs-gpio.h>//具体头文件位置/opt/FriendlyARM/mini2440/linux-2.6.29/arch/arm/mach-s3c2410/include/mach/*.h

    #include <mach/hardware.h>
    #include <linux/kernel.h>
    #include <linux/module.h>//具体的头文件位置为/opt/FriendlyARM/mini2440/linux-2.6.29/include/linux/*.h 

    #include <linux/init.h>
    #include <linux/mm.h>
    #include <linux/fs.h>
    #include <linux/types.h>
    #include <linux/delay.h>
    #include <linux/moduleparam.h>
    #include <linux/slab.h>
    #include <linux/errno.h>
    #include <linux/ioctl.h>
    #include <linux/cdev.h>
    #include <linux/string.h>
    #include <linux/list.h>
    #include <linux/pci.h>
    #include <asm/uaccess.h>
    #include <asm/atomic.h>
    #include <asm/unistd.h>


    #define DEVICE_NAME "leds" //定义驱动程序的名字为leds

    static unsigned long led_table [] = {
    S3C2410_GPB5,
    S3C2410_GPB6,
    S3C2410_GPB7,
    S3C2410_GPB8,
    };                                                   //定义引脚的寄存器数组(无符号长整形,对应于引脚的地址)

    static unsigned int led_cfg_table [] = {
    S3C2410_GPB5_OUTP,
    S3C2410_GPB6_OUTP,
    S3C2410_GPB7_OUTP,
    S3C2410_GPB8_OUTP,
    };                                               //定义引脚功能,为输出(无符号整形)

    static int sbc2440_leds_ioctl(
    struct inode *inode, 
    struct file *file, 
    unsigned int cmd, 
    unsigned long arg)
    {
    switch(cmd) {
    case 0:
    case 1:
       if (arg > 4) {                                 //设备节点,文件描述符,LED灯编号,LED灯状态四个命令参数
        return -EINVAL;
       }
       s3c2410_gpio_setpin(led_table[arg], !cmd);
       return 0;
    default:
       return -EINVAL;                           //EINVAL:表示向函数传递了无效的参数(errno符号变量)
    }
    }

    //初始化字符设备驱动的file_operations 的结构体

    static struct file_operations dev_fops = {
    .owner = THIS_MODULE,
    .ioctl = sbc2440_leds_ioctl,
    };

    static struct miscdevice misc = {
    .minor = MISC_DYNAMIC_MINOR,                    /* 动态设备号 */
    .name = DEVICE_NAME,                                  /* 将在/dev目录生成led设备 */
    .fops = &dev_fops,                                            /* 驱动接口 */

    };

    static int __init dev_init(void)
    {
    int ret;

    int i;

    for (i = 0; i < 4; i++) {

    /*设置GPIO对应的配置寄存器GPIOCON为输出状态*/

       s3c2410_gpio_cfgpin(led_table[i], led_cfg_table[i]);

    /*设置GPIO对应的数据寄存器GPIODAT为低电平,在模块加载结束后,四个LED应该是全部都是发光状态*/

    s3c2410_gpio_setpin(led_table[i], 0);
    }

    //注册设备

    ret = misc_register(&misc);

    printk (DEVICE_NAME"\tinitialized\n");

    return ret;
    }

    //注销设备驱动

    static void __exit dev_exit(void)
    {
    misc_deregister(&misc);
    }

    module_init(dev_init);                                              /*声明加载模块初始化函数*/
    module_exit(dev_exit);                                           /*声明卸载模块清除函数*/
    MOUDLE_LICENSE("GPL");                                     /*许可证声明*/
    MODULE_AUTHOR("FriendlyARM Inc.");                /*作者信息*/


    1 static 关键字的重要性

          全局变量和函数全部用static 进行修饰,则其作用的范围仅仅限于当前的文件,而不是整个系统。防止编译器在连接时,会报告命名错误的“名字空间污染”的问题。

    2 ioctl()函数

    static int sbc2440_leds_ioctl(struct inode *inode, struct file *file,   unsigned int cmd, unsigned long arg)

          ioctl函数是文件结构中的一个属性分量。ioctl是设备驱动程序中对设备的I/O通道进行管理的函数。所谓对I/O通道进行管理,就是对设备的一些特性进行控制,例如串口的传输波特率、马达的转速等等。
         struct inode *inode,是设备节点号。fd就是用户程序打开设备时使用open函数返回的文件标示符,cmd就是用户程序对设备的控制命令,unsigned long arg是控制命令的个数。
          驱动程序提供了对ioctl的支持,用户就可以在用户程序中使用ioctl函数控制设备的I/O通道。如果函数返回一个非负值,那么该值会被返回给调用程序,表示成功。韩式一般通过switch{case}对设备的一些特性进行控制。switch{case}结构,每一个case对应一个命令码,做出一些相应的操作。在本例中的cmd有两个可选项0和1.0表示灯灭,1表示灯亮。所以case 0,1都要进行操作。由于实际的硬件连接中,是低电平灯亮。所以在对引脚赋值时要取反。 s3c2410_gpio_setpin(led_table[arg], !cmd)

    3 static int __init dev_init(void)
    _init 宏,定义在include/linux/init.h中。对于非模块加载的驱动程序,通过_init 宏,会把函数中的代码放到.text.init段。这个段在系统启动后会被释放。这样函数代码只有在启动时执行一次,所以可以释放它们以节省内存空间,

    3初始化字符设备驱动的file_operations 的结构体

    结构体file_operations在头文件 linux/fs.h中定义,用来存储驱动内核模块提供的对 设备进行各种操作的函数的指针。该结构体的每个域都对应着驱动内核模块用来处理某个被请求的 事务的函数的地址。

    4ret = misc_register(&misc);

    misc_register()用主编号10调用 register_chrdev(),设备名称和函数表指针通过miscdevice数据结构获得。同样,miscdevice 数据结构还保存设备驱动程序所使用的次要号码。完成设备的注册。

    5 printk()

    利用 printk可以实现内核到Linux 控制台的格式化输出。其用法与标准C的printf类似。在调用驱动程序时,依靠printk输出信息跟踪程序,是很有效的方法。与标准C的printf 不同的是,printk支持分级输出。默认为第四级的输出KERN_ERR。

    二 LED测试程序

    /opt/FriendlyARM/mini2440/examples/leds

    #include <stdio.h>                              /*下面函数要用到的头文件*/
    #include <stdlib.h>
    #include <unistd.h>
    #include <sys/ioctl.h>

    int main(int argc, char **argv)                 /*运行时参数传递,开或关哪个LED*/
    {                                               
    int on;                                                   /*定义led状态变量,1表示灯亮,2表示灯灭*/
    int led_no;                                           /*定义led变量--哪个led*/
    int fd;                                                     /*定义led设备文件描述符的变量*/
    if (                         argc != 3 || \             /*判断命令输入参数个数*/
            sscanf(argv[1], "%d", &led_no) != 1 || \                   /* 第一个字符串参数表示要操作led*/

    sscanf(argv[2],"%d", &on) != 1 || \                                  /*第2个字符串参数作为LED状态*/
                           on < 0 || on > 1 || \                                           /*开和关,两个状态*/
                    led_no < 0 || led_no > 3 ) \                                     /*4个LED*/
    {
    fprintf(stderr, "Usage: leds led_no 0|1\n");                       /*如果条件不满足输出出错信息*/
    exit(1);                                                                               /*退出程序,返回1,表示出现错误*/
    }
    fd = open("/dev/leds0", 0);                   /*为只读打开leds0文件,取出文件描述符*/
    if (fd < 0) {                                 
       fd = open("/dev/leds", 0);                  /*如果打开leds0出错,再以只读方式打开leds文件*/
    }
    if (fd < 0) {
       perror("open device leds");                  /*如果打开led文件出错,拿不到文件描述符,用perror宏输出错原因及信息*/
    exit(1);                                                /*出错退出*/
    }
    ioctl(fd, on, led_no);                          /*用ioctl()函数控制LED,其中fd--是前面打开的LED文件描述符,on--是开关命令0和1,led_no--是哪个LED*/
    close(fd);                                           /*关闭LED描述符,与前面open()对应*/
    return 0;                                            /*正常返回*/
    }

    说明:sscanf() - 从一个字符串中读进与指定格式相符的数据. sscanf与scanf类似,都是用于输入的,只是后者以键盘(stdin)为输入源,前者以固定字符串为输入源。其中的format可以是一个或多个 {%[*] [width] [{h | l | I64 | L}]type | ' ' | '\t' | '\n' | 非%符号}


    sscanf(argv[1], "%d", &led_no) != 1 || sscanf(argv[2],"%d", &on) != 1 ||

    表示从argv[1](argv[2])读字符,并转换成整形给led_no(on)。             

    三 Makefile

    CROSS=arm-linux-                            #定义变量CROSS

    all: led

    led: led.c                                          #定义led的规则
    $(CROSS)gcc -o led led.c              #这句把CROSS变量替换,可以还原成arm-linux-gcc -o led led.c

    clean:
    @rm -vf led *.o *~                              #rm指令可删除 -v表示显示指令执行过程 -f表示强制文件或目录

  • 相关阅读:
    NSDate相差8小时
    IOS 多播委托(GCDMulticastDelegate)
    点云、矢量、三维模型数据的平移
    mongodb集群搭建过程记录
    mongodb 使用常见问题汇总(主要是集群搭建)
    安装py3ditles中遇到的问题
    linux常见问题集锦
    Ubuntu18.04 + win10双系统下时间问题
    Frank Dellaert Slam Speech 20190708
    计算机系统基础学习笔记及博客
  • 原文地址:https://www.cnblogs.com/hnrainll/p/2086566.html
Copyright © 2020-2023  润新知