Вызов функции с адреса не работает
Я не могу понять почему, если я позвоню CreateFileA
Функция с использованием адреса сбоя программы.
Этот фрагмент должен вызывать CreateFileA
используя его адрес, полученный из GetProcAddress
вызов. Конечно, в этом простом примере я могу просто вызвать его как обычно, но мне нужно получить каждый адрес вручную (даже LoadLibraryA
а также GetProcAddress
но для краткости я этого не сделал)
Что я делаю неправильно? Почему если я использую CreateFileA
вместо (*CreateFileAAddr)
на последней строке это работает, даже если они указывают на то же самое?
#include <windows.h>
typedef HMODULE(*_LoadLibraryA)(LPCSTR);
typedef FARPROC(*_GetProcAddress)(HMODULE, LPCSTR);
typedef HANDLE(*_CreateFileA)(LPCSTR, DWORD, DWORD, LPSECURITY_ATTRIBUTES, DWORD, DWORD, HANDLE);
int main() {
_LoadLibraryA LoadLibraryAAddr = (_LoadLibraryA)&LoadLibraryA;
_GetProcAddress GetProcAddressAddr = (_GetProcAddress)&GetProcAddress;
HMODULE kernel32Handle = (*LoadLibraryAAddr)("Kernel32.dll");
if (kernel32Handle == NULL) return -1;
_CreateFileA CreateFileAAddr = (_CreateFileA)((*GetProcAddressAddr)(kernel32Handle, "CreateFileA"));
if ((unsigned long)CreateFileAAddr != (unsigned long)&CreateFileA) return -1; //Just to check if the address is correct for Debug
(*CreateFileAAddr)("123.txt", GENERIC_WRITE, 0, NULL, OPEN_ALWAYS, OPEN_ALWAYS, NULL);
return 0;
}
1 ответ
Согласно моему комментарию, функции Windows API и типы указателей на указанные функции должны быть аннотированы __stdcall
(или же WINAPI
), который изменит их соглашение о вызовах на то, которое ожидается API (больше информации о соглашении о вызовах stdcall здесь).
typedef HMODULE(__stdcall *_LoadLibraryA)(LPCSTR);
typedef FARPROC(__stdcall *_GetProcAddress)(HMODULE, LPCSTR);
typedef HANDLE(__stdcall *_CreateFileA)(LPCSTR, DWORD, DWORD, LPSECURITY_ATTRIBUTES, DWORD, DWORD, HANDLE);
Также примечание стороны: *
должны быть размещены после __stdcall
по какой-то причине; в противном случае аннотация ничего не сделает.