2011-10-31 99 views
2

我写了自己的setjmp/longjmp,它符合我的需求,如下所示。我在32位系统上进行了测试,效果很好。我为此保存和恢复寄存器eax,ebx,ecx,edi,esi,esp,ebp和eip。x86中的额外寄存器64位

但是,我知道,这对64位系统来说是不够的。首先,我认为我需要用rX替换寄存器eX。其次,我想我需要保存在x86-64位中找到的8个额外寄存器,分别是r8,r9,r10,r11,r12,r13,r14,r15。这足够了,还是我需要做更多?

#define MY_SETJMP(n) __asm__ __volatile__ ("movl %eax, regax"#n";" \ 
    "movl %ebx, regbx"#n";" \ 
    "movl %ecx, regcx"#n";" \ 
    "movl %edi, regdi"#n";" \ 
    "movl %esi, regsi"#n";" \ 
    "movl %esp, regsp"#n";" \ 
    "movl %ebp, regbp"#n";" \ 
    "call next"#n";" \ 
    "next"#n": pop regip"#n";" \ 
    "addl $6, regip"#n";" \ 
    ) 

#define MY_LONGJMP(n) __asm__ __volatile__ ("movl regax"#n", %eax;" \ 
    "movl regbx"#n", %ebx;" \ 
    "movl regcx"#n", %ecx;" \ 
    "movl regdi"#n", %edi;" \ 
    "movl regsi"#n", %esi;" \ 
    "movl regsp"#n", %esp;" \ 
    "movl regbp"#n", %ebp;" \ 
    "jmp *regip"#n";" \ 
    ) 
+7

你为什么写自己的而不是使用_working_系统? –

+0

就像我说过的,我将它用于特殊目的,这是无法使用现有的setjmpt/longjmp完成的。 – MetallicPriest

回答

7

您的指令序列不保存flags register,它可能甚至应该在IA32中保存。维基百科页面包含您可以使用的说明pushfpopf

您需要保存所有向量寄存器,除非您知道该程序不使用它们。请注意:它们也可以用于标量浮点,因此您不需要在程序中使用向量化代码以供它们使用。 噢,如果程序使用浮点,你应该保存历史浮点栈以防使用。 Dan Kruchinin的答案显示了如何一步保存所有这些。

4

也许你需要保存的x87上下文中还使用FXSAVE/fxrestore说明:http://siyobik.info/main/reference/instruction/FXSAVE

虽然,我不知道FXSAVE/fxresrtore可以从用户空间应用程序可以安全使用(即超级用户模式外) ,但你几乎可以做自己做的每件事。

+0

为什么你认为fxsave在用户空间中可能不安全?据我所知,这不是特权指令。 – osgx