Linux應用程序設計:如何獲取線程棧的使用信息?
面對的問題
對于線程的棧空間,相信各位小伙伴都不陌生。它有下面的這幾項特性:
由操作系統(tǒng)分配固定的空間;
使用一個棧寄存器來保存實時位置;
后進先出。
今天,我們不聊操作系統(tǒng)層面對棧的管理,只從應用程序的角度,來看一下如何實時獲取棧的使用情況。
在一般的單片機/嵌入式程序開發(fā)過程中,在創(chuàng)建一個線程(或者稱作任務)的時候,是可以指定給該線程分配多少?臻g的。
然后在調(diào)試的時候呢,周期性的打印出棧區(qū)的使用情況:消耗了多少空間,還剩余多少空間。
這樣的話,跑完每一個測試用例之后,就能得到一個大致的統(tǒng)計數(shù)據(jù),從而最終決定:需要給這個線程分配多少?臻g。
例如:在 ucOS 系統(tǒng)中,提供了函數(shù) NT8U OSTaskStkChk(INT8U prio, OS_STK_DATA *p_stk_data),來獲取一個任務的棧使用信息。
但是在 Linux 系統(tǒng)中,并沒有這樣類似的函數(shù),來直接獲取棧使用信息。
因此,為了得到此線程的已使用和空閑?臻g,必須通過其他的方式來獲取。
下面,就提供 2 種解決方案:正規(guī)軍方式和雜牌軍方式!
正規(guī)軍方式
在 Linux 系統(tǒng)中,在創(chuàng)建一個線程的時候,是可以通過線程屬性來設置:為這個線程分配多少的棧(stack)空間的。
如果應用程序不指定的話,操作系統(tǒng)就設置為一個默認的值。
線程創(chuàng)建完畢之后,操作系統(tǒng)在內(nèi)核空間,記錄了這個線程的一切信息,當然也就包括給它分配的棧空間信息。
為了讓應用層能夠獲取到這個信息,操作系統(tǒng)也提供了相應的系統(tǒng)函數(shù)。代碼如下:
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 ", stack_addr);
printf("stack bottom = %p ", stack_addr + stack_size);
從上面這段代碼中可以看到,它只能獲取?臻g的地址開始以及總的空間大小,仍然不知道當前?臻g的實際使用情況!
我找了一下相關的系統(tǒng)調(diào)用,Linux 似乎沒有提供相關的函數(shù)。
怎么辦?只能迂回操作。
我們知道,在 Linux x86 平臺上,寄存器 ESP 就是來存儲棧指針的。對于一個滿遞減類型的棧,這個寄存器里的值,就代表了當前棧中最后背使用的、那個棧空間的地址。
因此,只要我們能夠獲取到 ESP 寄存器里的值,就相當于知道了當前這個棧有多少空間被使用了。
那么怎樣來獲取 ESP 寄存器的值呢?既然是寄存器,那就肯定是使用匯編代碼了。
很簡單,就 1 行:
size_t esp_val;
asm("movl %%esp, %0" : "=m"(esp_val) :);
對不起,我錯了!應該是 2 行代碼,忘記變量定義了。
對于匯編代碼不熟悉的小伙伴,可以參考之前總結的一篇文章:內(nèi)聯(lián)匯編很可怕嗎?看完這篇文章,終結它!
找到第 4 個示例,直接抄過來就行。
好了,拿到了以上的所有信息,就可以計算出棧的已使用和空閑空間的大小了:

請輸入評論內(nèi)容...
請輸入評論/評論長度6~500個字
最新活動更多
推薦專題
- 1 UALink規(guī)范發(fā)布:挑戰(zhàn)英偉達AI統(tǒng)治的開始
- 2 北電數(shù)智主辦酒仙橋論壇,探索AI產(chǎn)業(yè)發(fā)展新路徑
- 3 降薪、加班、裁員三重暴擊,“AI四小龍”已折戟兩家
- 4 “AI寒武紀”爆發(fā)至今,五類新物種登上歷史舞臺
- 5 國產(chǎn)智駕迎戰(zhàn)特斯拉FSD,AI含量差幾何?
- 6 光計算迎來商業(yè)化突破,但落地仍需時間
- 7 東陽光:2024年扭虧、一季度凈利大增,液冷疊加具身智能打開成長空間
- 8 地平線自動駕駛方案解讀
- 9 封殺AI“照騙”,“淘寶們”終于不忍了?
- 10 優(yōu)必選:營收大增主靠小件,虧損繼續(xù)又逢關稅,能否乘機器人東風翻身?