我是靠谱客的博主 生动鸵鸟,最近开发中收集的这篇文章主要介绍linux驱动开发7驱动框架之gpiolib,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

1.gpiolib学习重点

1)gpiolib的建立过程

2)gpiolib的使用方法:申请、使用、释放

3)gpiolib的架构:涉及哪些目录的哪些文件

2.gpiolib的学习方法

1)以一条主线进去,坚持主线

2)中途遇到杂碎知识,彻底搞定之,然后继续主线

3)随时做笔记以加深理解和记忆

4)学习途中注意架构思想,提升自己大脑的空间复杂度

3.主线1:gpiolib的建立

1)找到目标函数

smdkc110_map_io

         s5pv210_gpiolib_init               这个函数就是我们gpiolib初始化的函数

3.struct s3c_gpio_chip

struct s3c_gpio_chip {

    struct gpio_chip    chip;

    struct s3c_gpio_cfg *config;

    struct s3c_gpio_pm  *pm;

    void __iomem        *base; //IO端口操作寄存器的基地址的虚拟地址

    int         eint_offset;

    spinlock_t       lock;

#ifdef CONFIG_PM

    u32         pm_save[7];

#endif

};

1)这个结构体是一个GPIO端口的抽象,这个结构体的一个变量就可以完全的描述一个IO端口。

2)端口和IO口是两个概念。S5PV210有很多个IO口(160个左右),这些IO口首先被分成N个端口(port group),然后每个端口中又包含了M个IO口。譬如GPA0是一个端口,里面包含了8个IO口,我们一般记作:GPA0_0(或GPA0.0)、GPA0_1、

3)内核中为每个GPIO分配了一个编号,编号是一个数字(譬如一共有160个IO时编号就可以从1到160连续分布),编号可以让程序很方便的去识别每一个GPIO。

4.s5pv210_gpio_4bit

这个东西是一个结构体数组,数组中包含了很多个struct s3c_gpio_chip类型的变量。

static struct s3c_gpio_chip s5pv210_gpio_4bit[] = {

    {

        .chip   = {

            .base   = S5PV210_GPA0(0),//编号的基址

            .ngpio  = S5PV210_GPIO_A0_NR, //当前端口的IO数量

            .label  = "GPA0",   //当前端口的名字

            .to_irq = s5p_gpiolib_gpioint_to_irq,//当前端口换算成中断号

        },

    }, {

        .chip   = {

            .base   = S5PV210_GPA1(0),

            .ngpio  = S5PV210_GPIO_A1_NR,

            .label  = "GPA1",

            .to_irq = s5p_gpiolib_gpioint_to_irq,

        },

。。。。。。。。。。。。。。。。

5.S5PV210_GPA0

1)这个宏的返回值就是GPA0端口的某一个IO口的编号值,传参就是我们这个IO口在GPA0端口中的局部编号。

__init int s5pv210_gpiolib_init(void)

{

    struct s3c_gpio_chip *chip = s5pv210_gpio_4bit;

    int nr_chips = ARRAY_SIZE(s5pv210_gpio_4bit);

    int i = 0;

 

    for (i = 0; i < nr_chips; i++, chip++) {

        if (chip->config == NULL)

            chip->config = &gpio_cfg;

        if (chip->base == NULL)

            chip->base = S5PV210_BANK_BASE(i);

    }

 

    samsung_gpiolib_add_4bit_chips(s5pv210_gpio_4bit, nr_chips);

 

    return 0;

}

 

2)samsung_gpiolib_add_4bit_chips这个函数才是具体进行gpiolib的注册的。这个函数接收的参数是我们当前文件中定义好的结构体数组s5pv210_gpio_4bit(其实2个参数分别是数组名和数组元素个数),这个数组中其实就包含了当前系统中所有的IO端口的信息(这些信息包含:端口的名字、端口中所有GPIO的编号、端口操作寄存器组的虚拟地址基地址、端口中IO口的数量、端口上下拉等模式的配置函数、端口中的IO口换算其对应的中断号的函数)。

6.几个问题

1)哪个目录的哪个文件gpiolib.c

2)函数名中为什么有个4bit:三星的CPU中2440的CON寄存器是2bit对应一个IO口,而6410和210以及之后的系列中CON寄存器是4bit对应1个IO口。所以gpiolib在操作2440和210的CON寄存器时是不同的。

7.函数调用关系

void __init samsung_gpiolib_add_4bit_chips(struct s3c_gpio_chip *chip,

                       int nr_chips)

{

    for (; nr_chips > 0; nr_chips--, chip++) {

        samsung_gpiolib_add_4bit(chip);

        s3c_gpiolib_add(chip);

    }

}

samsung_gpiolib_add_4bit_chips

         samsung_gpiolib_add_4bit

void __init samsung_gpiolib_add_4bit(struct s3c_gpio_chip *chip)

{

    chip->chip.direction_input = samsung_gpiolib_4bit_input;

    chip->chip.direction_output = samsung_gpiolib_4bit_output;

    chip->pm = __gpio_pm(&s3c_gpio_pm_4bit);

}

         s3c_gpiolib_add

__init void s3c_gpiolib_add(struct s3c_gpio_chip *chip)

{

    struct gpio_chip *gc = &chip->chip;

    int ret;

 

    BUG_ON(!chip->base);

    BUG_ON(!gc->label);

    BUG_ON(!gc->ngpio);

 

    spin_lock_init(&chip->lock);

 

    if (!gc->direction_input)

        gc->direction_input = s3c_gpiolib_input;

    if (!gc->direction_output)

        gc->direction_output = s3c_gpiolib_output;

    if (!gc->set)

        gc->set = s3c_gpiolib_set;

    if (!gc->get)

        gc->get = s3c_gpiolib_get;

 

#ifdef CONFIG_PM

    if (chip->pm != NULL) {

        if (!chip->pm->save || !chip->pm->resume)

            printk(KERN_ERR "gpio: %s has missing PM functionsn",

                   gc->label);

    } else

        printk(KERN_ERR "gpio: %s has no PM functionn", gc->label);

#endif

 

    /* gpiochip_add() prints own failure message on error. */

    ret = gpiochip_add(gc);

    if (ret >= 0)

        s3c_gpiolib_track(chip);

}

 

经过分析,发现samsung_gpiolib_add_4bit内部其实并没有做gpiolib的注册工作,而是还在做填充,填充的是每一个GPIO被设置成输入模式/输出模式的操作方法。

8.s3c_gpiolib_add

1)首先检测并完善chip的direction_input/direction_ouput/set/get这4个方法

2)然后调用gpiochip_add方法进行真正的注册操作。其实这个注册就是将我们的封装了一个GPIO端口的所目录和文件结构:

mach-s5pv210/gpiolib.c               s5pv210_gpiolib_init

mach-s5pv210/include/mach/gpio.h                   #define S5PV210_GPA0(_nr)      (S5PV210_GPIO_A0_START + (_nr))

arch/arm/plat-samsung/gpiolib.c                里面是210/6410这种4bit CON寄存器类型的操作方法

arch/arm/plat-samsung/gpio.c            里面是24XX这种2bit CON寄存器类型的操作方法

drivers/gpio/gpiolib.c                              里面是内核开发者提供的gpiolib的驱动框架部分

有信息的chip结构体变量挂接到内核gpiolib模块定义的一个gpio_desc数组中的某一个格子中。

9.从驱动框架角度再来分析一下gpiolib

1)之前的分析已经告一段落,截至目前我们已经搞清楚了gpiolib的建立工程。但是这只是整个gpiolib建立的一部分,是厂商驱动工程师负责的那一部分;还有另一部分是内核开发者提供的驱动框架的那一部分,就是我们后面要去分析的第2条主线。

2)drivers/gpio/gpiolib.c这个文件中所有的函数构成了我们第2部分,也就是内核开发者写的gpiolib框架部分。这个文件中提供的函数主要有以下部分:

gpiochip_add:         是框架开出来的接口,给厂商驱动工程师用,用于向内核注册我们的gpiolib

gpio_request:       是框架开出来的接口,给使用gpiolib来编写自己的驱动的驱动工程师用的,

                                     驱动中要想使用某一个gpio,就必须先调用gpio_request接口来向内核的

                                     gpiolib部分申请,得到允许后才可以去使用这个gpio。

gpio_free:                对应gpio_request,用来释放申请后用完了的gpio

gpio_request_one/gpio_request_array: 这两个是gpio_request的变种

gpiochip_is_requested:        接口用来判断某一个gpio是否已经被申请了

gpio_direction_input/gpio_direction_output: 接口用来设置GPIO为输入/输出模式,

                                     注意该函数内部实际并没有对硬件进行操作,只是通过chip结构体变量的

                                     函数指针调用了将来SoC厂商的驱动工程师写的真正的操作硬件实现gpio

                                     设置成输出模式的那个函数。

                                    

以上的接口属于一类,这些都是给写其他驱动并且用到了gpiolib的人使用的

剩下的还有另外一类函数,这类函数是gpiolib内部自己的一些功能实现的代码

9.gpiolibattribute部分

1)CONFIG_GPIO_SYSFS

2)GPIO的attribute演示

10.相关代码分析

gpiolib_sysfs_init

         gpiochip_export

                   sysfs_create_group

echo 23 > export--------------------获取GPIO23的属性

11.使用gpiolib完成led驱动

11.1.流程分析

1)第1步:使用gpio_request申请要使用的一个GPIO

2)第2步:gpio_direction_input/gpio_direction_output 设置输入/输出模式

3)第3步:设置输出值gpio_set_value  获取IO口值gpio_get_value

11.2.代码实践

1)在led1上编写代码测试通过

2)扩展支持led2和led3、led4.可以分开注册也可以使用gpio_request_array去一次注册

3)学习linux中查看gpio使用情况的方法

内核中提供了虚拟文件系统debugfs,里面有一个gpio文件,提供了gpio的使用信息。

使用方法:mount -t debugfs debugfs /tmp,然后cat /tmp/gpio即可得到gpio的所有信息,使用完后umount /tmp卸载掉debugfs

12.将驱动添加到内核中

12.1驱动的存在形式

1)野生,优势是方便调试开发,所以在开发阶段都是这种

2)家养,优势可以在内核配置时make menuconfig决定内核怎么编译,方便集成

12.2驱动开发的一般步骤

1)以模块的形式在外部编写、调试

2)将调试好的驱动代码集成到kernel中

12.3实践

1)关键点:Kconfig、Makefile、make menuconfig

2)操作步骤:

第1步:将写好的驱动源文件放入内核源码中正确的目录下

第2步:在Makefile中添加相应的依赖

第3步:在Kconfig中添加相应的配置项

第4步:make menuconfig

#include <linux/module.h>       // module_init  module_exit

#include <linux/init.h>         // __init   __exit

#include <linux/fs.h>

#include <linux/leds.h>

#include <mach/regs-gpio.h>

#include <mach/gpio-bank.h>

#include <linux/io.h>

#include <linux/ioport.h>

#include <mach/gpio.h>

 

 

#define GPJ0CON     S5PV210_GPJ0CON

#define GPJ0DAT     S5PV210_GPJ0DAT

 

#define GPI0_LED1   S5PV210_GPJ0(3)

#define GPI0_LED2   S5PV210_GPJ0(4)

#define GPI0_LED3   S5PV210_GPJ0(5)

 

#define X210_LED_OFF    1

#define X210_LED_ON     0

 

static struct led_classdev mydev1;          // 定义结构体变量

static struct led_classdev mydev2;          // 定义结构体变量

static struct led_classdev mydev3;          // 定义结构体变量

 

// 这个函数就是要去完成具体的硬件读写任务的

static void s5pv210_led1_set(struct led_classdev *led_cdev,

                enum led_brightness value)

{

    printk(KERN_INFO "s5pv210_led1_setn");

 

    if (value == LED_OFF)

    {

        gpio_set_value(GPI0_LED1, X210_LED_OFF);

    }

    else

    {

        gpio_set_value(GPI0_LED1, X210_LED_ON);

    }

}

 

static void s5pv210_led2_set(struct led_classdev *led_cdev,

                enum led_brightness value)

{

    printk(KERN_INFO "s5pv210_led2_setn");

 

    if (value == LED_OFF)

    {

        gpio_set_value(GPI0_LED2, X210_LED_OFF);

    }

    else

    {

        gpio_set_value(GPI0_LED2, X210_LED_ON);

    }

}

 

static void s5pv210_led3_set(struct led_classdev *led_cdev,

                enum led_brightness value)

{

    printk(KERN_INFO "s5pv210_led3_setn");

 

    if (value == LED_OFF)

    {

        gpio_set_value(GPI0_LED3, X210_LED_OFF);

    }

    else

    {

        gpio_set_value(GPI0_LED3, X210_LED_ON);

    }

}

 

 

static int __init s5pv210_led_init(void)

{

    // 用户insmod安装驱动模块时会调用该函数

    // 该函数的主要任务就是去使用led驱动框架提供的设备注册函数来注册一个设备

    int ret = -1;

    //这里申请驱动所用到的各种资源

    if(gpio_request(GPI0_LED1, "led1_gpj0_3"))

    {

        printk(KERN_ERR "led1_gpj0_3 gpio_request failedn");

    }

    else

    {

        //设置为输出模式并且默认输出1,LED等灭

        gpio_direction_output(GPI0_LED2, 1);

    }  

   

    if(gpio_request(GPI0_LED2, "led1_gpj0_4"))

    {

        printk(KERN_ERR "led1_gpj0_4 gpio_request failedn");

    }

    else

    {

        //设置为输出模式并且默认输出1,LED等灭

        gpio_direction_output(GPI0_LED2, 1);

    }

    if(gpio_request(GPI0_LED3, "led1_gpj0_5"))

    {

        printk(KERN_ERR "led1_gpj0_5 gpio_request failedn");

    }

    else

    {

        //设置为输出模式并且默认输出1,LED等灭

        gpio_direction_output(GPI0_LED3, 1);

    }

   

    // led1

    mydev1.name = "led1";

    mydev1.brightness = 0; 

    mydev1.brightness_set = s5pv210_led1_set;

   

    ret = led_classdev_register(NULL, &mydev1);

    if (ret < 0) {

        printk(KERN_ERR "led_classdev_register failedn");

        return ret;

    }

   

    // led2

    mydev2.name = "led2";

    mydev2.brightness = 0; 

    mydev2.brightness_set = s5pv210_led2_set;

   

    ret = led_classdev_register(NULL, &mydev2);

    if (ret < 0) {

        printk(KERN_ERR "led_classdev_register failedn");

        return ret;

    }

   

    // led3

    mydev3.name = "led3";

    mydev3.brightness = 0; 

    mydev3.brightness_set = s5pv210_led3_set;

   

    ret = led_classdev_register(NULL, &mydev3);

    if (ret < 0) {

        printk(KERN_ERR "led_classdev_register failedn");

        return ret;

    }

   

    return 0;

}

 

static void __exit s5pv210_led_exit(void)

{

    led_classdev_unregister(&mydev1);

    led_classdev_unregister(&mydev2);

    led_classdev_unregister(&mydev3);

   

    //释放GPIO

    gpio_free(GPI0_LED1);

}

 

 

module_init(s5pv210_led_init);

module_exit(s5pv210_led_exit);

 

// MODULE_xxx这种宏作用是用来添加模块描述信息

MODULE_LICENSE("GPL");                          // 描述模块的许可证

MODULE_AUTHOR("aston <1264671872@qq.com>");     // 描述模块的作者

MODULE_DESCRIPTION("s5pv210 led driver");       // 描述模块的介绍信息

MODULE_ALIAS("s5pv210_led");                    // 描述模块的别名信息

最后

以上就是生动鸵鸟为你收集整理的linux驱动开发7驱动框架之gpiolib的全部内容,希望文章能够帮你解决linux驱动开发7驱动框架之gpiolib所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(61)

评论列表共有 0 条评论

立即
投稿
返回
顶部