Volatile unsigned long* mcu_reset = (volatile unsigned long*)0x00080000;
(((void()())mcu_reset))()
下面分析一下这个表达式
第一步:
假定fp是一个函数指针,那么如何在程序中调用函数指针fp所指向的函数呢?
调用方法如下:
(*fp)();
因为fp是一个函数指针,那么*fp就是该指针指向的函数,在函数后面加上(),(*fp)(),就是调用该函数的方法。
那么接下来的操作就是如何使用一个恰当的表达式来替换fp,这将在第二步中解决、
第二步:
如果C编译器能够理解我们的大脑对与数据类型的认识,那么我们姑且可以这样写:
(*0x00080000)();
显然,上式并不能执行生效,因为运算符*必须要使用一个指针来做操作数,而且这个指针还必须是一个函数指针,这样经过运算符* 作用之后的结果才能作为函数被调用。因此,必须要对0x00080000做类型转换,转换后的类型大致可以描述为“指向返回值为void类型的函数的指针”。
如果fp是一个指向返回值是void类型的函数的指针,那么(*fp)()的值的类型为void,fp的声明如下:
void (*fp)();
因此,我们可以使用下面的方式来完成调用存储位置为0x00080000的程序:
void (*fp)();//声明一个指向返回值为void类型的函数的指针变量fp
(*fp)();//调用fp指针指向的函数
这种写法的代价是多声明的一个“哑”变量,但是,我们已经知道如何声明一个指向返回值为void类型的函数的指针变量fp,那么我们就可以对一个常数地址0x00080000进行类型转换,将其转换为fp变量的类型,只需要在变量声明中将变量名去掉即可,如下:
(void (*)() )0x00080000
这样就将常数地址0x00080000的类型强制转换为“指向返回值void的函数的指针”类型
因此,我们可以用void (*)()0x00080000来替换(*fp)()中的fp,从而得到:
(*(void (*)())0x00080000);
这样就解决了问题
如果使用typedef就会使表述更准确清晰,
typedef void (*func)();//重新定义func为void(*)()类型
(*(func)0x00080000)();