/
What is the difference between cdecl and stdcall C function calling protocols?

What is the difference between cdecl and stdcall C function calling protocols?

To fully understand the differences between these two conventions consider the sequence of events that occur when a function is called.

  • First, the function parameters are pushed onto the stack from right to left.

  • Next, the address of the next instruction (after the function call) is pushed on the stack.

  • At this point, the function is now in control and can use the stack to allocate local variables. If the function requires the use of the CPU registers, it saves the register values onto the stack before using them.

  • After the function has completed its task, it restores the original register state by popping the appropriate values from the stack.

/* 1. calling function in C++ */ i = Function(x, y, z); /* 2. callee function body in C++ */ int Function(int a, int b, int c) { return a + b + c; }

A calling protocol is a contract between the caller and the callee which contains instructions for cleaning up the stack. In the cdecl calling protocol, the calling function cleans the stack whereas with the stdcall convention the callee function cleans the stack after executing.

CDECL

/* 1. calling CDECL 'Function' in pseudo-assembler (similar to what the compiler outputs) */ push on the stack a copy of 'z', then a copy of 'y', then a copy of 'x' call (jump to function body, after function is finished it will jump back here, the address where to jump back is in registers) move contents of register A to 'i' variable pop all from the stack that we have pushed (copy of x, y and z) /* 2. CDECL 'Function' body in pseudo-assembler */ /* Now copies of 'a', 'b' and 'c' variables are pushed onto the stack */ copy 'a' (from stack) to register A copy 'b' (from stack) to register B add A and B, store result in A copy 'c' (from stack) to register B add A and B, store result in A jump back to caller code (a, b and c still on the stack, the result is in register A)

Note line 5 in the calling function cleans the stack

STDCALL

/* 1. calling STDCALL in pseudo-assembler (similar to what the compiler outputs) */ push on the stack a copy of 'z', then a copy of 'y', then a copy of 'x' call move contents of register A to 'i' variable /* 2. STDCALL 'Function' body in pseaudo-assembler */ pop 'a' from stack to register A pop 'b' from stack to register B add A and B, store result in A pop 'c' from stack to register B add A and B, store result in A jump back to caller code (a, b and c are no more on the stack, result in register A)

Note line 7,8,10 where the callee function cleans the stack

In Summary

 

__cdecl

__stdcall

Move from stack to program counter

Next instruction address is being pushed from stack to program counter without removing the function parameters from the stack.

Clears the function from the stack before moving the next instruction address from the stack to the program counter

Responsibility for cleaning the stack

The Caller is responsible

The Callee is responsible

Number of function parameters

Can vary, the convention supports 'variadic' functions

Has to be fixed so that the convention can clear the stack

Performance

Every __cdecl call requires additional code from the caller to clear the stack parameters

Additional code is unnecessary since Callee is responsible for clearing stack parameters, thus has a marginal increase in performance

Uses

default for C and C++ programs

Chameleon uses and WinAPIs

Naming conventions

Function names are decorated with a leading underscore

Function names are decorated by a leading underscore and a trailing @-sign followed by the number of bytes of parameters taken by the function.