自定义回调函数原型的方法

 回调函数是不能显式调用的函数;通过将回调函数的地址传给调用者从而实现调用。
    回调函数使用是必要的,在我们想通过一个统一接口实现不同的内容,这时用回掉函数非常合适。比如,我们为几个不同的设备分别写了不同的显示函数:void TVshow(); void ComputerShow(); void NoteBookShow()...等等。这是我们想用一个统一的显示函数,我们这时就可以用回掉函数了。
    void show(void (*ptr)());
    使用时根据所传入的参数不同而调用不同的回调函数。

    不同的编程语言可能有不同的语法,下面举一个c语言中回调函数的例子,其中一个回调函数不带参数,另一个回调函数带参数。

       例子1:

//Test.c

//Test.h
#include <stdlib.h>
#include <stdio.h>

int Test1()
{
 for (int i=0; i<30; i++)
 {
  printf("The %d th charactor is: %c/n", i, (char)('a' + i%26));  
 }
 return 0;
}
int Test2(int num)

 for (int i=0; i<num; i++)
 {
  printf("The %d th charactor is: %c/n", i, (char)('a' + i%26));  
 }
 return 0;
}

void Caller1(int (*ptr)())//指向函数的指针作函数参数
{
 (*ptr)();
}

void Caller2(int n, int (*ptr)(int n))//指向函数的指针作函数参数,这里第一个参数是为指向函数的指针服务的,
{           
 (*ptr)(n);
}

int main( int argc, char *argv[], char *envp[] )
{
 printf("************************/n");
 Caller1(Test1); //相当于调用Test1();
 
 printf("&&&&&&************************/n");
 Caller2(30, Test2); //相当于调用Test2(30);
 return 0;
}

       以上通过将回调函数的地址传给调用者从而实现调用,但是需要注意的是带参回调函数的用法。要实现回调,必须首先定义函数指针。函数指针的定义这里稍微提一下。比如:

     int (*ptr)(int n); 这里ptr是一个函数指针,其中(*ptr)的括号不能省略,因为括号的优先级高于星号,那样就成了一个返回类型为整型的函数声明了。

=================================
转载地址: http://zq2007.blog.hexun.com/9068988_d.html

注: 狐狸认为, 回调函数实现多态函数只是附带功能之一, 最主要是提供了一种代码动态嵌入的手段, 实现函数内部中断的一种方式.

在 Linux 内核中,**`tasklet` 的自定义回调函数是通过初始化 `struct tasklet_struct` 时指定的函数指针实现的**。你可以在模块或驱动中定义任意符合签名的函数,并将其注册为 tasklet 的处理函数。 --- ### ✅ 1. 回调函数原型 `tasklet` 的回调函数必须遵循以下固定格式: ```c void my_tasklet_handler(unsigned long data); ``` - 返回值:`void` - 参数:一个 `unsigned long` 类型的数据(常用于传递参数或状态) > 这是因为内核的 `tasklet_action` 在执行时会这样调用: > > ```c > t->func(t->data); > ``` --- ### ✅ 2. 定义自定义回调函数 你可以像下面这样定义自己的处理逻辑: ```c static void my_custom_tasklet_func(unsigned long data) { printk(KERN_INFO "My custom tasklet running! Data: %lu\n", data); // 在这里执行你想做的任务(不能睡眠!) // 例如:处理硬件状态、调度定时器、唤醒工作队列等 } ``` --- ### ✅ 3. 初始化 tasklet 并绑定回调函数 有两种方式来初始化并绑定回调函数。 #### 方法一:使用 `tasklet_init()` 动态初始化(推荐) 适用于动态分配或需要传参的情况。 ```c static struct tasklet_struct my_tasklet; // 在模块初始化中: tasklet_init(&my_tasklet, my_custom_tasklet_func, 42); // 42 是传给 data 的值 ``` #### 方法二:使用宏静态声明 ```c // 定义并初始化一个激活的 tasklet DECLARE_TASKLET(my_tasklet, my_custom_tasklet_func, 42); // 或者定义一个初始禁用的 tasklet DECLARE_TASKLET_DISABLED(my_tasklet, my_custom_tasklet_func, 42); ``` > ⚠️ 注意:宏中的第三个参数必须是编译期常量(因为是宏展开),而 `tasklet_init` 可以运行时传参。 --- ### ✅ 4. 调度 tasklet 执行 当你要触发这个回调函数时,调用: ```c tasklet_schedule(&my_tasklet); ``` 这会将 tasklet 标记为可执行,并触发 `TASKLET_SOFTIRQ` 软中断,在稍后执行你的 `my_custom_tasklet_func` 函数。 --- ### ✅ 5. 完整示例代码(内核模块) ```c #include <linux/module.h> #include <linux/kernel.h> #include <linux/interrupt.h> static struct tasklet_struct my_tasklet; // 自定义回调函数 static void my_custom_tasklet_func(unsigned long data) { printk(KERN_INFO "Tasklet running: received data = %lu\n", data); // 示例:做一些快速处理 if (data == 100) { printk(KERN_INFO "Special operation triggered!\n"); } } // 模拟中断处理上半部 static void simulate_interrupt(void) { printk(KERN_INFO "Simulated interrupt: scheduling tasklet...\n"); tasklet_schedule(&my_tasklet); } static int __init tasklet_demo_init(void) { unsigned long my_data = 100; // 初始化 tasklet,绑定自定义函数和参数 tasklet_init(&my_tasklet, my_custom_tasklet_func, my_data); printk(KERN_INFO "Tasklet demo module loaded.\n"); // 模拟一次“中断” simulate_interrupt(); return 0; } static void __exit tasklet_demo_exit(void { // 等待 tasklet 执行完毕后再清理 tasklet_kill(&my_tasklet); printk(KERN_INFO "Tasklet demo exited.\n"); } module_init(tasklet_demo_init); module_exit(tasklet_demo_exit); MODULE_LICENSE("GPL"); MODULE_DESCRIPTION("Custom tasklet callback example"); ``` --- ### ✅ 6. 如何传递复杂数据? 由于 `data` 是 `unsigned long`,你可以: - 直接传递整数; - 强制转换指针(需确保内存有效): ```c struct my_dev *dev = kmalloc(sizeof(*dev), GFP_ATOMIC); tasklet_init(&my_tasklet, my_handler, (unsigned long)dev); ``` 然后在回调中还原: ```c static void my_handler(unsigned long data) { struct my_dev *dev = (struct my_dev *)data; // 使用 dev... } ``` > ⚠️ 注意:要保证该指针指向的内存不会在 tasklet 执行前被释放! --- ### ✅ 7. 注意事项 | 项目 | 说明 | |------|------| | ❌ 不可睡眠 | 回调函数运行在软中断上下文,不能调用 `sleep`, `msleep`, `schedule`, `copy_to_user` 等 | | ✅ 可以上锁 | 可使用自旋锁(`spinlock_t`)保护共享资源 | | ✅ 支持多实例 | 可创建多个不同的 tasklet,每个绑定不同函数或参数 | | ✅ 安全释放 | 使用 `tasklet_kill()` 防止 race condition | ---
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值