setjmp和longjmp的使用

int setjmp(jmp_buf envbuf); setjmp函数用缓冲区envbuf保存系统堆栈的内容,以便后续的longjmp函数使用。setjmp函数初次启用时返回0值。

void longjmp(jmp_buf envbuf, int val);

longjmp函数中的参数envbuf是由setjmp函数所保存的堆栈环境,参数val设置setjmp函数的返回值。longjmp函数本身是没有返回值的,它执行后跳转到保存envbuf参数的setjmp函数调用,并由setjmp函数调用返回,此时setjmp函数的返回值就是val。

调用longjmp函数时不能使setjmp函数返回0,如果val为0,则setjmp函数返回1。longjmp函数从来不返回,因为它调用后就跳转到setjmp函数保存的堆栈处,恢复堆栈开始执行,所以longjmp函数不会返回。

setjmp和longjmp函数使用示例:

#include <stdio.h>

#include <setjmp.h>
static jmp_buf buf;
int main()
{
    int b;
    b = 3;
    if (setjmp(buf) != 0) 
    {
        printf("b=%d\n", b); 
        return 0;
    }
    b = 5;
    longjmp(buf, 1);
    return 0;
}

setjmp执行时返回0,执行b等于5,调用longjmp,跳转到setjmp调用,setjmp返回1,打印b=5

setjmp与longjmp结合使用时,它们必须有严格的先后执行顺序,也即先调用setjmp函数,之后再调用longjmp函数,以恢复到先前被保存的“程序执行点”。否则,如果在setjmp调用之前,执行longjmp函数,将导致程序的执行流变的不可预测,很容易导致程序崩溃而退出

nt setjmp(jmp_buf  jmpb)设置缓冲区来保存堆栈的内容,将保存的上下文存入进程的自身的数据空间(u区),并继续在当前的上下文中执行,一旦碰到了longjmp,进城就从该进程的u区,取出先前保存的上下文,并恢复该进程的上下文为先前保存的上下文。这时核心将使得进程从setjmp处执行.

void longjmp(jmp_buf jmpb, int retval) 使进程返回到 setjmp处执行,retval表示此时setjmp的返回值。

longjmp必须在setjmp调用之后,而且longjmp必须在setjmp的作用域之内。

jmp_buf的的定义

typedef struct _jmp_buf  
{  
int _jp[_JBLEN+1];  
} jmp_buf[1];

jmp_buf 类型是一个大小为1的数组

传入setjmp中的值是一个结构体的首地址(指针),这同时也说明setjmp必须和处理的内容在同一个函数内,如果单独包装在一个函数内,等函数运行结束运行记录就会丢失,变为NULL,会引发未知后果.

具体来说,在一个函数中使用setjmp来初始化一个全局标号,然后只要该函数未曾返回,那么在其它任何地方都可以通过longjmp调用来跳转到setjmp的下一条语句执行。实际上setjmp函数将发生调用处的局部环境保存在了一个jmp_buf的结构当中,只要主调函数中对应的内存未曾释放(函数返回时局部内存就失效了),那么在调用longjmp的时候就可以根据已保存的jmp_buf参数恢复到setjmp的地方执行。

setjmp函数的返回值(直接返回时为0,longjmp跳转返回时为longjmp的状态参数retval,根据setjmp的返回值就可以判断程序是正常执行还是进行异常处理。

http://hi.baidu.com/passerryan/item/4a468a5ef5a9013394eb0528

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
setjmplongjmp是C语言中提供的两个非常重要的函数,目的是实现函数间的跳转和状态保存。 setjmp函数用于保存当前函数执行的上下文,同时返回一个int类型的。这个setjmp函数自动生成,用于之后调用longjmp函数时确定跳转目标。 longjmp函数则用于恢复之前setjmp所保存下来的上下文,使得程序能够直接跳转到指定的目标。 下面是setjmplongjmp在mips汇编语言中的实现: setjmp: ``` #setjmp implementation in MIPS assembly language .text .align 2 .globl setjmp .ent setjmp setjmp: addi sp, sp, -32 # allocate stack frame sw ra, 28(sp) # save return address sw fp, 24(sp) # save frame pointer addi fp, sp, 32 # set new frame pointer sw a0, 0(fp) # save jmp_buf li v0, 0 # return 0 jr ra .end setjmp ``` longjmp: ``` #longjmp implementation in MIPS assembly language .text .align 2 .globl longjmp .ent longjmp longjmp: lw fp, 24(sp) # restore frame pointer lw ra, 28(sp) # restore return address addi sp, sp, 32 # deallocate stack frame lw t0, 0(fp) # restore jmp_buf lw ra, 4(t0) # restore return address lw sp, 8(t0) # restore stack pointer lw s0, 12(t0) # restore frame pointer jr ra .end longjmp ``` 以上是setjmplongjmp在MIPS汇编语言中的实现。需要注意的是,这里使用了MIPS体系结构的调用约定,即函数返回在v0寄存器中,函数参数在a0-a3寄存器中,被调用者保存的寄存器在栈帧中。在实际使用中,应根据特定的编译器和操作系统的要求进行调整。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值