可重入函数与不可重入函数 —— Linux编程

1、定义

一个可重入的函数简单来说就是可以被中断的函数,也就是说,可以在这个函数执行的任何时刻中断它,转入OS调度下去执行另外一段代码,而返回控制时不会出现什么错误;而不可重入的函数由于使用了一些系统资源,比如全局变量区,中断向量表等,所以它如果被中断的话,可能会出现问题,这类函数是不能运行在多任务环境下的。可重入函数(用于信号处理函数、 且 安全时的叫法),即是在信号处理函数中可以调用的函数,他们是安全的,不安全的如malloc(试想:线程正在调用malloc进行分配,而信号来了,在处理函数里面有调用malloc,那么就很有可能对进程造成破坏,破坏储存区维护的链表)、getpwnam等。

换句话说,我们也可以这样理解,重入即表示重复进入,首先它意味着这个函数可以被中断,其次意味着它除了使用自己栈上的变量以外不依赖于任何环境(包括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 这个函数的时
候(函数递归),但是由于由于主函数没有立即释放掉信号量,此时就容易造成死锁
(http://blog.csdn.net/xy010902100449/article/details/45896055)。
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;//使用了静态局部变量
 }

保证函数的可重入性的方法:

  在写函数时候尽量使用局部变量(例如寄存器、堆栈中的变量),对于要使用的全局变量要加以保护(如采取关中断、信号量等方法),这样构成的函数就一定是一个可重入的函数。
    VxWorks中采取的可重入的技术有:
    * 动态堆栈变量(各子函数有自己独立的堆栈空间)
    * 受保护的全局变量和静态变量
    * 任务变量

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

在实时系统的设计中,经常会出现多个任务调用同一个函数的情况。如果这个函数不幸被设计成为不可重入的函数的话,那么不同任务调用这个函数时可能修改其他任务调用这个函数的数据,从而导致不可预料的后果。那么什么是可重入函数呢?所谓可重入函数是指一个可以被多个任务调用的过程,任务在调用时不必担心数据是否会出错。不可重入函数在实时系统设计中被视为不安全函数。满足下列条件的函数多数是不可重入的:
    1) 函数体内使用了静态的数据结构;
    2) 函数体内调用了malloc()或者free()函数;
    3) 函数体内调用了标准I/O函数。

问题1,如何编写可重入的函数?
    答:在函数体内不访问那些全局变量,不使用静态局部变量,坚持只使用局部变量,写出的函数就将是可重入的。如果必须访问全局变量,记住利用互斥信号量来保护全局变量
问题2,如何将一个不可重入的函数改写成可重入的函数?
    答:把一个不可重入函数变成可重入的唯一方法是用可重入规则来重写它。其实很简单,只要遵守了几条很容易理解的规则,那么写出来的函数就是可重入的。
    1) 不要使用全局变量。因为别的代码很可能覆盖这些变量值。
    2) 在和硬件发生交互的时候,切记执行类似 disinterrupt() 之类的操作,就是关闭硬件中断。完成交互记得打开中断,在有些系列上,这叫做“进入/退出核心”。
    3) 不能调用其它任何不可重入的函数。
    4) 谨慎使用堆栈。最好先在使用前先OS_ENTER_KERNAL。

    堆栈操作涉及内存分配,稍不留神就会造成益出导致覆盖其他任务的数据,所以,请谨慎使用堆栈!最好别用!很多黑客程序就利用了这一点以便系统执行非法代码从而轻松获得系统控制权。还有一些规则,总之,时刻记住一句话:保证中断是安全的!

    实例问题:曾经设计过如下一个函数,在代码检视的时候被提醒有bug,因为这个函数是不可重入的,为什么? 

    unsigned int sum_int( unsigned int base )
    {
        unsigned int index;
        static unsigned int sum = 0; // 注意,是static类型
        for (index = 1; index <= base; index++)
            sum += index;
        return sum;
    }

    分析:所谓的函数是可重入的(也可以说是可预测的),即只要输入数据相同就应产生相同的输出。
这个函数之所以是不可预测的,就是因为函数中使用了static变量,因为static变量的特征,这样的函
数被称为:带“内部存储器”功能的的函数。因此如果需要一个可重入的函数,一定要避免函数中使用
static变量,这种函数中的static变量,使用原则是,能不用尽量不用。
    将上面的函数修改为可重入的函数,只要将声明sum变量中的static关键字去掉,变量sum即变为一
个auto类型的变量,函数即变为一个可重入的函数。
    当然,有些时候,在函数中是必须要使用static变量的,比如当某函数的返回值为指针类型时,则必
须是static的局部变量的地址作为返回值,若为auto类型,则返回为错指针

4、可重入函数与信号

可重入函数在处理操作期间,会阻塞任何会引起不一致的信号发送。

试想下面一个情况:

因为在信号处理函数执行时,如果里面执行了低俗系统调用 and 此信号处理函数没有屏蔽其它信号(如SIGUSR1),那么这时若果产生了SIGUSR1信号,那么由将产生中断,调用SIGUSR1的处理函数:

#include <unistd.h>
#include <signal.h>
#include <stdio.h>
 
void signal_1(int sig)
{
    printf("Start signal_1...\n");
    //sleep(20);
    malloc memory ...
    printf("End signal_1\n");
}
void signal_2(int sig)
{
    malloc memory ...
    printf("Process signal_2 ,OK\n");
}
 
int main(int argc, char *argv[])
{
    sigset_t set_1;
    sigemptyset(&set_1);
    struct sigaction sa_1;
    sa_1.sa_handler = &signal_1;
    sa_1.sa_mask = set_1;
    sa_1.sa_flags = SA_RESTART;
    sigset_t set_2;
    sigemptyset(&set_2);
    struct sigaction sa_2;
    sa_2.sa_handler = &signal_2;
    sa_2.sa_mask = set_2;
    sa_2.sa_flags = SA_RESTART;
    
    sigaction(SIGUSR1, &sa_1, NULL);
    sigaction(SIGUSR2, &sa_2, NULL);
    printf("Start pause...(PID:%d)\n", getpid());
    pause();
    return 0;
}

上面程序运行,先发送SIGUSR1 然后SIGUSR2,(不注释sleep忽略memory )输出:

 $ ./sigproc
Start pause...(PID:7089)
Start signal_1...
Process signal_2 ,OK
End signal_1

 如果是使用了memory调用,好明显这两个信号处理函数也是不可重入的;

那么我们可以知道,如果此时阻塞了SIGUSR2信号,那么signal_1不就安全了吗(忽略其它信号也有影响的情况先)

对的,如果使用sigaction调用了设定了阻塞SIGUSR2,那么就行了,可重入函数就是基于这种思想,看上面粗线那句话。

补充:

errno是线程安全的,也就是每个线程会访问局部的errno

在一个main函数中,如果线程函数里调用低俗系统调用,那么就有可能产生错误、中断,那么errno值会被改变,而main函数调用也出错时,那么errno值就会被改变,

所以,在作为一个通用规则,处理函数最好的写法是:

void sigproc(int sig)
{
	int tmp = errno;
	//信号处理函数真正的操作开始...
 
      //处理函数任务完成
	errno = tmp;
}

引自:

https://blog.csdn.net/xy010902100449/article/details/46475735

https://blog.csdn.net/Jammg/article/details/51839479

猜你喜欢

转载自blog.csdn.net/u011285208/article/details/83003660