在linux上获得线程id的方法,在linux上获得线程id的方法

本文详细介绍了在Linux系统中,特别是在Linux2.4版本后,如何在用户态程序中获取线程ID的三种方法:gettid()系统调用、syscall()库函数以及pthread_self()。通过宏展开、汇编代码和pthread结构体解析,阐述了每种方法的工作原理和实现细节。
摘要由CSDN通过智能技术生成

在linux2.4版本后,linux使用了NPTL作为自己的线程库,为了兼容POSIX标准,所以在内核task中有两个域tgid和tid,前者是进程id,后者是线程id。在linux上获得线程id的方法,目前我所知的有三种,当然这里的三种是指在用户态的程序中,否则除非自己写的kernel module, 都是调用编号224的系统调用实现的(2.6版本)。第一种: gettid(), man gettid 可以看到gettid的使用方式。使用时要先定义:_syscall0(pid_t, gettid)其中_syscall0是一个宏(由于参数的不同还有_syscall1,_syscall2...),定义如下:#define _syscall0(type,name) \type name(void) \{ \long __res; \__asm__ volatile ("int $0x80" \   //int 80, 软中断: "=a" (__res) \          //输入输出都用的eax: "0" (__NR_##name)); \   //#define __NR_gettid 224__syscall_return(type,__res); \   //返回tid}编译时,宏展开之后,相当于定义了一个pid_t gettid(void)函数,用内嵌汇编实现,在程序中就可以使用gettid()获得线程id了。第二种:syscall(), 名字叫syscall(),却是glibc中的库函数。使用方式:syscall(__NR_gettid), 其中__NR_gettid就是224,同上。syscall的实现要到glibc中去找,不同的硬件平台有不同的实现版本,在i386上的实现在syscall.S中:#include .textENTRY (syscall)PUSHARGS_6 /* Save register contents.  */_DOARGS_6(44) /* Load arguments.  */movl 20(%esp), %eax /* Load syscall number into %eax.  */ENTER_KERNEL /* Do the system call.  */POPARGS_6 /* Restore register contents.  */cmpl $-4095, %eax /* Check %eax for error.  */jae SYSCALL_ERROR_LABEL /* Jump to error handler if error.  */L(pseudo_end):ret /* Return to caller.  */PSEUDO_END (syscall)其中ENTRY也是一个宏,展开了相当的长,主要用于在链接的时候让gcc能够"看见"并调用这段用汇编写成的syscall()函数。第三种:pthread_self()同样是一个glibc提供的函数,在linux的manual中说返回的是当前线程的thread ID.但是实际你看到的是一个很长的,似乎没有规律的值。什么原因得看看它的实现:在glibc中,pthread_self()返回的是THREAD_SELF,这又是一个宏定义如下# define THREAD_SELF \({ struct pthread *__self;      \asm ("movl %%gs:%c1,%0" : "=r" (__self)      \: "i" (offsetof (struct pthread, header.self)));      \__self;})这段代码返回了当前线程的descriptor,pthread_self()得到的就是这个descriptor的地址, 也就是unsigned long int类型的pthread_t。知道了这一点就好办了,找到thread descriptor的定义:struct pthread{...pid_t tid;...}接下来知道怎么做了吗?算好长度n,构造一个假的pthread结构。struct pthread_fake{void *nothing[n];pid_t tid;};用(struct pthread_fake *) pthread_self()->tid得到线程id了相比前两种做法,这种无疑是最繁琐的,但是同理,可以获取很多glibc中维护了,但是没有提供访问方法的数据。

posted on 2010-12-28 14:34 大龙 阅读(997) 评论(0)  编辑 收藏 引用

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值