在linux或者unix環(huán)境中,errno是一個(gè)十分重要的部分。在調(diào)用的函數(shù)出現(xiàn)問題的時(shí)候,我們可以通過errno的值來確定出錯(cuò)的原因,這就會(huì) 涉及到一個(gè)問題,那就是如何保證errno在多線程或者進(jìn)程中安全?我們希望在多線程或者進(jìn)程中,每個(gè)線程或者進(jìn)程都擁有自己獨(dú)立和唯一的一個(gè) errno,這樣就能夠保證不會(huì)有競爭條件的出現(xiàn)。一般而言,編譯器會(huì)自動(dòng)保證errno的安全性,但是為了妥善期間,我們希望在寫makefile的時(shí) 候把_LIBC_REENTRANT宏定義,比如我們?cè)跈z查<bits/errno.h>文件中發(fā)現(xiàn)如下的定義:
# ifndef __ASSEMBLER__
/* Function to get address of global `errno' variable. */
extern int *__errno_location (void) __THROW __attribute__ ((__const__));
# if !defined _LIBC || defined _LIBC_REENTRANT
/* When using threads, errno is a per-thread value. */
# define errno (*__errno_location ())
# endif
# endif /* !__ASSEMBLER__ */
#endif /* _ERRNO_H */
也就是說,在沒有定義__LIBC或者定義_LIBC_REENTRANT的時(shí)候,errno是多線程/進(jìn)程安全的。
一般而言, __ASSEMBLER__, _LIBC和_LIBC_REENTRANT都不會(huì)被編譯器定義,但是如果我們定義_LIBC_REENTRANT一次又何妨那? <!--[if !vml]-->
<!--[endif]-->
為了檢測一下你編譯器是否定義上述變量,不妨使用下面一個(gè)簡單程序。
#include <stdio.h>
#include <errno.h>
int main( void )
{
#ifndef __ASSEMBLER__
printf( "Undefine __ASSEMBLER__\n" );
#else
printf( "define __ASSEMBLER__\n" );
#endif
#ifndef __LIBC
printf( "Undefine __LIBC\n" );
#else
printf( "define __LIBC\n" );
#endif
#ifndef _LIBC_REENTRANT
printf( "Undefine _LIBC_REENTRANT\n" );
#else
printf( "define _LIBC_REENTRANT\n" );
#endif
return 0;
}
希望讀者在進(jìn)行移植的時(shí)候,讀一下相關(guān)的unix版本的<bits/errno.h>文件,來確定應(yīng)該定義什么宏。不同的unix版本可能存在著一些小的差別!
<!--[if !supportLineBreakNewLine]-->
<!--[endif]-->