這篇文章主要講解了“setjmp的使用方法”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“setjmp的使用方法”吧!
成都創(chuàng)新互聯(lián)公司堅(jiān)持“要么做到,要么別承諾”的工作理念,服務(wù)領(lǐng)域包括:網(wǎng)站制作、成都網(wǎng)站設(shè)計(jì)、企業(yè)官網(wǎng)、英文網(wǎng)站、手機(jī)端網(wǎng)站、網(wǎng)站推廣等服務(wù),滿足客戶于互聯(lián)網(wǎng)時(shí)代的五臺(tái)網(wǎng)站設(shè)計(jì)、移動(dòng)媒體設(shè)計(jì)的需求,幫助企業(yè)找到有效的互聯(lián)網(wǎng)解決方案。努力成為您成熟可靠的網(wǎng)絡(luò)建設(shè)合作伙伴!
以前對(duì)于C語(yǔ)言的setjmp和longjmp從來(lái)都是知道有這么個(gè)函數(shù), 但是不知道什么情況下要使用, 甚至于不知道setjmp的實(shí)現(xiàn)機(jī)制是什么樣子的.
這次在實(shí)現(xiàn)coroutine的過(guò)程中雖然沒(méi)有使用setjmp來(lái)實(shí)現(xiàn), 但是由于setjmp來(lái)實(shí)現(xiàn)coroutine的所有操作都是可以在用戶態(tài)進(jìn)行的, 因此順便研究了一下setjmp的實(shí)現(xiàn)機(jī)制.
與我猜測(cè)大致一樣, 在call stack上來(lái)講setjmp一定先于或等于longjmp處于的位置, 這樣其實(shí)longjmp所做的不過(guò)就是將當(dāng)前寄存器上下文恢復(fù)成setjmp函數(shù)保存的值即可. 調(diào)用longjmp的函數(shù)與調(diào)用setjmp的函數(shù)共用一個(gè)??臻g, 因此longjmp恢復(fù)完寄存器后造成的結(jié)果就是恢復(fù)寄存器上下文并將??臻g釋放到setjmp位于的地方.
那么這就很好解釋為什么不能將setjmp使用函數(shù)再封裝一層了, 假設(shè)使用函數(shù)封裝setjmp代碼如下:
int try(jmp_buf jmp)
{
int err;
err = setjmp(jmp);
printf("%d\n", err);
return err;
}
void execption(jmp_buff j, int err)
{
return longjmp(j, err)
}
int main()
{
jmp_buff j;
try(j);
execption(j, 2);
}
如上代碼其實(shí)try函數(shù)與execption共用部分棧空間, 當(dāng)longjmp回到try函數(shù)時(shí), 即使用寄存器被恢復(fù)了, 但是事實(shí)上try函數(shù)的棧結(jié)構(gòu)已經(jīng)被破壞了, 此時(shí)longjmp之后try函數(shù)的行為是無(wú)法預(yù)料的.
------------------------
| main |
|-----------------------
| try | excption |
|---------| |
|---------|------------|
在使用setjmp時(shí)一定要注意棧是否被覆蓋的問(wèn)題, 即調(diào)用setjmp時(shí)所保存的esp的信息直到調(diào)用前l(fā)ongjmp之前, 不應(yīng)該會(huì)出更有比setjmp保存esp時(shí)更多的空閑空間(如棧是自頂向下增加, 則setjmp到longjmp之間出現(xiàn)的esp的值絕不能大于setjmp保存的esp的值.
感謝各位的閱讀,以上就是“setjmp的使用方法”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)setjmp的使用方法這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!