zoukankan      html  css  js  c++  java
  • 字符设备驱动程序----led驱动

    以led驱动为例,讲解如何写一个字符设备驱动程序。

    1)看原理图

    a.确定引脚

    b.看芯片手册,确定如何操作引脚

    2)写驱动程序

    3)写测试程序

    最简单的字符设备驱动程序的框架:

    App:     open           read          write            ioctl

    ---------------------------------------------------------------

    驱动:  drv_open   drv_read    drv_write      drv_ioctl

    ----------------------------------------------------------------

    硬件:

     

    以面向对象的思想构造一个结构体,在这个结构体中填充drv_open、drv_write等这些函数。

    a  分配一个file_operations结构体体;

    b  设置它:

      .open = led_open

      .write  = led_write

    c  注册:告诉内核,就是把file_operations这个结构体放入某个数组里面。可以简单的这样认为,内核中有一个数组,这个数组存放一个又一个的驱动程序,file_operations这个结构体想放在数组哪一项中,就是通过主设备号指定的。这个主设备号可以写0,如果写0的话,会从头开始遍历这个数组,找到一项空的,就把file_operations这个结构体放入。

    d  入口函数:   register_chrdev

    e  出口函数:   unregister_chrdev

    我们提供的这些函数要实现什么样的功能?比如说led_open、led_write等函数实现的功能是什么?

    led_open: 把led引脚配置为输出引脚

    led_write: 根据app传入的值,设置引脚状态

    问:在驱动中如何指定引脚,看原理图可以确定引脚是哪一个,但我如何告诉驱动程序呢?有三种方法可供选择。

    1)传统写法。

    在驱动代码中写死

    2)总线设备驱动模型

    把驱动一分为二。led_drv.c    led_dev.c

    3)使用设备树指明引脚。

    此时驱动程序也是分为两部分,一部分是led_drv.c   另一部分是设备树,比如说jz2440.dts。

    注意:无论是传统方法、总线设备驱动模型法、设备树法驱动的核心都是一样的,都是分配、设置、注册一个file_operations结构体,唯一的差别就是在于如何指定这些引脚。

    驱动写法核心不变,差别在于如何指定硬件资源。

    以上三种驱动写法有什么优缺点呢?

    假设有这样的一个场景:

    公司里使用同一款芯片做了两个产品,一个是tv,一个是camera。

    如何写驱动程序?

    1)传统写法:

    led_drv.c

    分配一个file_opreation结构体

    设置

      .open = led_open------配置pin1为输出引脚

      .write =led_write------根据app传入的值,设置pin1状态

    注册

    入口

    出口

    现在在第二块板子上写驱动程序,可以将led_drv.c的驱动程序复制过来,然后在此基础上修改。把原来的pin1改成pin2.

    优点:简单

    缺点:不易扩展,大量重复的代码

    对于这种现象有没有办法改进呢?答案是肯定的。在我们这个场景里面,这两个设备使用同一个主芯片,引脚的操作是类似的。可以将这些引脚抽出来,通过其他方法来指定。由此引入总线设备驱动模型

    2)总线设备驱动模型

    将驱动分为两部分。一部分是led_dev.c, 一部分是led_drv.c

    一部分是led_dev.c(指定资源)-------------------------对应tv

      ----分配、设置、注册一个platform_device。里面会有各种资源指定引脚,比如说:

        .resource 指明引脚  pin1

                   .name 

    一部分是led_dev.c(指定资源)-------------------------对应camera

      ----分配、设置、注册一个platform_device。里面会有各种资源指定引脚,比如说:

        .resource 指明引脚  pin2

        .name

    tv的led_dev.c和camera的led_dev.c对应同一个led_drv.c

     一部分是led_drv.c(分配、设置、注册一个platform_drv结构体,这个结构体中有probe、.driver{  .name }) 。

    当内核发现有一个平台device和平台driver的名字相同时,probe函数就会被调用。probe函数里面会做什么事情呢?

    分配一个file_opreation结构体

    设置

      .open = led_open------在这个地方不会将引脚写死了,配置平台设备中指定的引脚

      .write =led_write------根据app传入的值,设置平台设备指定的引脚状态

    注册

    入口

    出口

     

    对于tv和camera,他们的led_drv.c保持不变,唯一需要修改的是led_dev.c中的资源。在TV中资源指定是pin1,在camera中资源指定是pin2。这样的驱动程序很容易扩展,只需要修改资源即可。那么缺点是什么呢?现在有两个版本的硬件,就需要有两个platform_device,如果继续增加的话,就需要更多的platform_device,里面就会存在很多冗余的代码,并且这些代码是以.c文件存在的,充斥着大量的污染,linux创始人说这样的代码存在就是垃圾。继续改进,从而设备树就登场了。

    3)设备树

    总线设备驱动模型的缺点在于通过这些.c文件来指定这些资源,使用的话需要重新编译。

    使用设备树写程序的时候,仍然分为两部分:

    led_drv.c

     分配、设置、注册一个platform_driver结构体(同总线设备驱动模型的platform_driver)

    通过dts文件来指定资源,内核会根据这个dts文件构造platform_device, 当需要更改单板时,只需重新定义dts文件就可以了。这个文件最终会被编译成一个dtb文件。启动单板时,既要启动内核,也要传入dtb文件。

    复杂、无冗余代码

        

  • 相关阅读:
    jquery each() 方法跳出循环
    webpack 启动 vue
    HTML DOM insertBefore() 方法 问题
    浮动元素的高度怎么撑起
    jquery toggle 方法被废除的替代方法
    函数声明和函数表达式
    href与src 区别
    ThreadLocal用法
    《计算机网络》ISO/OSI参考模型分层协议
    解决Idea创建的maven Web项目无法连接mysql数据库
  • 原文地址:https://www.cnblogs.com/-glb/p/11191720.html
Copyright © 2011-2022 走看看