static inline _syscall0(int,fork)
其中_syscall0()是unistd.h中的内嵌宏代码,它以嵌入汇编的形式调用Linux的系统调用中断int 0x80。根据include/unistd.h文件第133行上的宏定义,我们把这个宏展开并替代进上面一行中就可以看出这条语句实际上是int fork()创建进程系统调用,见如下所示。
// unistd.h文件中_syscall0()的定义。即为不带参数的系统调用宏函数:type name(void)。
133 #define _syscall0(type,name) \
134 type name(void) \
135 { \
136 long __res; \
137 __asm__ volatile ("int $0x80" \ // 调用系统中断0x80。
138 : "=a" (__res) \ // 返回值eax(__res)。
139 : "0" (__NR_##name)); \ // 输入为系统中断调用号__NR_name。
140 if (__res >= 0) \ // 如果返回值>=0,则直接返回该值。
141 return (type) __res; \
142 errno = -__res; \ // 否则置出错号,并返回-1。
143 return -1; \
144 }
// 根据上面定义把_syscall0(int, fork)展开代进第23行后我们可以得到如下语句:
static inline int fork(void)
{
long __res;
__asm__ volatile ("int $0x80" : "=a" (__res) : "0" (__NR_fork));
if (__res >= 0)
return (int) __res;
errno = -__res;
return -1;
}
其中_syscall0()是unistd.h中的内嵌宏代码,它以嵌入汇编的形式调用Linux的系统调用中断int 0x80。根据include/unistd.h文件第133行上的宏定义,我们把这个宏展开并替代进上面一行中就可以看出这条语句实际上是int fork()创建进程系统调用,见如下所示。
// unistd.h文件中_syscall0()的定义。即为不带参数的系统调用宏函数:type name(void)。
133 #define _syscall0(type,name) \
134 type name(void) \
135 { \
136 long __res; \
137 __asm__ volatile ("int $0x80" \
138
139
140 if (__res >= 0) \
141
142 errno = -__res; \
143 return -1; \
144 }
// 根据上面定义把_syscall0(int, fork)展开代进第23行后我们可以得到如下语句:
static inline int fork(void)
{
long __res;
__asm__ volatile ("int $0x80" : "=a" (__res) : "0" (__NR_fork));
if (__res >= 0)
errno = -__res;
return -1;
}
在编译链接前会对源文件进行预处理,它会将源文件中的头文件以及宏(在这里就是把用#define定义的_syscall宏展开)都展开。
gcc会把上述“函数”体中的语句直接插入到调用fork()语句的代码处,因此执行fork()不会引起函数调用。另外,宏名称字符串“syscall0”中最后的0表示无参数,1表示带1个参数。如果系统调用带有1个参数,那么就应该使用宏_syscall1()。
同时可以参考#define中的无参和有参宏定义#define