原创 S3C2410通用IO口按键驱动分析

2008-7-10 18:11 4562 5 5 分类: MCU/ 嵌入式
xiao.gifda.gif

arm开发板买了大半年了一直都没怎么用,趁着这些天回家不能上网,每天研究一点点,也算小有所得,便就记录下来,以备后忘.(不能上网虽然有很多不便,但也确实更能合理分配业余时间).

S3C2410_gpio_button.c
    模块初始化module_init(init_gpio_button)
    init_gpio_button
        循环处理gpio_buttons数组, 这个数组是在gpio_button_smdk2410_meri.c或者gpio_button_smdk2410_aiji.c中定义的,meri与 aiji是两个不同的开发板开发公司,所以他们的按键定义会有所不同,需要根据自己所采用的板子来配置(在内核编译里配置), 我们这里分析meri公司的.
        如果当前按键有对应的irq,
            设置此按键对应的irq的触发方式(set_external_irq). 包括上升沿/下降沿触发, 是否允许内部上拉电阻. 这两个参数是在gpio_buttons数组里已经作为常量定义好了的,这里只要读出来就行了
            申请中断(request_irq). 注册中断处理函数gpio_button_interrupt, 设置中断属性为FIQ(SA_INTERRUPT), 注册按键名, 设置gpio_buttons数组的地址作为dev_id(其实是gpio_button_interrupt的第二个参数)
        如果当前按键没有对应的irq
            如果当前按键有对应的pm_callback(就是按键对应的电源管理的回调函数), 就初始化请求(PM_MZ_INIT)作为参数调用这个函数
        打印一些信息
        重复循环过程
        如果定义了了CONFIG_PM, 即内核有电源管理模块
             初始化一个定时器双向链表的结构体,这里结构提里主要是设置了回调函数pwButton_timer_callback
             向电源管理模块注册按键的回调函数(pm_register). 注册的是设备类型(PM_USER_DEV, 在这个enum的定义后面有这么一句话: when wakeup, user must be handle this), 设备ID(PM_USER_INPUT), 以及回调函数(gpio_button_pm_callback)
             gpio_button_pm_callback
                如果当前新状态是休眠
                   如果此按键有对应的pm_callback, 则以睡眠状态为参数运行此函数
                   若此按键无对应的pm_callback
                      若此按键使用irq, 则禁止之
                如果当前新状态是唤醒
                    如果此按键有对应的pm_callback, 则以唤醒状态做参数运行此函数
                   若此按键无对应的pm_callback
                      若此按键使用irq, 则开放之                
        如果定义了CONFIG_S3C2410_SMDK, 也就是基于SMDK2410构架的实现
             初始化2410按键(s3c2410tk_button_init). 这里的按键指非特殊按键(不是Ctrl, Alt, shift等)
             s3c2410tk_button_init
                设置GPIO的B5为输入模式并禁止上拉(set_gpio_ctrl)
                设置GPIO的B6为输入模式并禁止上拉(set_gpio_ctrl)
                设置GPIO的E11为输入模式并禁止上拉(set_gpio_ctrl)
                设置GPIO的E12为输入模式并禁止上拉(set_gpio_ctrl) 
                初始化一个定时器(init_timer(&button_check_timer))并设置回调函数为tk_button_check_timer_handler, 超时时间为BUTTON_DELAY
                将定时器加入定时器链表(即现在开始这个定时器开始工作, 并在BUTTON_DELAY毫秒后第一次调用函数tk_button_check_timer_handler)
                    tk_button_check_timer_handler:这个函数每隔BUTTON_DELAY毫秒运行一次,并扫描gpio_buttons_tk数组中的按键
                    禁止中断(cli). 为了防止在扫描期间有按键中断进来,因为handle_scancode不是可重入函数
                    开始循环检测gpio_buttons_tk数组中每个按键是否有改变
                       暂存按键的原状态(按下还是未按)
                       读取按键的现状态(read_gpio_bit), 也就是读此按键对应的IO口电平
                       若现状态与原状态同,则表示此按键未改变,继续下一按键的检测
                       若状态有改变,则将按键的扫描值与按键状态发送给系统处理(handle_scancode). 在handle_scancode内主要是做长按的autorepeat处理, 将scancode转换成keycode, 唤醒等待进程, 将键加入队列等. 在这里有个函数比较有意思: handle_scancode->add_keyboard_randomness->add_timer_randomness->int_ln_12bits. 这个函数分析看附录.
                       重复循环
                    循环结束
                    开放中断
                    修改定时器的下次触发时间
    模块初始化结束

我们再来看按键的中断响应部分gpio_button_interrupt
    循环寻找当前irq对应的按键结构
    读取此按键状态(按下还是松开)
    判断此按键是否有对应的按键处理函数, 如果有,则以按键状态和一个私有成员(BUTTON_TYPE::priv, 其实就是当前按键在gpio_buttons数组中的索引)作为参数调用处理函数. 我们查看gpio_buttons数组中的各个按键对应的处理函数可以发现,除了电源键外,其他所有按键对应的处理函数都是keypad_handler
        keypad_handler普通功能键的处理函数)
           根据传入的priv(实际上是索引值)取得按键信息结构
           根据传入的按键状态, 将从按键信息结构体中取得的按键值和按键状态给系统处理(handle_scancode).这里与上面定时扫描部分相同
           (这里有点不大明白的是,为什么不直接传按键值或按键信息结构指针而是要传索引号呢?)
        pwButtonHandler电源键的处理函数)
           如果电源键之前未被按下
              打印一条信息
              任务调度power_button_task::power_button_task_handler, 这里函数将调用pm_user_suspend使系统进入休眠状态
           如果电源建之前已经被按下
              设置电源键新状态为未按
    如果没有处理函数,则将按键的扫描值与按键状态发送给系统处理(handle_scancode).

    模块卸载exit_gpio_button就是将一些申请的资源释放掉

总结: S3C2410的按键驱动模块分两部分:功能键(包括电源键)和普通键
功能键全部用irq来实现,这样就可以实现同时按多个功能键以及功能键与普通键的组合. 其主数据结构数组为gpio_buttons
普通键用定时扫描的方法取得按键值.其初始化函数为s3c2410tk_button_init, 扫描算法函数为tk_button_check_timer_handler, 其数据结构数组为gpio_buttons_tk
PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
5
关闭 站长推荐上一条 /3 下一条