syscall
理论
用户应用可以通过两种方式使用系统调用。
-
第一种方式是通过C库函数,包括系统调用在C库中的封装函数和其他普通函数。
-
第二种方式是使用_syscall宏。2.6.18版本之前的内核,在include/asm-i386/unistd.h文件中定义有7个_syscall宏,分别是:
_syscall0(type,name)
_syscall1(type,name,type1,arg1)
_syscall2(type,name,type1,arg1,type2,arg2)
_syscall3(type,name,type1,arg1,type2,arg2,type3,arg3)
_syscall4(type,name,type1,arg1,type2,arg2,type3,
arg3,type4,arg4)
_syscall5(type,name,type1,arg1,type2,arg2,type3,
arg3,type4,arg4,type5,arg5)
_syscall6(type,name,type1,arg1,type2,arg2,type3,
arg3,type4,arg4,type5,arg5,type6,arg6)
但是自2.6.19版本开始,_syscall宏被废除,我们需要使用syscall函数,通过指定系统调用号和一组参数来调用系统调用。
/*
* 作用: syscall() 执行一个系统调用,根据指定的参数number和所有系统调用的汇编语言接口来确定调用哪个系统调用。
* 系统调用所使用的符号常量可以在头文件里面找到。
* 参数: number是系统调用号,number后面应顺序接上该系统调用的所有参数
*/
int syscall(int number, ...);
实践
在linux下每一个进程都一个进程id,类型pid_t,可以由 getpid()获取。
POSIX线程也有线程id,类型pthread_t,可以由 pthread_self()获取,线程id由线程库维护。
但是各个进程独立,所以会有不同进程中线程号相同节的情况。那么这样就会存在一个问题,我的进程p1中的线程pt1要与进程p2中的线程pt2通信怎么办,进程id不可以,线程id又可能重复,所以这里会有一个真实的线程id唯一标识,tid。glibc没有实现gettid的函数,所以我们可以通过linux下的系统调用 syscall(SYS_gettid) 来获得。
#include<stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <syscall.h>
pid_t gettid(void) {
return (pid_t)syscall(SYS_gettid);
}
int main()
{
printf("%d", gettid());
}