• DLL注入——使用全局钩子


    1.简介

    因为进程的地址空间是独立的,发生对应事件的进程不能调用其他进程地址空间的钩子函数。如果钩子函数的实现代码在DLL中,则在对应事件发生时,系统会把这个DLL加载到发生事件的进程地址空间中,使它可以调用钩子函数进行处理。

    所以只要在系统中安装了全局钩子,那么只要进程接收到可以发出钩子的消息,全局钩子的DLL就会被系统自动或者强行加载到进程空间中,这就可以实现DLL注入。

    安装挂钩是通过下列方法实现的:

    1. HHOOK SetWindowsHookExW(
    2. [in] int idHook,
    3. [in] HOOKPROC lpfn,
    4. [in] HINSTANCE hmod,
    5. [in] DWORD dwThreadId
    6. );
    • 参数1:表示要安装的挂钩类型。
    • 参数2:表示的是钩子的回调函数。
    • 参数3:包含由lpfn参数执行的钩子过程的DLL句柄
    • 参数4:与钩子过程关联的线程标识符,如果为0则表示与所有线程相关联。

    这里将参数1设置为WH_GETMESSAGE,因为这种类型的钩子会监视消息队列,又因为Windows系统是基于消息驱动的,所以所有的进程都会有自己的一个消息队列,都会加载WH_GETMESSAGE类型的全局钩子。

    当idHook设置为WH_GETMESSAGE的时候,回调函数lpfn的定义如下:

    LRESULT GetMsgProc(int code, WPARAM wParam, LPARAM lParam);

    2.示例

    2.1生成动态库

    我使用的vs2019,创建一个动态链接库,如下图所示。

     2.2在pch.h文件总声明几个函数如下

    1. // 设置全局钩子
    2. extern"C" __declspec(dllexport) BOOL SetGlobalHook();
    3. // 钩子回调函数
    4. extern"C" __declspec(dllexport) LRESULT GetMsgProc(int code, WPARAM wParam, LPARAM lParam);
    5. 卸载钩子
    6. extern"C" __declspec(dllexport) BOOL UnsetGlobalHook();

    注意:必须添加extern"C" __declspec(dllexport)。

    2.3pch.cpp中定义

    1. // 共享内存
    2. #pragma data_seg("mydata")
    3. HHOOK g_hHook = NULL;
    4. #pragma data_seg()
    5. #pragma comment(linker, "/SECTION:mydata,RWS")
    6. extern HMODULE g_hDllModule;
    7. // 设置全局钩子
    8. BOOL SetGlobalHook()
    9. {
    10. g_hHook = SetWindowsHookEx(WH_GETMESSAGE, (HOOKPROC)GetMsgProc, g_hDllModule, 0);
    11. if (NULL == g_hHook)
    12. {
    13. return FALSE;
    14. }
    15. return TRUE;
    16. }
    17. // 钩子回调函数
    18. LRESULT GetMsgProc(int code, WPARAM wParam, LPARAM lParam)
    19. {
    20. return CallNextHookEx(g_hHook, code, wParam, lParam);
    21. }
    22. //卸载钩子
    23. BOOL UnsetGlobalHook()
    24. {
    25. if (g_hHook)
    26. {
    27. UnhookWindowsHookEx(g_hHook);
    28. }
    29. return TRUE;
    30. }

    回调函数中,简单的调用CallNextHookEx函数表示将当前钩子传递给钩链中的下一个钩子。第一个参数要指定当前钩子的句柄。如果直接返回0,则表示中断钩子传递,这就实现了对钩子进行拦截。

    为了让任意一个独立的进程中对句柄的修改都可以影响到其他进程,就需要在DLL中使用共享内存,来保证将DLL中加载到多个进程以后,一个进程对它的修改可以影响到其他进程。设置共享内存的方式如下:

    1. #pragma data_seg("mydata")
    2. HHOOK g_hHook = NULL;
    3. #pragma data_seg()
    4. #pragma comment(linker, "/SECTION:mydata,RWS")

    2.4在dllmain.cpp中设置g_hDllModule,编译成库。

    1. HMODULE g_hDllModule = NULL;
    2. BOOL APIENTRY DllMain( HMODULE hModule,
    3. DWORD ul_reason_for_call,
    4. LPVOID lpReserved
    5. )
    6. {
    7. switch (ul_reason_for_call)
    8. {
    9. case DLL_PROCESS_ATTACH:
    10. g_hDllModule = hModule;
    11. break;
    12. case DLL_THREAD_ATTACH:
    13. case DLL_THREAD_DETACH:
    14. case DLL_PROCESS_DETACH:
    15. break;
    16. }
    17. return TRUE;
    18. }

    2.5编写一个控制台应用程序来设置全局钩子

    1. typedef BOOL(*SetHook)();
    2. typedef BOOL(*UnsetHook)();
    3. int main()
    4. {
    5. //加载dll
    6. HMODULE hDll = LoadLibrary(L"F:\\hookDll.dll");
    7. if (NULL == hDll)
    8. {
    9. return -1;
    10. }
    11. BOOL isHook = FALSE;
    12. //导出SetGlobalHook函数地址
    13. SetHook SetGlobalHook = (SetHook)GetProcAddress(hDll, "SetGlobalHook");
    14. if (NULL == SetGlobalHook)
    15. {
    16. return -1;
    17. }
    18. //导出UnsetGlobalHook函数地址
    19. UnsetHook UnsetGlobalHook = (UnsetHook)GetProcAddress(hDll, "UnsetGlobalHook");
    20. if (NULL == UnsetGlobalHook)
    21. {
    22. return -1;
    23. }
    24. //调用设置全局钩子
    25. isHook = SetGlobalHook();
    26. if (isHook)
    27. {
    28. printf("Hook is ok!\n");
    29. }
    30. else
    31. {
    32. printf("Hook is error\n");
    33. }
    34. system("pause");
    35. //调用卸载全局钩子
    36. UnsetGlobalHook();
    37. FreeLibrary(hDll);
    38. return 0;
    39. }

    3.使用Process Explorer工具查看

    直接ctrl+F,搜索加载的dll名称,可以看到dll已经注入到很多程序当中。

    4.卸载钩子

     UnhookWindowsHookEx(g_hHook);

  • 相关阅读:
    android开发连接网络
    Postgresql源码(58)元组拼接heap_form_tuple剖析
    【高级程序设计】Week2-2 HTTP and Web
    Spring Boot @Validated 和Javax的@Valid配合使用
    发票查验API如何打破传统发票管理难题?
    Java如何读取OS环境变量吗?
    Java自动化框架:jenkins执行git命令
    基于JAVA人事管理系统测试视频计算机毕业设计源码+系统+mysql数据库+lw文档+部署
    当我们做后仿时我们究竟在仿些什么(一)
    hw share usb device caption description
  • 原文地址:https://blog.csdn.net/wzz953200463/article/details/126513928