深入浅出剖析C语言函数指针与回调函数(二)

上一篇博文的地址:

http://blog.csdn.net/morixinguan/article/details/65494239

这节,我们来看看函数指针与回调函数在Linux内核中的应用。

从上节我们了解到,函数指针和回调函数在开发者和用户之间的一个例子,那么这节,我将引用Linux内核中文件操作结构体来详细的说明。

我们首先来看到这个结构体,这段代码位于linux内核的include/linux/fs.h中,由于代码众多,我只截取几个最基本的例子:

File_operations文件操作结构体:

这段代码中,利用结构体的封装思想,将函数指针封装在一个file_operations结构体里,然后,在具体实现驱动的时候,实现具体的函数,再赋值给结构体里的函数指针做好初始化操作,我们来看看友善之臂的led驱动就明白了。

以下这段代码截取友善之臂提供的linux内核中的tiny4412_leds.c

首先,先是定义了一个结构体变量,并对结构体变量进行初始化,在这个驱动中,只实现了ioctl函数,对照着上面的结构体,ulocked_ioctl就是结构体中的这个函数指针。

long (*unlocked_ioctl) (struct file *,unsigned int, unsigned long);

再来看看友善实现的adc驱动里,也是这么来做,这里看到 : 也是C语言结构体的一种初始化方式,也是合理的。

在内核中,有很多这样的函数指针,所以,当我们了解了这样的套路以后,再去学习linux内核,我们的思想就会清晰很多了。

再来看看回调函数在linux内核里的基本应用。

从上节我们了解到,回调函数的本质其实也就是函数指针,只不过定义有所区别。它的定义就是:你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。

接下来我们来看一个例子:

这段代码摘自友善之臂的button驱动:

我们在tiny4412_buttons_open函数里看到

err = request_irq(irq, button_interrupt, IRQ_TYPE_EDGE_BOTH,

buttons[i].name,(void *)&buttons[i]);

我们来看看request_irq这个函数:

这个函数的作用是请求中断,我们来看看函数的第二个参数irq_handler_t handler是什么?

到这里我们就明白了,第二个参数是一个用typedef重新定义的一个新类型的函数指针。

那么也就是说一旦执行了tiny4412的open函数,就会通过request_irq去通过回调函数去执行按键中断,并返回一个中断句柄。这个回调函数,其实就是一个中断服务函数。

回调函数在内核中就是这么来使用的,当然,还有其它的,比如我们在tiny4412的open函数里面还看到:

setup_timer(&buttons[i].timer,tiny4412_buttons_timer,

(unsignedlong)&buttons[i]);

这个函数的作用是注册一个定时器,通过回调函数tiny4412_buttons_timer来进行触发。

如果你不看它的定义,你可能以为它是一个普通函数,其实它是一个宏函数。

这个宏函数通过调用setup_timer_key这个函数来实现定时器的注册:

通过这一节,我们了解到回调函数在Linux内核中的应用,为学习Linux内核,分析linux内核源代码打下了基础。

时间: 2024-10-04 17:36:45

深入浅出剖析C语言函数指针与回调函数(二)的相关文章

【C语言】函数指针与回调函数

在C语言中:指针是C语言的特色,有着各种各样的指针,普通的变量指针,常量指针,数组指针,指针数组,函数指针,指针函数.我们就讲一下函数指针与回调函数吧 首先关于函数指针,其实很简单. 对于一个函数指针来说,顾名思义,就是一个指向函数的指针,需要知道的是,对于指针而言,他总是存储一块地址,地址里面有着一个,一组,或者一块数据,在函数中,函数的存储是放在代码段的,每个函数都有着一个函数首地址,调用了这个地址相当于调用的这个函数. 具体的可以观看我的这篇博客,其中就通过在内存阶段改变栈帧返回值,成功的

函数指针,回调函数

函数指针的定义:返回值类型 ( * 指针变量名) (形参列表); 1:"返回值类型"说明函数的返回类型,"(指针变量名 )"中的括号不能省,括号改变了运算符的优先级.若省略整体则成为一个函数说明,说明了一个返回的数据类型是指针的函数,后面的"形参列表"表示指针变量指向的函数所带的参数列表. int func(int x); /* 声明一个函数 */ int (*f) (int x); /* 声明一个函数指针 */ f=func; /* 将func

函数指针、回调函数、系统调用区别

函数指针.回调函数.系统调用区别1.函数指针 1 int add(int a, int b) 2 { 3 return a+b; 4 } 5 6 int (*fp)(int,int); 7 8 int main() 9 { 10 int c; 11 fp = add; //有指针赋值操作,静态绑定 12 c = fp(2,4); 13 printf("c = %d\n",c); 14 return 0; 15 } 2.回调函数 1 int add(int a, int b) //回调函

c++函数指针用于回调函数

#include "stdafx.h" typedef int (*func)(int,int);class data{public: data() { a=12; b=13; } ~data() { } static int setdata(int c,int d) { return c+d; } int a; int b;protected:private: };void test(func _func,void* _data){ data* mydata=(data*)_data

深入浅出剖析C语言函数指针与回调函数(一)【转】

本文转载自:http://blog.csdn.net/morixinguan/article/details/65494239 关于静态库和动态库的使用和制作方法. http://blog.csdn.NET/morixinguan/article/details/52451612 今天我们要搞明白的一个概念叫回调函数. 什么是回调函数? 百度的权威解释如下: 回调函数就是一个通过函数指针调用的函数.如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说

函数指针与回调函数

一.函数指针 1.  函数指针就是一个指针变量,用来指向函数地址.正在运行的程序(进程)在内存中占据一定的空间.进程包括编译好的程序代码和需要使用的变量.因此,程序代码中的函数就是一些字符域,要得到一个函数地址,也就是得到这些字符域的起始地址. 2. 函数指针的三种形式:    指向普通C函数的函数指针 --- C语言中的函数指针    指向C++类静态成员函数的函数指针 --- C++    指向C++类非静态成员函数的函数指针 --- C++ 函数指针的本质自然也就是函数地址.  类成员函数

函数指针,回调函数的定义和使用

一.函数指针 定义:函数指针是指向函数的指针变量,即本质是一个指针变量. int (*f) (int x); /* 声明一个函数指针 */ f=func; /* 将func函数的首地址赋给指针f */ 指向函数的指针包含了函数的地址,可以通过它来调用函数.声明格式如下:  类型说明符 (*函数名)(参数)其实这里不能称为函数名,应该叫做指针的变量名.这个特殊的指针指向一个返回整型值的函数.指针的声明笔削和它指向函数的声明保持一致.指针名和指针运算符外面的括号改变了默认的运算符优先级.如果没有圆括

函数指针和回调函数

函数指针 函数指针是指向函数调用地址的指针.它和函数名究竟有什么关系呢?且看下文. 且看一小程序 首先,先请看下边程序: 1 #include <iostream> 2 #include <string> 3 using namespace std; 4 5 void func(string s) 6 { 7 cout << s << endl; 8 } 9 10 void (*pFunc)(string s); // 不能够写成 void *pFunc(s

函数指针和回调函数的事

一. 函数指针 关于函数指针的概念,可以想到一个整型指针指向的是一个整型,它的值是所指向对象的地址:一个字符串指针指向的是一个字符串,它的值是所指向字符串的首地址:因此,一个函数指针当然是一个指针变量了,它所指向的是一个函数,它的值就是所指向函数的入口地址. 函数指针的定义如下: typedef int data_type; data_type (*pfun)(data_type, data_type); 上面的语句中定义了一个函数指针pfun,它表示指向一个返回值为data_type,参数为两