У errno.h ця змінна оголошується як extern int errno;
Ось що говорить стандарт C:
Макрос errno
не повинен бути ідентифікатором об'єкта. Він може розширитися до зміни значення, що змінюється в результаті виклику функції (наприклад, *errno()
).
Як правило, errno
це макрос, який викликає функцію, що повертає адресу номера помилки для поточного потоку, а потім відмінює її.
Ось що я маю в Linux, в /usr/include/bits/errno.h:
/* 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
Зрештою, він генерує такий тип коду:
> cat essai.c
#include <errno.h>
int
main(void)
{
errno = 0;
return 0;
}
> gcc -c -Wall -Wextra -pedantic essai.c
> objdump -d -M intel essai.o
essai.o: file format elf32-i386
Disassembly of section .text:
00000000 <main>:
0: 55 push ebp
1: 89 e5 mov ebp,esp
3: 83 e4 f0 and esp,0xfffffff0
6: e8 fc ff ff ff call 7 <main+0x7> ; get address of errno in EAX
b: c7 00 00 00 00 00 mov DWORD PTR [eax],0x0 ; store 0 in errno
11: b8 00 00 00 00 mov eax,0x0
16: 89 ec mov esp,ebp
18: 5d pop ebp
19: c3 ret