x86 Assembly: Why Do I Need Stack Frames?

2019-02-22 08:35发布

问题:

On most x86 Assembly (NASM especifically) code samples I see around (even on the ones generated by GCC) I see what's called "setup of stack frame". Like this:

main: 
        /*setting the stack frame*/
        push    ebp     
        mov     ebp,esp

        ...
        code goes here
        ...

        /*removing the stack frame*/
        mov     esp, ebp
        pop     ebp

I have 3 questions about this practice:

  1. If my code doesn't touch the stack then setting/removing the stack frame as above is completely useless, right?

  2. Even if my code uses the stack, as long as pop everything I push (leaving the stack as it was essentially) then again setting up a stack frame is completely useless, right?

  3. As I see it the only purpose of this would be to save the value of ESP so that I can play around with it on my code without worrying about messing things up, and once I am done I simply restore its original value. Is this the purpose of the stack frame setup or am I missing something?

Thanks

回答1:

Well, in fact, you don't need stack frames.

Stack frames are convenient when you save registers and store local variables in stack - to make writing and debugging easier: you just set ebp to a fixed point in stack and address all stack data using ebp. And it's easier to restores esp at the end.

Also, debuggers often expect stack frames to be present, otherwise you can get inaccurate stack call for example.

So, the answer to 1 is yes, the answer to 2 and 3 is above.



回答2:

You are essentially correct.

Stack frames do have certain benefits, though, even when you don't need a fixed stack reference to access parameters and locals. In particular having them there allows for accurate stack walking for generating stack traces for debugging purposes.



回答3:

This is done by convention. C language functions use stack frames to access parameters that are sent to functions, and to set up dynamic local variables. That is why they do it in the sample code that you are looking at. Of course if you want to do it your way, you can, but you won't be able to invoke your code from C etc.

EDIT: I am also pretty sure that there are compilers that implement different calling conventions that optimize that and maybe do not create a frame at all. So basically, you are right. Stack frames are not necessary.