可重入函数与不可重入函数概念以及编写规范

1、定义

一个可重入的函数简单来说就是可以被中断的函数,也就是说,可以在这个函数执行的任何时刻中断它,转入OS调度下去执行另外一段代码,而返回控制时不会出现什么错误;而不可重入的函数由于使用了一些系统资源,比如全局变量区,中断向量表等,所以它如果被中断的话,可能会出现问题,这类函数是不能运行在多任务环境下的。

换句话说,我们也可以这样理解,重入即表示重复进入,首先它意味着这个函数可以被中断,其次意味着它除了使用自己栈上的变量以外不依赖于任何环境(包括static),这样的函数就是 purecode(纯代码)可重入,可以允许有该函数的多个副本在运行,由于它们使用的是分离的栈,所以不会互相干扰。如果确实需要访问全局变量(包括static),一定要注意实施互斥手段。可重入函数在并行运行环境中非常重要,但是一般要为访问全局变量付出一些性能代价。

编写可重入函数时,若使用全局变量,则应通过关中断、信号量(即P、V操作)等手段对其加以保护。

说明:若对所使用的全局变量不加以保护,则此函数就不具有可重入性,即当多个进程调用此函数时,很有可能使有关全局变量变为不可知状态。

2、案例

void swap1(int* x, int* y) {tmp=*x;*x=*y;*y=tmp;}void swap2(int* x, int* y) {int tmp;tmp=*x;*x=*y;*y=tmp;}

swap1是不可重入的,swap是可重入的。因为在多线程条件下,操作系统会在swap1还没有执行完的情况下,切换到另一个线程中,那个线程可能再次调用swap1,这样状态就错了。

假如我们引入了全局变量到重入函数中比如下面,这个案例

unsigned int example( int para ){unsigned int temp;Exam = para; // (**)temp = Square_Exam( );return temp;}此函数若被多个进程调用的话,其结果可能是未知的,因为当(**)语句刚执行完后,另外一个使用本函数的进程可能正好被激活,那么当新激活的进程执行到此函数时,将使Exam赋与另一个不同的para值,所以当控制重新回到“temp = Square_Exam( )”后,计算出的temp很可能不是预想中的结果。此函数应如下改进。unsigned int example( int para ) {unsigned int temp;[申请信号量操作] //(1)Exam = para;temp = Square_Exam( );[释放信号量操作]return temp;}

倘若上述中Square_Exam( ) 依旧是一个重入函数,在执行过程中,继续调用 example 这个函数的时候(函数递归),但是由于由于主函数没有立即释放掉信号量,此时就容易造成死锁()。保证函数的可重入性的方法:

,这样构成的函数就一定是一个可重入的函数。 VxWorks中采取的可重入的技术有: * 动态堆栈变量(各子函数有自己独立的堆栈空间) * 受保护的全局变量和静态变量 * 任务变量

————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————————–

3、不可重入怎么改改成可重入函数

。不可重入函数在实时系统设计中被视为不安全函数。满足下列条件的函数多数是不可重入的: 1) 函数体内使用了静态的数据结构; 2) 函数体内调用了malloc()或者free()函数; 3) 函数体内调用了标准I/O函数。 下面举例加以说明。 A. 可重入函数 void strcpy(char *lpszDest, char *lpszSrc)

{ while(*lpszDest++=*lpszSrc++); *dest=0; } B.不可重入函数1 charcTemp;//全局变量 void SwapChar1(char *lpcX, char *lpcY)

{ cTemp=*lpcX; *lpcX=*lpcY; lpcY=cTemp;//访问了全局变量 } C. 不可重入函数2 void SwapChar2(char *lpcX,char *lpcY)

{ static char cTemp;//静态局部变量 cTemp=*lpcX; *lpcX=*lpcY; lpcY=cTemp;//使用了静态局部变量 } 问题1,如何编写可重入的函数? 答:在函数体内不访问那些全局变量,不使用静态局部变量,坚持只使用局部变量,写出的函数就将是可重入的。如果必须访问全局变量,记住利用互斥信号量来保护全局变量。 问题2,如何将一个不可重入的函数改写成可重入的函数? 答:把一个不可重入函数变成可重入的唯一方法是用可重入规则来重写它。其实很简单,只要遵守了几条很容易理解的规则,那么写出来的函数就是可重入的。 1) 不要使用全局变量。因为别的代码很可能覆盖这些变量值。 2) 在和硬件发生交互的时候,,切记执行类似 disinterrupt() 之类的操作,就是关闭硬件中断。完成交互记得打开中断,在有些系列上,这叫做“进入/退出核心”。 3) 不能调用其它任何不可重入的函数。 4) 谨慎使用堆栈。最好先在使用前先OS_ENTER_KERNAL。 堆栈操作涉及内存分配,稍不留神就会造成益出导致覆盖其他任务的数据,所以,请谨慎使用堆栈!最好别用!很多黑客程序就利用了这一点以便系统执行非法代码从而轻松获得系统控制权。还有一些规则,总之,时刻记住一句话:保证中断是安全的! 实例问题:曾经设计过如下一个函数,在代码检视的时候被提醒有bug,因为这个函数是不可重入的,为什么? unsigned int sum_int( unsigned int base )

人之所以能,是相信能。

可重入函数与不可重入函数概念以及编写规范

相关文章:

你感兴趣的文章:

标签云: