这些小活动你都参加了吗?快来围观一下吧!>>
电子产品世界 » 论坛首页 » DIY与开源设计 » 开源硬件 » 芯灵思SinlinxA64开发板Linux内核 tasklet 机制(附实测代码

共2条 1/1 1 跳转至

芯灵思SinlinxA64开发板Linux内核 tasklet 机制(附实测代码)

助工
2019-03-12 16:45:24     打赏

Linux 中断编程分为中断顶半部,中断底半部
中断顶半部: 做紧急,耗时短的事情,同时还启动中断底半部。
中断底半部: 做耗时的事件,这个事件在执行过程可以被中断。
中断底半部实现方法: tasklet,工作队列,软中断等机制实现。实际上是把耗时事件推后执行,不在中断程序执行。

什么是tasklet?
Tasklet 一词的原意是“小片任务”的意思,这里是指一小段可执行的代码,且通常以函数的形式出现。这个 tasklet 绑定的函数在一个时刻只能在一个 CPU 上运行  ,tasklet(小任务)机制是中断处理下半部分最常用的一种方法,其使用也是非常简单的。一个使用 tasklet 的中断程序首先会通过执行中断处理程序来快速完成上半部分的工作,接着通过调用 tasklet 使得下半部分的工作得以完成。可以看到,下半部分被上半部分所调用,至于下半部分何时执行则属于内核的工作。

tasklet 机制核心数据结构

tasklet 机制核心数据结构
Interrupt.h linux-3.5\include\Linux
struct tasklet_struct
{
      struct tasklet_struct *next; // tasklet_struct 结构链表
      unsigned long state; //当前这个 tasklet 是否已经被调度
      atomic_t count;
      void (*func)(unsigned long); //指向 tasklet 绑定的函数的指针
      unsigned long data; //传递给tasklet 绑定的函数的参数

};

tasklet 相关 API
初始化相关
1. 静态初始化 DECLARE_TASKLET(name, func, data)
作用:定义一个名字为 name 的 tasklet_struct 结构变量,并且初始化这个结构。 所定义的这个 tasklet 是可以被调度,默认是处于被使能状态。

2. 静态初始化 DECLARE_TASKLET_DISABLED(name, func, data)
作用:定义一个名字为 name 的 tasklet_struct 结构变量,并且初始化这个结构。所定义的这个 tasklet 是不能被调度,默认是处于被禁能状态。 要调度这个 tasklet 需要先使能。

3.动态初始化
void tasklet_init(struct tasklet_struct *t,void (*func)(unsigned long), unsigned long data)
作用:初始化一个 tasklet_struct 结构变量,初始化的结构默认是处于激活状态,可以被调度。

tasklet使能函数
1. void tasklet_disable(struct tasklet_struct *t)
作用:函数激活给定的 tasklet被 tasklet_schedule 调度
2. void tasklet_enable (struct tasklet_struct *t)
作用:函数禁止给定的 tasklet被 tasklet_schedule 调度
tasklet 调度函数  
void tasklet_schedule (struct tasklet_struct *t)
作用:调用 tasklet_schedule 函数去通知内核帮我们调度所绑定的函数
void tasklet_kill(struct tasklet_struct *t);
作用:取消调度函数
编程步骤
Step1  定义并静态初始化tasklet_struct 结构变量
Step2  编写tasklet服务函数
Step3  在适当的地地方进行调度
Step4  在适当的地地方取消调度


实验平台:芯灵思SINA64开发板
嵌入式linux 开发板交流 QQ:641395230
驱动代码:

#include <linux/module.h>
#include <linux/init.h>
#include <linux/interrupt.h>

void tasklet_fun(unsigned long data);
//Step1 定义并静态初始化tasklet_struct 结构变量
DECLARE_TASKLET(mytasklet, tasklet_fun, 651);
//Step2 tasklet服务函数
void tasklet_fun(unsigned long data)
{
        static unsigned long count = 0;
        printk("count:%lu,%s is call! data:%lu\r\n",count++,__FUNCTION__,data);
        tasklet_schedule(&mytasklet); //在工作函数中重新调度自己,这样会循环调用tasklet_fun
}
static int __init mytasklet_init(void)
{
        //Step3 开始调度 mytasklet
        tasklet_schedule(&mytasklet);
        printk("%s is call!\r\n",__FUNCTION__);
        return 0;
}
static void __exit mytasklet_exit(void) //Module exit function specified by module_exit()
{
        //Step4 删除 tasklet
        tasklet_kill(&mytasklet);
}
module_init(mytasklet_init);
module_exit(mytasklet_exit);
MODULE_LICENSE("GPL");






管理员
2019-03-13 09:40:28     打赏
2楼

谢谢楼主分享


共2条 1/1 1 跳转至

回复

匿名不能发帖!请先 [ 登陆 注册 ]