今天咋给线程起个名字,哈哈,这个在gdb中调试的时候,将会非常有用。例如用i threads命令查看所有的线程的的时候,或者用thread apply all bt命令打印所有线程的调用栈时,能知道哪些信息对应哪个线程,还是很有必要的。

给线程起名字,两种方法,分别用prctl和pthread_setname_np。
详细使用方法参见:
https://blog.csdn.net/zhizhengguan/article/details/112062624

prctl在线程函数中使用,因为没有参数指定pthread_t类型的变量。
pthread_setname_np需要比较高的glibc版本,很可惜我的Ubuntu和板子的版本都不高,所以使用不了,只能使用最原始的prctl。

示例代码:

#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <pthread.h>
#include <errno.h>
#include <sys/prctl.h>

void *thread_1(void *args)
{
    prctl(PR_SET_NAME, (unsigned long)"thread_1");
    while ( 1 )
    {
        printf("thread 1\n");
        sleep(1);
    }
}

void *thread_2(void *args)
{
    prctl(PR_SET_NAME, (unsigned long)"thread_2");
    while ( 1 )
    {
        printf("thread 2\n");
        usleep(800*1000);
    }
}

int main(int argc, char *argv[])
{
    pthread_t t1;
    pthread_t t2;
    pthread_create(&t1, NULL, thread_1, NULL);
    pthread_create(&t2, NULL, thread_2, NULL);

#ifdef  _GNU_SOURCE
    if ( pthread_setname_np(t1, "thread_1") !=0 || pthread_setname_np(t2, "thread_2") != 0 )
    {
        printf("err: set thread name failed , errno is %d\n", errno);
    }
#else
    printf("pthread_setname_np is not supported\n");
#endif
    pthread_join(t1, NULL);
    return 0;
}

gdb中实际输出线程信息示例:

(gdb) i threads
  Id   Target Id           Frame
* 1    LWP 5978 "a.out"    0x76fc0620 in pthread_join () from /lib/libpthread.so.0
  2    LWP 5984 "thread_1" 0x76f1d060 in nanosleep () from /lib/libc.so.6
  3    LWP 5985 "thread_2" 0x76f1d060 in nanosleep () from /lib/libc.so.6
(gdb) q
A debugging session is active.
Logo

更多推荐