2017-10-16 325 views
1

我正在为WH_GETMESSAGE写入全局钩子。但是,当从dll调用GetProcAddress函数时,我得到错误代码127,即ERROR_PROC_NOT_FOUND。它无法找到GetMsgProc。任何想法为什么?调用GetProcAddress时出现错误127

此外,我是这种编程的新手,所以对任何错误都不抱歉。

DLL文件:

#include "windows.h" 
#include <stdio.h> 

BOOL APIENTRY DllMain(HANDLE hModule, DWORD ul_reason_for_call, LPVOID lpReserved) 
{ 
    return TRUE; 
} 

__declspec(dllexport) LRESULT CALLBACK GetMsgProc(int nCode, WPARAM wParam, LPARAM lParam) 
{ 
    MessageBox(NULL, TEXT("I am in"),TEXT("In a DLL"), MB_OK); 

    return CallNextHookEx(NULL, nCode, wParam, lParam); 
} 

程序加载的DLL文件:

#include <windows.h> 
#include <tchar.h> 
#include <stdio.h> 

typedef LRESULT(CALLBACK *LPGetMsgProc)(int nCode, WPARAM wParam, LPARAM lParam); 

int main() 
{ 
    HMODULE hDll = LoadLibrary(_T("../../dllTouchInputHook/x64/Debug/dllTouchInputHook.dll")); 
    LPGetMsgProc proc = (LPGetMsgProc)GetProcAddress(hDll, "GetMsgProc"); 
    if (proc == NULL) { 
     printf("The error code is %d", GetLastError()); 
    } 

    HHOOK hMsgHook = SetWindowsHookEx(WH_GETMESSAGE, proc, hDll, 0); 

    MSG msg; 
    while (GetMessage(&msg, NULL, 0, 0) > 0) { 

     TranslateMessage(&msg); 
     DispatchMessage(&msg); 
    } 

    UnhookWindowsHookEx(hMsgHook); 

    return 0; 
} 
+0

你没有检查从'LoadLibrary'结果。 –

+0

当然它没有找到'GetMsgProc',因为它没有以这个名字出口。 – RbMm

+0

@DanielSęk:我检查了LoadLibrary的结果。它不是空的 – Roger1990

回答

4

的功能没有被发现,因为像你期望它不被导出为"GetMsgProc"。它实际上更像"[email protected]"(32位)或"[email protected]"(64位)。如果您希望导出为"GetMsgProc",那么在编译DLL时需要使用.DEF文件。

你不应该以这种方式实施钩子开始。你应该将呼叫转移到SetWindowsHookEx() DLL本身里面,然后导出函数来调用它,例如:

#include "windows.h" 
#include <stdio.h> 

HINSTANCE hThisDLL; 
HHOOK hMsgHook; 

BOOL APIENTRY DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved) 
{ 
    hThisDLL = hinstDLL; 
    return TRUE; 
} 

LRESULT CALLBACK GetMsgProc(int nCode, WPARAM wParam, LPARAM lParam) 
{ 
    MessageBox(NULL, TEXT("I am in"), TEXT("In a DLL"), MB_OK); 
    return CallNextHookEx(NULL, nCode, wParam, lParam); 
} 

__declspec(dllexport) BOOL WINAPI InstallHook() 
{ 
    if (!hMsgHook) 
     hMsgHook = SetWindowsHookEx(WH_GETMESSAGE, &GetMsgProc, hThisDll, 0); 
    return (hMsgHook != NULL); 
} 

__declspec(dllexport) VOID WINAPI UninstallHook() 
{ 
    if (hMsgHook) 
    { 
     UnhookWindowsHookEx(hMsgHook); 
     hMsgHook = NULL; 
    } 
} 

#include <windows.h> 
#include <tchar.h> 
#include <stdio.h> 

typedef BOOL (WINAPI *LPInstallHook)(); 
typedef VOID (WINAPI *LPUninstallHook)(); 

int main() 
{ 
    HMODULE hDll = LoadLibrary(_T("../../dllTouchInputHook/x64/Debug/dllTouchInputHook.dll")); 
    if (!hDll) 
    { 
     printf("The error code is %d", GetLastError()); 
     return -1; 
    } 

    LPInstallHook installProc = (LPInstallHook) GetProcAddress(hDll, "InstallHook"); // or "_InstallHook" 
    LPUninstallHook uninstallProc = (installProc) ? (LPUninstallHook) GetProcAddress(hDll, "UninstallHook") : NULL; // or "_UninstallHook" 

    if (!(installProc && uninstallProc)) 
    { 
     printf("The error code is %d", GetLastError()); 
     FreeLibrary(hDll); 
     return -1; 
    } 

    if (!installProc()) 
    { 
     printf("The error code is %d", GetLastError()); 
     FreeLibrary(hDll); 
     return -1; 
    } 

    MSG msg; 
    while (GetMessage(&msg, NULL, 0, 0) > 0) { 
     TranslateMessage(&msg); 
     DispatchMessage(&msg); 
    } 

    uninstallProc(); 

    FreeLibrary(hDll); 

    return 0; 
} 
+0

*“_ GetMsgProc @ 20”(64位)。 - - 不,x64不装饰名称。在x64上它将被作为'GetMsgProc'输出,如果它写在* c *上或者用'extern“C''声明'否则将是复杂的* C++ *名称 – RbMm

+0

@RbMm:它实际上取决于使用的编译器。你所描述的是微软的编译器所做的事(默认情况下,它可以在64位中修改名称 - 例如,即使你在另一个评论中说,名称被破坏,即使代码明显加载了x64 DLL) 。如果我们为Windows编程,我们需要使用来自sdk/wdk的windows * lib *文件来静态链接到windows api,其他编译器不需要(并且很少)遵循微软的 –

+0

。结果编译器mangle * c *的名称必须与相应的* lib *文件完全匹配 - 否则我们会得到错误 - 无法解析的外部符号。例如,必须为x86的'_ReadFile @ 20'和x64的'ReadFile'。究竟。我们一些编译器使用另一个名称转换 - 它将与Windows lib文件不兼容 – RbMm

相关问题