Linux应用程序设计:用一种讨巧方式,来获取线程栈的使用信息

在 Linux系统中,在创建一个线程的时候,是可以通过线程属性来设置:为这个线程分配多少的栈(stack)空间的。

如果应用程序不指定的话,操作系统就设置为一个默认的值。

线程创建完毕之后,操作系统在内核空间,记录了这个线程的一切信息,当然也就包括给它分配的栈空间信息。

为了让应用层能够获取到这个信息,操作系统也提供了相应的系统函数。代码如下:

pthread_attr_t attr;
void *stack_addr;
int stack_size;

memset(&attr, 0, sizeof(pthread_attr_t));
pthread_getattr_np(pthread_self(), &attr);
pthread_attr_getstack(&attr, &stack_addr, &stack_size);
pthread_attr_destroy(&attr);

printf(“statck top = %p \n”, stack_addr);
printf(“stack bottom = %p \n”, stack_addr + stack_size);
从上面这段代码中可以看到,它只能获取栈空间的地址开始以及总的空间大小,仍然不知道当前栈空间的实际使用情况!

我找了一下相关的系统调用,Linux 似乎没有提供相关的函数。

怎么办?只能迂回操作。

我们知道,在 Linux x86 平台上,寄存器 ESP 就是来存储栈指针的。对于一个满递减类型的栈,这个寄存器里的值,就代表了当前栈中最后背使用的、那个栈空间的地址。

因此,只要我们能够获取到 ESP 寄存器里的值,就相当于知道了当前这个栈有多少空间被使用了。

那么怎样来获取 ESP 寄存器的值呢? 既然是寄存器,那就肯定是使用汇编代码了。

很简单,就 1 行:

size_t esp_val;
asm(“movl %%esp, %0” : “=m”(esp_val) 😃;
对不起,我错了!应该是 2 行代码,忘记变量定义了。

对于汇编代码不熟悉的小伙伴,可以参考之前总结的一篇文章:内联汇编很可怕吗?看完这篇文章,终结它!

找到第 4 个示例,直接抄过来就行。

好了,拿到了以上的所有信息,就可以计算出栈的已使用和空闲空间的大小了:

把以上代码放在一起:

#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <pthread.h>
#include

void print_stack1()
{
size_t used, avail;
pthread_attr_t attr;
void *stack_addr;
int stack_size;

// 获取栈寄存器 ESP 的当前值
size_t esp_val;
asm("movl %%esp, %0" : "=m"(esp_val) :);

// 通过线程属性,获取栈区的起始地址和空间总大小
memset(&amp;attr, 0, sizeof(pthread_attr_t));
pthread_getattr_np(pthread_self(), &amp;attr);
pthread_attr_getstack(&amp;attr, &amp;stack_addr, &amp;stack_size);
pthread_attr_destroy(&amp;attr);

printf("espVal = %p \n", esp_val);
printf("statck top   = %p \n", stack_addr);
printf("stack bottom = %p \n", stack_addr + stack_size);

avail = esp_val - (size_t)stack_addr;
used = stack_size - avail;

printf("print_stack1: used = %d, avail = %d, total = %d \n", 
        used, avail, stack_size);

}

int main(int argc, char *agv[])
{
print_stack1();
return 0;
}

杂牌军方式

上面的正规军方法,主要是通过系统函数获取了线程的属性信息,从而获取了栈区的开始地址和栈的总空间大小。

为了获取这两个值,调用了 3 个函数,有点笨重!

不知各位小伙伴是否想起:Linux 操作系统会为一个应用程序,都提供了一些关于 limit 的信息,这其中就包括堆栈的相关信息。

这样的话,我们就能拿到一个线程的栈空间总大小了。

此时,还剩下最后一个变量不知道:栈区的开始地址!

我们来分析一下哈:当一个线程刚刚开始执行的时候,栈区里可以认为是空的,也就是说此时 ESP 寄存器里的值就可以认为是指向栈区的开始地址!

是不是有豁然开朗的感觉?!

但是,这仍然需要调用汇编代码来获取。

再想一步,既然此时栈区里可以认为是空的,那么如果在线程的第一个函数中,定义一个局部变量,然后通过获取这个局部变量的地址,不就相当于是获取到了栈区的开始地址了吗?

如下图所示:

我们可以把这个局部变量的地址,记录在一个全局变量中。然后在应用程序的其他代码处,就可以用它来代表栈的起始地址。

知道了 3 个必需的变量,就可以计算栈空间的使用情况了:

// 用来存储栈区的起始地址
size_t top_stack;

void print_stack2()
{
size_t used, avail;

size_t esp_val;
asm("movl %%esp, %0" : "=m"(esp_val) :);
printf("esp_val = %p \n", esp_val);

used = top_stack - esp_val;

struct rlimit limit;
getrlimit(RLIMIT_STACK, &amp;limit);
avail = limit.rlim_cur - used;
printf("print_stack2: used = %d, avail = %d, total = %d \n", 
        used, avail, used + avail);

}

int main(int argc, char *agv[])
{
int x = 0;
// 记录栈区的起始地址(近似值)
top_stack = (size_t)&x;
print_stack2();
return 0;
}

更讨巧的方式

在上面的两种方法中,获取栈的当前指针位置的方式,都是通过汇编代码,来获取寄存器 ESP 中的值。

是否可以继续利用刚才的技巧:通过定义一个局部变量的方式,来间接地获取 ESP 寄存器的值?

void print_stack3()
{
int x = 0;
size_t used, avail;
// 局部变量的地址,可以近似认为是 ESP 寄存器的值
size_t tmp = (size_t)&x;
used = top_stack - tmp;

struct rlimit limit;
getrlimit(RLIMIT_STACK, &amp;limit);
avail = limit.rlim_cur - used;
printf("print_stack3: used = %d, avail = %d, total = %d \n", 
        used, avail, used + avail);

}

int main(int argc, char *agv[])
{
int x = 0;
top_stack = (size_t)&x;
print_stack3();
return 0;
}

总结

以上的几种方式,各有优缺点。

我们把以上 3 个打印堆栈使用情况的函数放在一起,然后在 main 函数中,按顺序调用 3 个测试函数,每个函数中都定义一个整型数组(消耗 4K 的栈空间),然后看一下这几种方式的打印输出信息:

// 测试代码(3个打印函数就不贴出来了)
void print_stack1()
{

}

void print_stack2()
{

}

void print_stack3()
{

}

void func3()
{
int num[1024];
print_stack1();
printf("\n\n ********* \n");
print_stack2();
printf("\n\n ********* \n");
print_stack3();
}

void func2()
{
int num[1024];
func3();
}

void func1()
{
int num[1024];
func2();
}

int main(int argc, char *agv[])
{
int x = 0;
top_stack = (size_t)&x;
func1();
return 0;
}
打印输出信息:

espVal = 0xffe8c980
statck top = 0xff693000
stack bottom = 0xffe90000
print_stack1: used = 13952, avail = 8362368, total = 8376320


esp_val = 0xffe8c9a0
print_stack2: used = 12456, avail = 8376152, total = 8388608


print_stack3: used = 12452, avail = 8376156, total = 8388608
USB Microphone https://www.soft-voice.com/
Wooden Speakers https://www.zeshuiplatform.com/
亚马逊测评 www.yisuping.cn
深圳网站建设www.sz886.com

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值