测量 Linux 多线程应用程序的堆栈使用情况

mta*_*abz 8 linux stack multithreading pthreads

我正在为 Linux 嵌入式平台开发一个多线程应用程序。

目前,我将每个线程的堆栈大小(通过pthread_set_attr)设置为相当大的默认值。我想将每个线程的值微调为较小的值,以减少应用程序的内存使用量。我可以尝试将每个线程的堆栈大小设置为逐渐减小的值,直到程序崩溃,但应用程序使用约 15 个线程,每个线程具有完全不同的功能/属性,因此这种方法将非常耗时。

我更愿意能够直接测量每个线程的堆栈使用情况。人们可以推荐一些实用程序来执行此操作吗?(例如,我来自 vxWorks 背景,并使用 vxWorks shell 中的“ti”命令直接提供有关堆栈使用情况的统计信息以及有关任务状态的其他有用信息。)

d99*_*ris 8

以下是测量 Linux 应用程序中(本机 pthreads)堆栈使用情况的两个工具:

瓦尔格林德

用法:

valgrind --tool=drd --show-stack-usage=yes PROG
Run Code Online (Sandbox Code Playgroud)

Valgrind 是一个稳定而强大的工具,不仅可用于测量堆栈使用情况。但它可能不支持所有嵌入式 CPU 型号。

堆栈使用

用法:

stackusage PROG
Run Code Online (Sandbox Code Playgroud)

Stackusage 是一个轻量级工具,专门用于测量线程堆栈使用情况,对于大多数配备 glibc 的嵌入式 Linux 平台来说应该是可移植的。目前它可能不像 Valgrind/drd 那样经过充分测试或成熟。

全面披露:我是 Stackusage 的作者。


Tob*_*obi 4

我不知道有什么好的工具,但作为最后的手段,您可以在应用程序中包含一些代码来检查它,类似于以下内容:

__thread void* stack_start;
__thread long stack_max_size = 0L;

void check_stack_size() {
  // address of 'nowhere' approximates end of stack
  char nowhere;
  void* stack_end = (void*)&nowhere;
  // may want to double check stack grows downward on your platform
  long stack_size = (long)stack_start - (long)stack_end;
  // update max_stack_size for this thread
  if (stack_size > stack_max_size)
    stack_max_size = stack_size;
}
Run Code Online (Sandbox Code Playgroud)

必须在某些嵌套最深的函数中调用 check_stack_size() 函数。

然后作为线程中的最后一条语句,您可以将 stack_max_size 输出到某个地方。

stack_start 变量必须在线程启动时初始化:

void thread_proc() {
  char nowhere;
  stack_start = (void*)&nowhere;
  // do stuff including calls to check_stack_size()
  // in deeply nested functions
  // output stack_max_size here
}
Run Code Online (Sandbox Code Playgroud)