日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

请使劲回答一个关于UNIX/Linux自动扩展stack的问

發布時間:2023/12/31 综合教程 27 生活家
生活随笔 收集整理的這篇文章主要介紹了 请使劲回答一个关于UNIX/Linux自动扩展stack的问 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

系統運維

有本事就出來,沒本事就當鱉!

如果讓我回答關于進程棧,線程棧的問題,只要問題不籠統,只要問題明確,我會一五一十地回答,正確率上九成,然而,可悲的是,問題往往他媽的都不是
很明確,因此,游戲到此結束??!艸。但是如果給我一個問的機會,我會問下面一個問題,記住,使出你拉屎的勁來回答(該問題足夠糙!不必太當回事,重要的東
西在下面-):

UNIX/Linux
的stack在大多數平臺是向下擴展的(注意,我已經告訴他事實了,我并沒有問...是如何擴展的,這是可以背誦下來并朗讀出來的),在一個執行流調用了
一個函數A,而該函數A在stack上分配了一個大數組導致了stack擴展(注意,這又是一段陳述,我還沒有給出問題),然后A返回
了,UNIX/Linux理應回收調A里面大數組分配stack空間-因為它再也沒有用了,但是它并沒有這么做(這里可能是一個陷阱,真的是UNIX
/Linux理應這么做卻沒有做,還是說我只是在逗你玩...不確定,但陳述就是如此)。(注意,我的問題來了),請問,UNIX/Linux為什么這么
做???!??!

凡是回答操作系統這么規定的之類的答案,一律零分!況且,你能證明我說的一定對嗎?萬一我是逗你玩呢?操作系統能規定一個錯的東西嗎?或者我可以繼續問為
什么這么規定,直到像我初中的歷史老師被我問到朝我眼角猛打一拳那樣,如果我能找回點賤人所謂的快感,那么來打吧!問題就是這樣,不管這個問題是一個偽命
題還是你有自己的想法,能說5分鐘的,我覺得也夠可以了。該題目的答題要求如下:
時間限制:5分鐘。
作答方式:全口述,不能畫圖,不能打手勢...語言含糊不清的,表達能力不好的,算錯誤。
答題建議:如果你對OS虛擬內存管理以及Linux的VMA實現細節沒有相當深入的理解,請不要猜測答案。請直接回答“不知道”,然后看完此文。
.................................
5分鐘過去。我要公布一點我的想法了。
首先,這個問題在本身看來,有問題。因為雖然Linux理應這么做,但它:
第一,它不一定能做到;
第二,它根本沒有必要做。
那么論據是什么?憑什么這樣說?

積極論點

沒必要這樣做。執行流還會調用別的函數或者再次調用A,頻繁回收棧損耗性能;

消極論點

很難或者不能做到。stack操作是處理器控制的,和OS內核地址空間管理機制之間沒有同步機制,一個函數調用結束后,CPU自動處理stack寄存器的收縮,彈出棧幀,然而它無法通知OS內存管理系統去更新進程地址空間的映射關系。

如何處理stack所在地址空間區域的爭議

stack
會一直擴展到碰到異常的地址B,B可能是一個readonly的地址或者是一個保護空洞,在向下擴展stack情況下,如果地址B偏上,會導致stack
空間變小,如果偏下,一旦函數局部變量幾乎占滿了stack底到B的空間,mmap雖然也能unmap掉這段區域然后remap,然而這會使數據混亂,造
成嚴重問題。

拍腦袋的結論

mmap或者brk期間,比較stack頂部與esp寄存器,若小于則回收(等于是正常的,大于是不可能的)。

Linux真實的做法

Linux
沒有判斷什么esp寄存器,Linux的原則很簡單,只要一個地址處在一個vma范圍內或者處在stack可擴展的范圍內,且擁有權限的,它就是可以訪
問,內核是不管這個VMA是屬于stack還是heap或者別的什么,具體由應用程序自己控制,也就是說,你完全可以寫一段代碼,把地址空間中所有可以寫
的區域全部清零,這完全有可能,緩沖區溢出可能是一種蓄意的破壞,然而程序員偶然的錯誤也會造成破壞,雖然他們大多數都不知道錯誤是如何發生的。我不想用
文字長篇大論Linux是如何管理VMA的,你知道這個應該是一個前提,你必須知道這個。我用一段代碼以及兩個圖示來展示Linux系統內核是如何管理
stack附近的地址空間映射的,并且在第二張圖中給出,如果你非要蓄意破壞,會造成什么問題。也就是說,一旦發生莫名奇妙的錯誤,你必須能從細節上理解
這個錯誤是如何發生的。

演示代碼

#include<stdio.h>
#include<stdlib.h>
#include<sys/types.h>
#include<unistd.h>
#include<sys/mman.h>

#defineLARGE70000000
#definePAGESIZE4096

//該函數什么都不做,僅僅為了把stack向下擴展
//請注意用ulimit將stack大小限制去掉,這樣會更容易說明問題
voidcall()
{
inti;
chara[LARGE];

//請相信,一定是在中間的賦值中觸發segfault,因為兩邊的元素要么處在stack/fixmapvma,
//要么處在游離的,及其孤獨的,被fixmap給截斷了的vma中。因此下面的賦值不會引發段錯誤:
//a[0]=1;
//a[LARGE-1]=1;
for(i=0;i<LARGE;i++){
a[i]=1;
}
}


intmain(intargc,char**argv)
{
inti;
char*p_map,*p_base,*p_base2;

printf(%d\\ninitstate\\n,getpid());

//獲取stack的大致地址,并且PAGESIZE對齊。
p_base=(char*)&i;
p_base2=(char*)(((unsignedlong)p_base)&~4095);

//獲取pagesize對齊的用來fixmap的地址,該地址起點在當前stack的下面。
p_base2=(char*)((unsignedlong)p_base2-(unsignedlong)36*PAGESIZE);
getchar();

//調用fixmap,顯然,如果你仔細在getchar期間分析了/proc/xx/maps文件并且
//得到了上述的那些magicnumber,下面的mmap無論如何是會成功的!
p_map=(char*)mmap((void*)p_base2,PAGESIZE*3,PROT_READ|PROT_WRITE,
MAP_FIXED|MAP_PRIVATE|MAP_ANONYMOUS,-1,0);
if(p_map==MAP_FAILED){
printf(failed1\\n);
}else{
printf(beforeunmapfixmaparoundstack\\n);
getchar();
//成功了就釋放掉它,此時的地址空間恢復成mmap之前的狀況
munmap(p_map,PAGESIZE*3);
}

printf(afterunmapfixmaparoundstack\\n);
getchar();

call();

printf(afterextendstack[first]\\n);
getchar();

//依然調用之前的那個一模一樣的mmap進行fixmap,由于調用了call,stack
//空間已經擴展到了這個fixmap的fixaddress,很遺憾,成功了,然而它將stackvma
//一刀切成了兩段。不管怎樣,訪問還是可以進行的。
p_map=(char*)mmap((void*)p_base2,PAGESIZE*3,PROT_READ|PROT_WRITE,
MAP_FIXED|MAP_PRIVATE|MAP_ANONYMOUS,-1,0);
if(p_map==MAP_FAILED){
printf(failed2\\n);
}

printf(aftersecondfixmaparoundstackatthesameaddress\\n);
getchar();

//這里更狠!部分unmap了上面那個fixmapvma,留下一個空洞。
munmap(p_map,PAGESIZE);

printf(afterunmapfixmaparoundstackincompletely\\n);
getchar();

//當空洞被touch的時候,不會引發stackextend!而是直接segfault!爆炸!
call();

//永遠不會到達這里!
printf(afterextendstack[second]\\n);
getchar();

return0;
}

針對上述代碼的圖解

下面一幅圖展示一直到出事之前,該進程的stack附近的地址空間映射區域是怎么演化的:

下面一幅圖展示出事的過程以及這個事故的原因:

測試方式


果你覺得圖是我自己畫出來的,那么肯定有一個疑問,我是基于什么畫出來的,事實上,我并不是通過看代碼畫出來的,我是通過不斷查看procfs的maps
文件實時了解該進程地址空間的細節,將其轉換成了上面的圖示,為了讓我有機會到另一個終端去查maps文件,我在代碼中增加了getchar調用,每次查
看完maps文件,我會拍一下鍵盤的回車鍵。我的測試如下:

代碼編譯后的進程輸出

root@abcd:~# ./a.out
7846
init state

before unmap fixmap around stack

after unmap fixmap around stack

after extend stack[first]

after second fixmap around stack at the same address

after unmap fixmap around stack incompletely

段錯誤 (core dumped)

以下是查看每一步進程maps文件的輸出:

root@abcd:~# cat /proc/`ps -e|grep a.out|awk \'{print $1}\'`/maps |tail -n 6|head -n 4
2b2c01483000-2b2c01487000 r--p 00157000 fe:00 387296 /lib/libc-2.11.2.so
2b2c01487000-2b2c01488000 rw-p 0015b000 fe:00 387296 /lib/libc-2.11.2.so
2b2c01488000-2b2c0148f000 rw-p 00000000 00:00 0
7fff6236a000-7fff6237f000 rw-p 00000000 00:00 0 [stack]
root@abcd:~# cat /proc/`ps -e|grep a.out|awk \'{print $1}\'`/maps |tail -n 6|head -n 4
2b2c01487000-2b2c01488000 rw-p 0015b000 fe:00 387296 /lib/libc-2.11.2.so
2b2c01488000-2b2c0148f000 rw-p 00000000 00:00 0
7fff62359000-7fff6235c000 rw-p 00000000 00:00 0
7fff6236a000-7fff6237f000 rw-p 00000000 00:00 0 [stack]
root@abcd:~# cat /proc/`ps -e|grep a.out|awk \'{print $1}\'`/maps |tail -n 6|head -n 4
2b2c01483000-2b2c01487000 r--p 00157000 fe:00 387296 /lib/libc-2.11.2.so
2b2c01487000-2b2c01488000 rw-p 0015b000 fe:00 387296 /lib/libc-2.11.2.so
2b2c01488000-2b2c0148f000 rw-p 00000000 00:00 0
7fff6236a000-7fff6237f000 rw-p 00000000 00:00 0 [stack]
root@abcd:~# cat /proc/`ps -e|grep a.out|awk \'{print $1}\'`/maps |tail -n 6|head -n 4
2b2c01483000-2b2c01487000 r--p 00157000 fe:00 387296 /lib/libc-2.11.2.so
2b2c01487000-2b2c01488000 rw-p 0015b000 fe:00 387296 /lib/libc-2.11.2.so
2b2c01488000-2b2c0148f000 rw-p 00000000 00:00 0
7fff5e0bb000-7fff6237f000 rw-p 00000000 00:00 0 [stack]
root@abcd:~# cat /proc/`ps -e|grep a.out|awk \'{print $1}\'`/maps |tail -n 6|head -n 4
2b2c01488000-2b2c0148f000 rw-p 00000000 00:00 0
7fff5e0bb000-7fff62359000 rw-p 00000000 00:00 0
7fff62359000-7fff6235c000 rw-p 00000000 00:00 0
7fff6235d000-7fff6237f000 rw-p 00000000 00:00 0 [stack]
root@abcd:~# cat /proc/`ps -e|grep a.out|awk \'{print $1}\'`/maps |tail -n 6|head -n 4
2b2c01488000-2b2c0148f000 rw-p 00000000 00:00 0
7fff5e0bb000-7fff62359000 rw-p 00000000 00:00 0
7fff6235a000-7fff6235c000 rw-p 00000000 00:00 0
7fff6235d000-7fff6237f000 rw-p 00000000 00:00 0 [stack]
我怕上面的文字信息太亂,格式在不同瀏覽器會有問題,我還特意截了一張圖:

有什么用


可以用這種方式徹底限制一個進程的stack的大小,越界了不是報錯,而是segfault。然后你可以signal捕獲這個segfault,在里面把
那個未完全unmap的fixmap vma以及那個可憐且孤獨的殘缺的stack
vma給徹底unmap掉。不過這確實沒什么好玩的。有什么用呢?它的作用就是讓你更加深入理解Linux對虛擬地址空間的管理方式。

小Tips

本文不涉及線程棧,但是倒也不難,線程棧一般在heap區或者中間的大塊mmap區動態分配,mmap的時候給它一個MAP_GROWSDOWN標志就可以了。關于它的管理方式,沒啥差別。核心問題在于,缺頁異常處理程序是怎么識別到一個缺頁是一個vma內部的缺頁(結果就是調頁),還是vma外部的缺頁。在后一種情況下,缺頁處理邏輯還要進一步識別是stack的缺頁(結果就是extend stack然后調頁),還是非stack缺頁(結果就是segfault...)。
Linux的stack除非遇到本文所述的這種方式的擠兌收縮,它是永遠擴展的。如果你想閱讀Linux的內核代碼,那么也需要理解下面的事實:
1.find_vma函數能找到vma只有一個限制,即輸入地址只要小于查找vma的end即可,并非很多人想象的那樣輸入地址必須處在查找vma的start和end之間;
2.find_vma函數之所以實現得如此incompletely,是因為為了簡化缺頁中斷的處理,同時也是為了提供一種更加統一的方式同時處理upgrows和downgrows的vma。

后話


然這個問題問得有點亂,但是如果能找上述回答連續扯5分鐘的,應該是真行!不過我不知道怎樣的語言表達能力可以不用圖解和代碼把上面的每一個細節說清
楚...總之,我覺得我的這個題目是一個好題目??梢越ㄗh給看到此文的人,把它做面試題吧。凡是發現不了題目問題的以及說不出所以然的,一律不要!這真是
一道好測試題啊,它是如此之好,以至于我還想再出幾道比它更好的。

總結

以上是生活随笔為你收集整理的请使劲回答一个关于UNIX/Linux自动扩展stack的问的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。