Advertisement

【笔记】Linux驱动学习第三章

阅读量:

作者:Exculivor
日期:2015年06月29日

前面我们学习了如何写简单的驱动以及简单的测试软件。
今天我们来正式的写一个能够使用的简单驱动以及对应的测试软件。
注:目标平台为RT5350,使用Linux 3.18.16内核,OpenWrt版本r46104。其他平台的方法大同小异,可以按照步骤结合相应的数据手册、原理图以及配套平台进行。

    • 本次学习内容

    • 设备驱动开发

      • 第一步 了解硬件特点及操作方法
    • LED发光以及控制原理

    • RT5350的IO口控制方法

      • 第二步 寻找相似驱动模板修改或者自行编写
    • 一 设备操作函数编写

    • 二 设备初始化

    • 完整代码

    • Makefile

      • 第三步 编写测试程序测试驱动
    • 测试程序

    • Makefile

    • 小结

本次学习内容

  • 设备驱动开发
  • 小结

设备驱动开发

开始之前我们学回顾一下开发设备驱动的一般步骤:

  • 查看原理图、数据手册,了解设备的操作方法。
  • 在内核中找到相近的驱动程序,以它为模板进行开发,有时候需要从零开始。
  • 实现驱动程序的初始化:比如向内核注册这个驱动程序,这样应用程序传入文件名时,内核才能找到相应的驱动程序。
  • 设计所要实现的操作,比如open、close、read、write等函数。
  • 实现中断服务(中断并不是每个设备驱动所必须的)。
  • 编译该驱动程序到内核中,或者用insmod命令加载。
  • 测试驱动程序。

下面我们简化这个步骤来写一个硬件学习中的“Hello World”——LED驱动!

第一步 了解硬件特点及操作方法

LED发光以及控制原理

LED,即发光二极管。是一种通过PN结电子与空穴复合时释放辐射发出可见光为发光原理的电子器件。其本质就是一种特殊的二极管
因此LED在电路中的符号通常表示如下:
LED
当LED正向偏置,即正极(图中引脚1)电位高于负极(图中引脚2)电位时,电流从正极流向负极,LED便可发光。当然,要点亮不同的LED,需要考虑不同LED的特点,选择合适的电压电流,这样LED才能正常工作。
在下使用的开发板给出了原理图,其中LED的连接关系如下:
LED连接关系
以LED1为例,当LED1正极即GPIO#7为高电平(根据RT5350的数据手册可知其I/O口电压为3.3V)时,LED1可正向导通,电流流过LED1,LED1点亮。
而当GPIO#7一端为低电平(0V)时,LED1两端电位相同,LED1此时关断,没有正向电流流过。因此LED1熄灭。
因此,我们可以根据上述原理,通过控制RT5350的IO口——GPIO#7的电平来控制LED1的亮灭。
同理,其他三个LED也是如此。

RT5350的I/O口控制方法

通过阅读RT5350的数据手册我们可以知道其I/O口特点:

I/O Features
I/O Features

下面贴出I/O口的方框图:

Block Diagram
block diagram

RT5350的I/O口都是多功能复用的,我们可以通过控制SYSCFGGPIOMODE两个寄存器来控制这些引脚的功能:

RT5350_GPIO_Share_Scheme
RT5350_GPIO_Share_Scheme

我们使用的GPIO7~GPIO10是和UARTF引脚复用的。根据下面这张表:

UART_Pin_Share_Scheme
UART_Pin_Share_Scheme

可以确定我们将要使用到的方案。
此次我们只需要使用GPIO7、GPIO8、GPIO9、GPIO10四个I/O口,没有用到UARTF的功能。因此通过查表可以知晓我们可以通过在相应寄存器中写入二进制100或者111数据来实现。
接下来我们来分析I/O口功能设置的相关寄存器:

  • I/O模式寄存器

GPIOMODE :
I/O模式寄存器
基地址:0x1000_0000
偏移地址:0x0060

以下I/O口的设置相关寄存器基地址 为:0x1000_0600

  • I/O中断状态寄存器

GPIO21_00_INT :
I/O中断状态寄存器
偏移地址:0x0000

GPIO27_22_INT:
I/O中断状态寄存器
偏移地址:0x0060

  • I/O边沿状态寄存器

GPIO21_00_EDGE:
I/O边沿状态寄存器
偏移地址:0x0004

GPIO27_22_EDGE:
I/O边沿状态寄存器
偏移地址:0x0064

  • I/O上升沿中断使能寄存器

GPIO21_00_RENA:
I/O上升沿中断使能寄存器
偏移地址:0x0008

GPIO27_22_RENA:
 I/O上升沿中断使能寄存器
偏移地址:0x0068

  • I/O下降沿中断使能寄存器

GPIO21_00_FENA:
I/O下降沿中断使能寄存器
偏移地址:0x000C

GPIO27_22_FENA:
I/O下降沿中断使能寄存器
偏移地址:0x006C

  • I/O数据寄存器

GPIO21_00_DATA:
I/O数据寄存器
偏移地址:0x0020

GPIO27_22_DATA:
I/O数据寄存器
偏移地址:0x0070

  • I/O方向寄存器

GPIO21_00_DIR:
I/O方向寄存器
偏移地址:0x0024

GPIO27_22_DIR:
 I/O方向寄存器
偏移地址:0x0074

另外还有几个不常用寄存器,有需要可以查看数据手册。

首先,要点亮这几个LED,我们确定了要使用GPIO7~10。因此需要通过GPIOMODE寄存器设置引脚模式工作在GPIO模式。
而点亮LED我们并不需要从I/O口读取数据,只需要输出高低电平即0或者1就可以了。所以我们需要GPIO21_00_DIR这个寄存器来设置I/O方向为输出。
设置完方向,我们需要对I/O口进行数据写入,才能控制其输出的电平高低,因此我们还需要设置GPIO21_00_DATA寄存器。
因此只需要设置这三个寄存器,就可以完成对这四个LED的完全控制了。

第二步 寻找相似驱动模板修改或者自行编写

在学习阶段,我们当然是自己来写了。下面我们便正式开始编写驱动。
上一章我们了解了驱动程序的一般框架,下面我们套用这个框架来编写这次的LED驱动。

一 设备操作函数编写

让我们想想这个LED设备初始化需要进行哪些操作。
首先从硬件方面考虑,我们需要操作三个寄存器,所以我们需要定义几个变量来控制寄存器:

复制代码
    volatile unsigned long *GPIOMODE;
    volatile unsigned long *GPIO21_00_DIR;
    volatile unsigned long *GPIO21_00_DATA;

因为寄存器是需要经常读写的,为了防止编译器编译的时候把数值优化掉,我们需要使用volatile修饰符。
接下来,考虑可能用到的硬件操作。
第一个肯定是open没错了,而且我们要求设备在open的时候完成初始化:

复制代码
    static int myleds_open( struct inode *inode, struct file *file )
    {
    *GPIOMODE       |=  ( ( 1<<4 )|( 1<<3 )|( 1<<2 ) );
    *GPIO21_00_DIR  |=  ( ( 1<<7 )|( 1<<8 )|( 1<<9 )|(1<<10) );
    return 0;
    }

通过设置GPIOMODE的第2、3、4位为111来使引脚功能变为GPIO。
通过设置GPIO21_00_DIR的7、8、9、10位为1来确定相应引脚为输出功能。

接下来我们写一个函数用于控制I/O口输出高低电平驱动LED。首先为了方便记忆,我们给每个LED控制指令一个宏定义:

复制代码
    #define LED1_ON  0x11
    #define LED1_OFF 0x01
    
    #define LED2_ON  0x22
    #define LED2_OFF 0x02
    
    #define LED3_ON  0x44
    #define LED3_OFF 0x04
    
    #define LED4_ON  0x88
    #define LED4_OFF 0x08

为了实现LED驱动程序的灵活性 ,我们除了需要让用户能够单独控制每个LED之外,还应做到多个LED控制之间不受影响。因此,我们设计了上述控制指令。
指令的低四位标识将要控制的LED对象,高四位标识所要执行的LED的动作:

Bits Name Description
7 LED4ACT LED4动作。0:熄灭;1:点亮
6 LED3ACT LED3动作。0:熄灭;1:点亮
5 LED2ACT LED2动作。0:熄灭;1:点亮
4 LED1ACT LED1动作。0:熄灭;1:点亮
3 LED4SEL 选中LED4
2 LED3SEL 选中LED3
1 LED2SEL 选中LED2
0 LED1SEL 选中LED1

控制函数我们使用的是ioctl函数,而不是write函数:

复制代码
    static long myleds_unlocked_ioctl( struct file *file, unsigned int cmd, unsigned long arg )
    {
    int count,temp;
    temp = cmd;
    for( count=0; count<4; count++ )
    {
        if( temp&0x01 )
        {
            if( temp&0x10 )
            {
                *GPIO21_00_DATA |=  ( 1<<( count+7 ) );  
            }
            else
            {
                *GPIO21_00_DATA &= ~( 1<<( count+7 ) ); 
            }
        }
        temp = ( temp&0xEE )>>1;
    }
    return 0;
    }

二者的区别,《Linux设备驱动程序》给出了如下描述:

除了读取和写入设备之外,大部分驱动程序还需要另外一种能力,即通过设备驱动程序执行各种类型的硬件控制。简单的数据传输之外,大部分设备可以执行其他一些操作,比如控制设备锁门、弹出介质、报告错误信息、改变波特率或者执行自破坏等。这些操作通常通过ioctl方法支持。

而在kernel 2.6.36 中已经完全删除了struct file_operations 中的ioctl 函数指针,取而代之的是unlocked_ioctl。至于其中区别,我们在以后的学习中再做介绍。此处先了解怎么使用。
这两个函数写完,LED所有基础操作就都能通过它们完成了。但是我们还必须让内核知道这个设备的存在,才能通过内核控制它。

二 设备初始化

首先设备号是必须的:

复制代码
    int major;

接下来是关联操作设备的函数的file_operations结构体:

复制代码
    static struct file_operations myleds_fops = {
    .owner          = THIS_MODULE,
    .open           = myleds_open,
    .unlocked_ioctl = myleds_unlocked_ioctl,
    };

open操作使用我们刚才编写的myleds_open函数。
unlocked_ioctl操作使用myleds_unlocked_ioctl函数。
接下来我们就可以注册设备了:

复制代码
    major = register_chrdev( 0, "myleds", &myleds_fops );

register_chrdev函数:

复制代码
    int register_chardev (unsigned int major, const char *name, struct file_operations *fops);

我们上一章介绍过,它的三个参数:

参数 描述
major 主设备号,该值为 0 时,自动运行分配。
name 设备名。
fops file_operations 结构体变量地址(指针)。

函数的返回值:
major 值为 0时:正常注册后返回分配的主设备号。如果分配失败,返回 EBUSY 的负值 ( -EBUSY ) 。major 值若大于255,则返回EINVAL 的负值 (-EINVAL) 。
major值不为0时:正常注册后返回 0 值。若有注册的设备,返回 EBUSY 的负值 (-EBUSY)。
这里我们动态注册主设备号,注册设备名为“myleds”,fops结构体使用刚才写的myleds_fops
在这些做完之后我们就可以来创建设备节点了:
首先定义此设备的设备类:

复制代码
    static struct class *myleds_class;

并且为结构体初始化:

复制代码
    myleds_class = class_create( THIS_MODULE, "myleds" );

最后创建设备节点:

复制代码
    device_create( myleds_class, NULL, MKDEV( major, 0 ), NULL, "myleds" );

函数中用到的三个参数一个是刚才创建的class指针,一个是dev_t类型的设备编号,最后一个是设备名。
但是刚才我们定义的设备号是int型的,所以我们需要使用MKDEV函数来转换。此函数第一个参数是主设备号,第二个参数是次设备号。成功的返回值是dev_t类型的设备编号。
写到这里设备就已经在内核中注册完毕了,设备节点也已经创建,可以从/dev下查看了。
但是到这里还没有完,我们需要将IO地址空间映射到内核的虚拟地址空间上去,以便应用程序控制:

复制代码
    GPIOMODE=(volatile unsigned long *)ioremap(0x10000060, 4);
    GPIO21_00_DIR=(volatile unsigned long *)ioremap(0x10000624,4);
    GPIO21_00_DATA=(volatile unsigned long *)ioremap(0x10000620,4);

但是别忘了我们还需要一个退出函数来处理退出设备的操作:

复制代码
    static void __exit myleds_exit( void )
    {
    unregister_chrdev( major, "myleds" );
    device_destroy( myleds_class, MKDEV( major, 0 ) );
    class_destroy( myleds_class );
    
    iounmap( GPIOMODE );
    iounmap( GPIO21_00_DIR );
    iounmap( GPIO21_00_DATA );
    }
完整代码:
复制代码
    #include <linux/mm.h>
    #include <linux/miscdevice.h>
    #include <linux/slab.h>
    #include <linux/vmalloc.h>
    #include <linux/mman.h>
    #include <linux/random.h>
    #include <linux/init.h>
    #include <linux/raw.h>
    #include <linux/tty.h>
    #include <linux/capability.h>
    #include <linux/ptrace.h>
    #include <linux/device.h>
    #include <linux/highmem.h>
    #include <linux/crash_dump.h>
    #include <linux/backing-dev.h>
    #include <linux/bootmem.h>
    #include <linux/splice.h>
    #include <linux/pfn.h>
    #include <linux/export.h>
    #include <linux/io.h>
    #include <linux/aio.h>
    #include <linux/kernel.h>
    #include <linux/module.h>
    #include <asm/uaccess.h>
    
    //==================== 宏定义LED操作 ===================//
    #define LED1_ON  0x11
    #define LED1_OFF 0x01
    
    #define LED2_ON  0x22
    #define LED2_OFF 0x02
    
    #define LED3_ON  0x44
    #define LED3_OFF 0x04
    
    #define LED4_ON  0x88
    #define LED4_OFF 0x08
    //======================================================//
    
    
    //====================== 定义寄存器 =====================//
    volatile unsigned long *GPIOMODE;
    volatile unsigned long *GPIO21_00_DIR;
    volatile unsigned long *GPIO21_00_DATA;
    //======================================================//
    
    
    //====================== 定义设备类 =====================//
    static struct class *myleds_class;
    //======================================================//
    
    
    //====================== 设备open ======================//
    static int myleds_open( struct inode *inode, struct file *file )
    {
    *GPIOMODE       |=  ( ( 1<<4 )|( 1<<3 )|( 1<<2 ) );
    *GPIO21_00_DIR  |=  ( ( 1<<7 )|( 1<<8 )|( 1<<9 )|(1<<10) );
    return 0;
    }
    //======================================================//
    
    
    //====================== 设备操作 =======================//
    static long myleds_unlocked_ioctl( struct file *file, unsigned int cmd, unsigned long arg )
    {
    int count,temp;
    temp = cmd;
    for( count=0; count<4; count++ )
    {
        //判断LED对象
        if( temp&0x01 )
        {
            if( temp&0x10 )
            {
                *GPIO21_00_DATA |=  ( 1<<( count+7 ) );  //点亮相应led
            }
            else
            {
                *GPIO21_00_DATA &= ~( 1<<( count+7 ) ); //熄灭相应led
            }
        }
        temp = ( temp&0xEE )>>1;
    }
    return 0;
    }
    //======================================================//
    
    
    //================== fops结构初始化 ====================//
    static struct file_operations myleds_fops = {
    .owner          = THIS_MODULE,
    .open           = myleds_open,
    .unlocked_ioctl = myleds_unlocked_ioctl,
    };
    //======================================================//
    
    
    //===================== 主设备号 =======================//
    int major;
    //======================================================//
    
    
    //===================== 设备初始化 =====================//
    static int __init myleds_init( void )
    {
    major = register_chrdev( 0, "myleds", &myleds_fops );
    myleds_class = class_create( THIS_MODULE, "myleds" );
    device_create( myleds_class, NULL, MKDEV( major, 0 ), NULL, "myleds" );
    
    GPIOMODE       = ( volatile unsigned long * )ioremap( 0x10000060, 4 );
    GPIO21_00_DIR  = ( volatile unsigned long * )ioremap( 0x10000624, 4 );
    GPIO21_00_DATA = ( volatile unsigned long * )ioremap( 0x10000620, 4 );     
    
    return 0; 
    }
    //======================================================//
    
    
    //===================== 设备退出 =======================//
    static void __exit myleds_exit( void )
    {
    unregister_chrdev( major, "myleds" );
    device_destroy( myleds_class, MKDEV( major, 0 ) );
    class_destroy( myleds_class );
    
    iounmap( GPIOMODE );
    iounmap( GPIO21_00_DIR );
    iounmap( GPIO21_00_DATA );
    } 
    //======================================================//
    
    
    module_init( myleds_init );
    module_exit( myleds_exit );
    
    MODULE_LICENSE("GPL");
Makefile

./myleds/src目录下Makefile只有一句话:

复制代码
    obj-m += myleds.o

./myleds目录下Makefile:

复制代码
    include $(TOPDIR)/rules.mk
    include $(INCLUDE_DIR)/kernel.mk
    
    PKG_NAME:=myleds
    PKG_RELEASE:=1
    
    include $(INCLUDE_DIR)/package.mk
    
    define KernelPackage/myleds
    SUBMENU:=Other modules
    #DEPENDS:=@!LINUX_3_3
    TITLE:=My led driver
    FILES:=$(PKG_BUILD_DIR)/myleds.ko
    #AUTOLOAD:=$(call AutoLoad, 30, myleds, 1)
    endef
    
    define KernelPackage/myleds/description
    This is a leds control driver
    endef
    
    MAKE_OPTS:= \
    ARCH="$(LINUX_KARCH)"  \
    CROSS_COMPILE="$(TARGET_CROSS)" \
    SUBDIRS="$(PKG_BUILD_DIR)"
    
    define Build/Prepare
    mkdir -p $(PKG_BUILD_DIR)
    $(CP) ./src/* $(PKG_BUILD_DIR)/
    endef
    
    define Build/Compile
    $(MAKE) -C "$(LINUX_DIR)"\
    $(MAKE_OPTS)             \
    modules
    endef
    
    $(eval $(call KernelPackage,myleds))

第三步 编写测试程序测试驱动

测试程序

测试程序就不多讲了,代码如下:
./myleds/src目录下新建app_led.c:

复制代码
    #include <stdio.h>
    #include <curses.h>
    #include <sys/types.h>
    #include <sys/stat.h>
    #include <fcntl.h>
    #include <unistd.h>
    #include <sys/ioctl.h>
    #include <string.h>
    
    #define LED1_ON  0x11
    #define LED1_OFF 0x01
    
    #define LED2_ON  0x22
    #define LED2_OFF 0x02
    
    #define LED3_ON  0x44
    #define LED3_OFF 0x04
    
    #define LED4_ON  0x88
    #define LED4_OFF 0x08
    
    int main(int argc, char **argv)
    {
    int fd;
        int count;
    if (argc != 3)
    {
        printf("示例:<dev> ledn <on|off>\n");
        return 0;
    }
    
    /* 1.打开设备节点 */
    fd = open("/dev/myleds", O_RDWR | O_NONBLOCK);
    if (fd < 0)
    {
        printf("can't open!\n");
        return -1;
    }
    
    /* 2.根据参数不同,控制LEDs */
    if(!strcmp("led1", argv[1]))
    {
        if (!strcmp("on", argv[2]))
        {
            // 亮灯
            ioctl(fd, LED1_ON);
        }
        else if (!strcmp("off", argv[2]))
        {
            // 灭灯
            ioctl(fd, LED1_OFF);
        }
        else
        {
            printf("示例:<dev> ledn <on|off>\n");
            return 0;
        }
    }
    else if(!strcmp("led2", argv[1]))
    {
        if (!strcmp("on", argv[2]))
        {
            // 亮灯
            ioctl(fd, LED2_ON);
        }
        else if (!strcmp("off", argv[2]))
        {
            // 灭灯
            ioctl(fd, LED2_OFF);
        }
        else
        {
            printf("示例:<dev> ledn <on|off>\n");
            return 0;
        }
    }
    else if(!strcmp("led3", argv[1]))
    {
        if (!strcmp("on", argv[2]))
        {
            // 亮灯
            ioctl(fd, LED3_ON);
        }
        else if (!strcmp("off", argv[2]))
        {
            // 灭灯
            ioctl(fd, LED3_OFF);
        }
        else
        {
            printf("示例:<dev> ledn <on|off>\n");
            return 0;
        }
    }
    else if(!strcmp("led4", argv[1]))
    {
        if (!strcmp("on", argv[2]))
        {
            // 亮灯
            ioctl(fd, LED4_ON);
        }
        else if (!strcmp("off", argv[2]))
        {
            // 灭灯
            ioctl(fd, LED4_OFF);
        }
        else
        {
            printf("示例:<dev> ledn <on|off>\n");
            return 0;
        }
    }
    
    else if(!strcmp("led", argv[1]))
    {
        if (!strcmp("flash", argv[2]))
        {
            for(count=0; count<10; count++)
                        { 
                            // 亮灯
                ioctl(fd, LED1_ON|LED2_ON|LED3_ON|LED4_ON);
                usleep(500000);
                            ioctl(fd, LED1_OFF|LED2_OFF|LED3_OFF|LED4_OFF);
                            usleep(500000);
                        }
        }
        else
        {
            printf("示例:<dev> ledn <on|off>\n");
            return 0;
        }
    }
    
    else
    {
        printf("示例:<dev> ledn <on|off>\n");
        return 0;
    }
    
    return 0;
    }

测试函数包含单独的LED控制和LED混合控制。

Makefile

同目录下Makefile如下:

复制代码
    CC = gcc
    CFLAGS = -Wall
    OBJS = app_led.o
    
    all: app_led
    
    %.o: %.c
    $(CC) $(CFLAGS) -c -o $@ $< $(LDFLAGS)
    
    fbtest: $(OBJS)
    $(CC) -o $@ $(OBJS) $(LDFLAGS)
    
    clean:
    rm -f rbcfg *.o

./myleds目录下另建Makefile:

复制代码
    #
    # Copyright (C) 2012 OpenWrt.org
    #
    # This is free software, licensed under the GNU General Public License v2.
    # See /LICENSE for more information.
    #
    
    include $(TOPDIR)/rules.mk
    
    PKG_NAME:=app_led
    PKG_RELEASE:=1
    
    PKG_BUILD_DIR := $(BUILD_DIR)/$(PKG_NAME)
    
    include $(INCLUDE_DIR)/package.mk
    
    define Package/app_led
    SECTION:=utils
    CATEGORY:=Utilities
    TITLE:=Frame buffer device testing tool
    DEPENDS:=+libncurses
    endef
    
    define Build/Prepare
    mkdir -p $(PKG_BUILD_DIR)
    $(CP) ./src/* $(PKG_BUILD_DIR)/
    endef
    
    define Build/Configure
    endef
    
    TARGET_LDFLAGS :=
    
    define Build/Compile
    $(MAKE) -C $(PKG_BUILD_DIR) \
        CC="$(TARGET_CC)" \
        CFLAGS="$(TARGET_CFLAGS) -Wall" \
        LDFLAGS="$(TARGET_LDFLAGS)"
    endef
    
    define Package/app_led/install
    $(INSTALL_DIR) $(1)/usr/sbin
    $(INSTALL_BIN) $(PKG_BUILD_DIR)/app_led $(1)/usr/sbin/
    endef
    
    $(eval $(call BuildPackage,app_led))

小结

通过这几天的学习,初步掌握了驱动程序开发的步骤。但这些只是粗浅的知识,就和编程语言的HelloWorld一样,只是开始。
接下来将进入实战阶段,目标是开发一款一体机控制器。平台使用RT5350,使用I2C芯片PCF8574T进行I/O扩展,使用WM8978进行音频解码和混音,这也将使用到I2S接口。同时因为UART接口的紧张,还需要动态切换控制台和普通串口。所以接下来任务很重。希望能进展顺利吧。

全部评论 (0)

还没有任何评论哟~