C ++程序集“RETN 10”

时间:2013-07-17 06:20:15

标签: c++ assembly

我有以下功能:

int __declspec() MyFunc(SOCKET sSocket, const char* sData, int sSize, int sFlag)
{
    pSocket = sSocket;
    return send(sSocket,sData, sSize, sFlag);
}

以下是编译后的汇编代码:

PUSH EBP
MOV EBP,ESP
PUSH DWORD PTR SS:[EBP+14]               // Flags
MOV EAX,DWORD PTR SS:[EBP+8]
PUSH DWORD PTR SS:[EBP+10]               // DataSize
MOV DWORD PTR DS:[pSocket],EAX
PUSH DWORD PTR SS:[EBP+C]                // Data
PUSH EAX                                 // Socket
CALL DWORD PTR DS:[<&WS2_32.#19_send>]   // send
POP EBP
RETN

我的问题是:

  1. RETNRETN 8RETN 10之间的区别是什么?
  2. 我必须将最终RETN更改为RETN 10,我应该对C ++代码进行哪些更改?

1 个答案:

答案 0 :(得分:1)

  

我必须将最终的RETN更改为RETN 10,我应该对我的c ++代码进行哪些更改?

要让被调用的函数在返回之前(使用RET(N) imm或通过其他方式)清理堆栈,请使用stdcall calling convention

如果您使用的是Microsoft的C / C ++编译器,则可以通过添加__stdcall modifier来实现此目的,如:

void __stdcall foo(int arg1, int arg2) {
  // ...
}

如果您正在使用GCC,则可以使用stdcall attribute

来实现此目的
void __attribute__ ((stdcall)) foo(int arg1, int arg2) {
  // ...
}

您当然可以#define __stdcall作为__attribute__ ((stdcall))来节省一些打字,并使代码更具可移植性。

在Cygwin中使用GCC编译类似上面的函数会导致以下程序集:

_foo@8:
push    ebp
mov ebp, esp
... (omitted for brevity)
leave
ret 8