pthread线程创建过程(未完)

本文详细解析了pthread_create线程创建的过程,从调用__pthread_create_2_1开始,涉及内存分配、stack的mmap、create_thread与sys_clone的交互,再到glibc的clone.S和内核的do_fork。讨论了关键标志如CLONE_VM、CLONE_FILES和CLONE_SETTLS的作用,以及它们如何影响线程的页表、文件描述符和TLS区域。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1.  当我们使用pthread_create来创建线程的时候, 实际上调用的是__pthread_create_2_1
     versioned_symbol (libpthread, __pthread_create_2_1, pthread_create, GLIBC_2_1); 
2.  而__pthread_create_2_1里面呢,
      根据传入的pthread_attr_t 来分配stack,这个stack可以被thread使用
      其调用函数allocate_stack来分配, 这个函数又调用mmap来得到内存
      mem = mmap (NULL, size, prot, MAP_PRIVATE | MAP_ANONYMOUS | MAP_STACK, -1, 0);

     然后调用函数create_thread() 函数来创建线程
3.  而create_thread实际上调用sys_clone来创建,
     而clone的flag呢, 在这里指定。
     clone_flags = (CLONE_VM | CLONE_FS | CLONE_FILES | CLONE_SIGNAL
                              | CLONE_SETTLS | CLONE_PARENT_SETTID
                              | CLONE_CHILD_CLEARTID | CLONE_SYSVSE
     
     调用do_clone的参数为
      int res = do_clone (pd, attr, clone_flags, start_thread, STACK_VARIABLES_ARGS, 1);
      而STACK_VARIABLES_PARMS 被定义为void *stackaddr, size_t stacksize
       注意: 这里start_thread实际上是线程返回后执行的函数。其定义在nptl/pthread_create.c中
4.   do_clone里面则是调用glibc-2.14/sysdeps/unix/sysv/linux/i386/clone.S 来创建线程
5. clone.s 又通过系统调用sys_clone来创建内核线程
6. sys_clone呢,
    调用do_fork来创建线程
7. sys_clone 和 kernel_thread的函数区别呢, 在于
    sys_clone调用的时候, 从用户态传入栈的起始地址和栈的大小,把这两个参数传给了do_fork
    而kernel_thread函数, 调用do_fork的时候, 这两个参数都是0
     同时呢, sys_clone还传入了地址用换存放parent_tid 和child_tid

8.  现在再关注下 clone的时候,传入的clone_flag
     1. 这里CLONE_VM非常重要, 在kernel_thread 这个flag也是指定的, 而kernel_thread所有的线程都是使用同一个页表。
         而这里, 意味着所有的线程也是使用同一个页表。
     2. 而CLONE_FILES, 则只是增加了parent task_struck->files_struct的ref count。 对比进程的fork, 则是把file_struct 复制了一份。
     3. CLONE_SETTLS 则为线程创建新的TLS

TLS的用途,还需要弄清楚。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值