首页 | 资讯动态 | linux基础 | 系统管理 | 网络管理 | 编程开发 | linux数据库 | 服务器技术 | linux相关 | linux认证 | 嵌入式 | 下载中心 | 专题 | linux招聘 | HR | 镜像
OKLinux中文技术站
·设为首页
·加入收藏
·联系我们
系统管理: 中文环境 系统管理 桌面应用 内核技术 | Linux基础: 基础入门 安装配置 常用命令 经验技巧 软件应用 | Linux数据库: Mysql Postgre Oracle DB2 Sybase other
网络管理: 网络安全 网络应用 Linux服务器 环境配置 黑客安全 | 编程开发: PHP CC++ Python Perl Shell 嵌入式开发 java jsp | PHP技术: PHP基础 PHP技巧 PHP应用 PHP文摘
搜索中心 Linux招聘 Linux专题 Apache | Linux相关: 硬件相关 Linux解决方案 Linux认证 企业应用 其它Unix | 相关下载: 资料下载 参考手册 开发工具 服务器类 软路由 其它
 技术搜索:
会员中心 注册会员 高级搜索  
  → 当前位置:首页>编程开发>其他编程>正文

在Linux上获得线程id的方法

http://www.oklinux.cn  2008-08-13  linuxidc   会员收藏  游客收藏  【 】 
您查看的文章来源于http://www.oklinux.cn

在Linux上获得线程id的方法我使用了第二种方法,很方便:

#define gettid() syscall(__NR_gettid)

用到的地方 gettid()

在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
.text
ENTRY (syscall)
PUSHARGS_6 /* Save register contents. */
_DOARGS_6(44) /* Load arguments. */
movl 20(%esp), 陎 /* Load syscall number into 陎. */
ENTER_KERNEL /* Do the system call. */
POPARGS_6 /* Restore register contents. */
cmpl $-4095, 陎 /* Check 陎 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:�,%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中维护了,但是没有提供访问方法的数据。


上一篇:Linux下C连接MySQL找不到MySQL.H的问题   下一篇:终端中上irc的方法──irssi

收藏于收藏夹】 【评论】 【推荐】 【打印】 【关闭
相关文档
·终端中上irc的方法──irssi
·交叉编译directfb在arm Linux平台上
·一些不安全的常用的C函数
·Linux系统在多个VI终端间实现文件内容拷贝
·Linux下内存占用多的原因
·Lucene Demo安装,Linux
·Linux自动加载驱动模块
·Linux条件变量用法
·Linux下/dev/mem和/dev/kmem的区别
·cramfs文件系统和嵌入式Linux
·Linux下的string与Windows下的区别
·Linux下多语言开发
·在Linux2.6内核下实现进程隐藏
·在Linux下用g 编译SDL程序的方法
·Linux驱动与RTC驱动泛泛研究
·嵌入式:关于Linux开发环境的建立
发表评论
密码: 匿名评论
评论内容:

(不超过250字,需审核后才会公布,请自觉遵守互联网相关政策法规)
 
  最新文档
·Linux源代码已超过1000万行 价值达5亿
·Linux下安装PCI转串口卡及USB转串口器
·基于3c2410的Linux2.6.22移植
·Java基础:Java初学者必须了解的基础问
·Linux下while循环中数组赋值的方法
·C 异步编程方式
·Linux驱动开发之input子系统
·Linux下paste命令
·Ubuntu Linux下C编译错误解决实例
·Ubuntu下向系统日志写记录
·Linux内核模块编程helloworld小程序问
·Ubuntu下QT的开发环境
  阅读排行
·Linux编程时获取当前时间实例解析
·Linux中断处理学习笔记
·Linux环境下OpenGL编程学习
·Linux下安装g77 fortran complier过程
·Linux添加路径到PATH
·Linux多线程编程学习之线程同步
·Linux内核补丁与补丁的安装
·Linux - 非阻塞socket编程处理EAGAIN错
·Linux下Qtopia Core 4.3(QT/E)交叉编译
·Linux平台下学写批处理
·初识Linux脚本编程(shell)
·Linux文件系统ext4 相关知识大全
·新手学堂 Linux的sort命令使用
·Linux环境下重新编译GCC-4.3.0
·解决Linux中Matlab中文乱码问题
网摘收藏: