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的用途,还需要弄清楚。