• 从源码视角彻底搞懂Linux线程实现原理


    1.Linux线程接口

    • pthread_create : 创建线程
    • pthread_join : 等待线程执行完毕,获取线程执行结果,释放线程资源(线程栈内存等)
    • pthread_detach: 设置线程为分离状态,线程处于unjoinable状态,如果调用pthread_join返回错误;线程执行完比自动释放资源。
    • pthread_exit : 退出当前线程。

    2.pthread_create

    可以参照如下文章

    从线程的一生看系统调用实现原理(bionic c库为例)_nginux的博客-CSDN博客

    要点:

    1. 如果不特定指定PTHREAD_ATTR_FLAG_DETACHED,线程创建完状态时:THREAD_NOT_JOINED;如果指定上面FLAG,创建完状态是:THREAD_DETACHED。

    2.线程执行完成的时候,默认都会调用pthread_exit(即使程序中不显示调用pthread_exit)。

    源码分析:

    1. __BIONIC_WEAK_FOR_NATIVE_BRIDGE
    2. int pthread_create(pthread_t* thread_out, pthread_attr_t const* attr,
    3. void* (*start_routine)(void*), void* arg) {
    4. ErrnoRestorer errno_restorer;
    5. pthread_attr_t thread_attr;
    6. if (attr == NULL) {
    7. pthread_attr_init(&thread_attr);
    8. } else {
    9. thread_attr = *attr;
    10. attr = NULL; // Prevent misuse below.
    11. }
    12. pthread_internal_t* thread = NULL;
    13. void* child_stack = NULL;
    14. //分配线程栈内存和创建代表线程的数据结构
    15. int result = __allocate_thread(&thread_attr, &thread, &child_stack);
    16. if (result != 0) {
    17. return result;
    18. }
    19. // Create a lock for the thread to wait on once it starts so we can keep
    20. // it from doing anything until after we notify the debugger about it
    21. //
    22. // This also provides the memory barrier we need to ensure that all
    23. // memory accesses previously performed by this thread are visible to
    24. // the new thread.
    25. thread->startup_handshake_lock.init(false);
    26. thread->startup_handshake_lock.lock();
    27. //用户传入的线程入口函数
    28. thread->start_routine = start_routine;
    29. thread->start_routine_arg = arg;
    30. thread->set_cached_pid(getpid());
    31. int flags = CLONE_VM | CLONE_FS | CLONE_FILES | CLONE_SIGHAND | CLONE_THREAD | CLONE_SYSVSEM |
    32. CLONE_SETTLS | CLONE_PARENT_SETTID | CLONE_CHILD_CLEARTID;
    33. void* tls = reinterpret_cast(thread->tls);
    34. #if defined(__i386__)
    35. // On x86 (but not x86-64), CLONE_SETTLS takes a pointer to a struct user_desc rather than
    36. // a pointer to the TLS itself.
    37. user_desc tls_descriptor;
    38. __init_user_desc(&tls_descriptor, false, tls);
    39. tls = &tls_descriptor;
    40. #endif
    41. //内部通过linux clone系统调用实现
    42. int rc = clone(__pthread_start, child_stack, flags, thread, &(thread->tid), tls, &(thread->tid));
    43. if (rc == -1) {
    44. int clone_errno = errno;
    45. // We don't have to unlock the mutex at all because clone(2) failed so there's no child waiting to
    46. // be unblocked, but we're about to unmap the memory the mutex is stored in, so this serves as a
    47. // reminder that you can't rewrite this function to use a ScopedPthreadMutexLocker.
    48. thread->startup_handshake_lock.unlock();
    49. if (thread->mmap_size != 0) {
    50. munmap(thread->attr.stack_base, thread->mmap_size);
    51. }
    52. async_safe_format_log(ANDROID_LOG_WARN, "libc", "pthread_create failed: clone failed: %s",
    53. strerror(clone_errno));
    54. return clone_errno;
    55. }
    56. int init_errno = __init_thread(thread);
    57. if (init_errno != 0) {
    58. // Mark the thread detached and replace its start_routine with a no-op.
    59. // Letting the thread run is the easiest way to clean up its resources.
    60. atomic_store(&thread->join_state, THREAD_DETACHED);
    61. __pthread_internal_add(thread);
    62. thread->start_routine = __do_nothing;
    63. thread->startup_handshake_lock.unlock();
    64. return init_errno;
    65. }
    66. // Publish the pthread_t and unlock the mutex to let the new thread start running.
    67. *thread_out = __pthread_internal_add(thread);
    68. thread->startup_handshake_lock.unlock();
    69. return 0;
    70. }
    71. // Called from the __bionic_clone assembler to call the thread function then exit.
    72. extern "C" __LIBC_HIDDEN__ void __start_thread(int (*fn)(void*), void* arg) {
    73. BIONIC_STOP_UNWIND;
    74. pthread_internal_t* self = __get_thread();
    75. if (self && self->tid == -1) {
    76. self->tid = syscall(__NR_gettid);
    77. }
    78. //相当于执行pthread_start函数
    79. int status = (*fn)(arg);
    80. __exit(status);
    81. }
    82. static int __pthread_start(void* arg) {
    83. pthread_internal_t* thread = reinterpret_cast<pthread_internal_t*>(arg);
    84. // Wait for our creating thread to release us. This lets it have time to
    85. // notify gdb about this thread before we start doing anything.
    86. // This also provides the memory barrier needed to ensure that all memory
    87. // accesses previously made by the creating thread are visible to us.
    88. thread->startup_handshake_lock.lock();
    89. __init_alternate_signal_stack(thread);
    90. //调用到用户调用pthread_create传入的线程入口函数
    91. void* result = thread->start_routine(thread->start_routine_arg);
    92. //线程执行完都会调用pthread_exit
    93. pthread_exit(result);
    94. return 0;
    95. }

     内核执行完clone系统调用,创建完线程之后会跳转到__start_thread执行,最终会调用到用户pthread_create传入的线程执行函数体中。

    3.pthread_join

    man pthread_join如下:

    1. PTHREAD_JOIN(3) Linux Programmer's Manual PTHREAD_JOIN(3)
    2. NAME
    3. pthread_join - join with a terminated thread
    4. SYNOPSIS
    5. #include
    6. int pthread_join(pthread_t thread, void **retval);
    7. Compile and link with -pthread.
    8. DESCRIPTION
    9. The pthread_join() function waits for the thread specified by thread to terminate. If that thread has already terminated, then pthread_join()
    10. returns immediately. The thread specified by thread must be joinable.
    11. If retval is not NULL, then pthread_join() copies the exit status of the target thread (i.e., the value that the target thread supplied to
    12. pthread_exit(3)) into the location pointed to by *retval. If the target thread was canceled, then PTHREAD_CANCELED is placed in *retval.
    13. If multiple threads simultaneously try to join with the same thread, the results are undefined. If the thread calling pthread_join() is can‐
    14. celed, then the target thread will remain joinable (i.e., it will not be detached).

    要点:

    1. 等待指定的线程结束,且等待的线程必须是joinable(不然报错)。

    2. 多个线程pthread_join一个线程结果未知。

    3. pthread_join会释放等待线程的资源。

    4.pthread_join执行成功,线程状态:THREAD_JOINED。

    使用场景:

    线程A需要请线程B帮忙干活,且要等待线程B的结果才继续执行,属于串行执行,A负责释放线程B的资源。

    pthread_join的实现源码分析:

    1. __BIONIC_WEAK_FOR_NATIVE_BRIDGE
    2. int pthread_join(pthread_t t, void** return_value) {
    3. if (t == pthread_self()) {
    4. return EDEADLK;
    5. }
    6. pthread_internal_t* thread = __pthread_internal_find(t);
    7. if (thread == NULL) {
    8. return ESRCH;
    9. }
    10. //将线程状态修改为:THREAD_JOINED
    11. ThreadJoinState old_state = THREAD_NOT_JOINED;
    12. while ((old_state == THREAD_NOT_JOINED || old_state == THREAD_EXITED_NOT_JOINED) &&
    13. !atomic_compare_exchange_weak(&thread->join_state, &old_state, THREAD_JOINED)) {
    14. }
    15. //如果等待的线程是detached(调用过pthread_detach)或者已经被pthread_join过,返回错误。
    16. if (old_state == THREAD_DETACHED || old_state == THREAD_JOINED) {
    17. return EINVAL;
    18. }
    19. pid_t tid = thread->tid;
    20. volatile int* tid_ptr = &thread->tid;
    21. // We set thread->join_state to THREAD_JOINED with atomic operation,
    22. // so no one is going to remove this thread except us.
    23. // Wait for the thread to actually exit, if it hasn't already.
    24. //等待线程结束
    25. while (*tid_ptr != 0) {
    26. __futex_wait(tid_ptr, tid, NULL);
    27. }
    28. //拿到等待线程的返回值
    29. if (return_value) {
    30. *return_value = thread->return_value;
    31. }
    32. //释放等待线程的资源,删除线程的数据结构和unmap栈内存
    33. __pthread_internal_remove_and_free(thread);
    34. return 0;
    35. }

    4.pthread_detach

    man pthread_detach:

    1. NAME
    2. pthread_detach - detach a thread
    3. SYNOPSIS
    4. #include <pthread.h>
    5. int pthread_detach(pthread_t thread);
    6. Compile and link with -pthread.
    7. DESCRIPTION
    8. The pthread_detach() function marks the thread identified by thread as detached. When a detached thread terminates, its resources are automati‐
    9. cally released back to the system without the need for another thread to join with the terminated thread.
    10. Attempting to detach an already detached thread results in unspecified behavior.
    11. RETURN VALUE
    12. On success, pthread_detach() returns 0; on error, it returns an error number.

    使用场景:

    与pthread_join使用的场景相反,如果线程A请线程B帮忙做事,线程A和线程B可以并行执行,不需要等待线程B的结果反馈,此时由于可能B线程完毕了,那么B要负责自己的资源释放(比如栈内存释放)

    要点:

    1.detach的线程自己负责释放资源(数据结构和栈内存)

    2.重复detach同一个线程是个未定义行为。

    3.pthread_detach执行成功,线程状态为:THREAD_DETACHED

    源码分析:

    1. __BIONIC_WEAK_FOR_NATIVE_BRIDGE
    2. int pthread_detach(pthread_t t) {
    3. pthread_internal_t* thread = __pthread_internal_find(t);
    4. if (thread == NULL) {
    5. return ESRCH;
    6. }
    7. //线程状态修改为:THREAD_DETACHED
    8. ThreadJoinState old_state = THREAD_NOT_JOINED;
    9. while (old_state == THREAD_NOT_JOINED &&
    10. !atomic_compare_exchange_weak(&thread->join_state, &old_state, THREAD_DETACHED)) {
    11. }
    12. if (old_state == THREAD_NOT_JOINED) {
    13. return 0;
    14. } else if (old_state == THREAD_EXITED_NOT_JOINED) {
    15. // Use pthread_join to clean it up.
    16. return pthread_join(t, NULL);
    17. }
    18. return EINVAL;
    19. }

    上面代码看完是否产生一个疑问,man手册中说pthread_detach的线程要自己释放资源,而上面源码中并未看到释放资源的代码,线程执行完比之后是怎么实现自己释放资源的?答案是在pthread_exit接口中。 

    5.pthread_exit

    通过pthread_create小节的分析,每个线程创建完成后,会先执行__start_thread,该函数进而执行pthread_start函数,这里执行线程函数主题,线程退出的时候调用了pthread_exit,也就说每个线程退出的时候都会调用下pthread_exit,而这里正式detached线程释放自己释放资源的地方:

    1. extern "C" __noreturn void __exit(int);
    2. 1,1 Top
    3. memset(&ss, 0, sizeof(ss));
    4. ss.ss_flags = SS_DISABLE;
    5. sigaltstack(&ss, NULL);
    6. // Free it.
    7. munmap(thread->alternate_signal_stack, SIGNAL_STACK_SIZE);
    8. thread->alternate_signal_stack = NULL;
    9. }
    10. ThreadJoinState old_state = THREAD_NOT_JOINED;
    11. while (old_state == THREAD_NOT_JOINED &&
    12. !atomic_compare_exchange_weak(&thread->join_state, &old_state, THREAD_EXITED_NOT_JOINED)) {
    13. }
    14. //如果线程是DETACHED状态,释放自己的资源(栈内存和数据结构)
    15. if (old_state == THREAD_DETACHED) {
    16. // The thread is detached, no one will use pthread_internal_t after pthread_exit.
    17. // So we can free mapped space, which includes pthread_internal_t and thread stack.
    18. // First make sure that the kernel does not try to clear the tid field
    19. // because we'll have freed the memory before the thread actually exits.
    20. __set_tid_address(NULL);
    21. // pthread_internal_t is freed below with stack, not here.
    22. __pthread_internal_remove(thread);
    23. if (thread->mmap_size != 0) {
    24. // We need to free mapped space for detached threads when they exit.
    25. // That's not something we can do in C.
    26. // We don't want to take a signal after we've unmapped the stack.
    27. // That's one last thing we can do before dropping to assembler.
    28. ScopedSignalBlocker ssb;
    29. __pthread_unmap_tls(thread);
    30. _exit_with_stack_teardown(thread->attr.stack_base, thread->mmap_size);
    31. }
    32. }
    33. // No need to free mapped space. Either there was no space mapped, or it is left for
    34. // the pthread_join caller to clean up.
    35. __pthread_unmap_tls(thread);
    36. __exit(0);
    37. }

  • 相关阅读:
    安卓截屏;前台服务
    Centos7安装Docker
    【LeetCode热题100】--240.搜索二维矩阵II
    什么样的触达方式,会员会喜欢?
    jvm 内存模型介绍
    【前端学习】—JS判断数据类型的方式有哪些(八)
    四十二、路由层
    从TF-IDF 到BM25, BM25+,一文彻底理解文本相关度
    Java输入/输出之RandomAccessFile的功能和用法
    JavaSE基础之(十九)Java内部类
  • 原文地址:https://blog.csdn.net/GetNextWindow/article/details/126923821