Что такое __stdcall? - PullRequest
       61

Что такое __stdcall?

132 голосов
/ 18 ноября 2008

Я изучаю программирование на Win32, а прототип WinMain выглядит так:

int WINAPI WinMain ( HINSTANCE instance, HINSTANCE prev_instance, PSTR cmd_line, int cmd_show )

Я был сбит с толку относительно того, для чего этот идентификатор WINAPI был найден:

#define WINAPI      __stdcall

Что это делает? Я смущен тем, что что-то вообще есть после типа возврата. Для чего __stdcall? Что это означает, когда между типом возвращаемого значения и именем функции есть что-то?

Ответы [ 8 ]

154 голосов
/ 18 ноября 2008

__stdcall - соглашение о вызовах, используемое для функции. Это сообщает компилятору правила, которые применяются для настройки стека, передачи аргументов и получения возвращаемого значения.

Существует ряд других соглашений о вызовах, __cdecl, __thiscall, __fastcall и чудесно названных __naked. __stdcall - это стандартное соглашение о вызовах для системных вызовов Win32.

Википедия охватывает подробности .

Это в первую очередь имеет значение, когда вы вызываете функцию вне вашего кода (например, API-интерфейс ОС) или ОС вызывает вас (как в случае с WinMain). Если компилятор не знает правильного соглашения о вызовах, вы, вероятно, получите очень странные сбои, так как стек не будет управляться правильно.

33 голосов
/ 18 ноября 2008

C или C ++ не определяют эти идентификаторы. Они являются расширениями компилятора и обозначают определенные соглашения о вызовах. Это определяет, куда поместить аргументы, в каком порядке, где вызываемая функция найдет адрес возврата и так далее. Например, __fastcall означает, что аргументы функций передаются через регистры.

Статья Википедии содержит обзор различных соглашений о вызовах, найденных там.

16 голосов
/ 18 ноября 2008

Ответы до сих пор охватывали детали, но если вы не собираетесь переходить к сборке, то все, что вам нужно знать, это то, что и вызывающий, и вызываемый должны использовать одно и то же соглашение о вызовах, иначе получить ошибки, которые трудно найти.

10 голосов
/ 30 июня 2009
10 голосов
/ 18 ноября 2008

Я согласен, что все ответы пока верны, но вот причина. Компиляторы Microsoft C и C ++ предоставляют различные соглашения о вызовах для (предполагаемой) скорости вызовов функций в функциях C и C ++ приложения. В каждом случае вызывающий абонент и вызываемый абонент должны договориться о том, какое соглашение о вызовах использовать. Теперь сама Windows предоставляет функции (API), и они уже скомпилированы, поэтому, когда вы вызываете их, вы должны соответствовать им. Любые вызовы API-интерфейсов Windows и обратные вызовы из API-интерфейсов Windows должны использовать соглашение __stdcall.

5 голосов
/ 18 ноября 2008

Это связано с тем, как вызывается функция - в основном порядок, в котором вещи помещаются в стек и кто отвечает за очистку.

Вот документация, но она мало что значит, если вы не понимаете первую часть:
http://msdn.microsoft.com/en-us/library/zxk0tw93.aspx

4 голосов
/ 17 апреля 2009

__stdcall используется для помещения аргументов функции в стек. После завершения функции он автоматически освобождает память. Это используется для фиксированных аргументов.

void __stdcall fnname ( int, int* )
{
    ...
}

int main()
{
    CreateThread ( NULL, 0, fnname, int, int*...... )
}

Здесь fnname содержит аргументы, которые он непосредственно помещает в стек.

1 голос
/ 23 мая 2016

Мне никогда не приходилось использовать это раньше, до сегодняшнего дня. Это потому, что в моем коде я использую многопоточность и многопоточность API, который я использую, это Windows One (_beginthreadex).

Для начала темы:

_beginthreadex(NULL, 0, ExecuteCommand, currCommand, 0, 0);

Функция ExecuteCommand ДОЛЖНА использовать ключевое слово __stdcall в сигнатуре метода, чтобы beginthreadex мог вызвать его:

unsigned int __stdcall Scene::ExecuteCommand(void* command)
{
    return system(static_cast<char*>(command));
}
...