• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            S.l.e!ep.¢%

            像打了激速一樣,以四倍的速度運轉,開心的工作
            簡單、開放、平等的公司文化;尊重個性、自由與個人價值;
            posts - 1098, comments - 335, trackbacks - 0, articles - 1
              C++博客 :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

            DLL_THREAD_DETACH 認識誤區

            Posted on 2010-02-07 17:18 S.l.e!ep.¢% 閱讀(920) 評論(0)  編輯 收藏 引用 所屬分類: RootKit
            DLL_THREAD_DETACH 認識誤區
            2009-06-01 22:20

            轉自:CSDN 微軟 陳本峰 http://blog.csdn.net/WinGeek/

            文章鏈接:http://blog.csdn.net/WinGeek/archive/2009/06/01/4230741.aspx

            DLL 里面使用TLS (Local Thread Storage) 的常見做法是:在DLLMain的DLL_PROCESS_ATTACH/DLL_THREAD_ATTACH 被調用的時候為每個線程(Thread)分配內存,而在DLL_THREAD_DETACH/DLL_PROCESS_DETACH 被調用的時候釋放內存。 MSDN文章《Using Thread Local Storage in a Dynamic-Link Library》 上有這樣的示例代碼。

            BOOL WINAPI DllMain(HINSTANCE hinstDLL, // DLL module handle
            ??? DWORD fdwReason,??????????????????? // reason called
            ??? LPVOID lpvReserved)???????????????? // reserved
            {
            ??? LPVOID lpvData;
            ??? BOOL fIgnore;

            ??? switch (fdwReason)
            ??? {
            ??????? case DLL_PROCESS_ATTACH:
            ??????????? // Allocate a TLS index.
            ??????????? if ((dwTlsIndex = TlsAlloc()) == TLS_OUT_OF_INDEXES)??????????????? return FALSE;
            ???????? case DLL_THREAD_ATTACH:
            ???????????? lpvData = (LPVOID) LocalAlloc(LPTR, 256); //為每個Thread分配內存
            ??????????? if (lpvData != NULL)
            ??????????????? fIgnore = TlsSetValue(dwTlsIndex, lpvData);
            ??????????? break;
            ???????? case DLL_THREAD_DETACH:
            ???????????? lpvData = TlsGetValue(dwTlsIndex);
            ??????????? if (lpvData != NULL)
            ??????????????? LocalFree((HLOCAL) lpvData); //釋放內存
            ??????????? break;
            ???????? case DLL_PROCESS_DETACH:
            ??????????? lpvData = TlsGetValue(dwTlsIndex);
            ??????????? if (lpvData != NULL)
            ??????????????? LocalFree((HLOCAL) lpvData); //釋放內存
            ??????????? TlsFree(dwTlsIndex);
            ??????????? break;
            ???????? default:
            ??????????? break;
            ??? }
            ???? return TRUE;
            }


            這段代碼認為DLL_THREAD_DETACH 總是會被調用, 但實際情況并非如此。在某些情況下DLL_THREAD_DETACH并不會被調用, 結果造成內存泄漏。 接下來做2個簡單實驗說明這個問題。

            實驗代碼:

            typedef void (__stdcall *FNSLEEP)();

            void CallTestDLL()
            {
            ??? FNSLEEP pfnSleep = (FNSLEEP)::GetProcAddress(g_hDLLModule, "DoSleep");
            ??? ATLASSERT(pfnSleep);
            ??? (*pfnSleep)();
            }

            DWORD WINAPI ThreadProc( LPVOID lpParam)
            {
            ??? CallTestDLL();
            ??? return 0;
            }??

            g_hDLLModule = ::LoadLibrary(_T("TestDLL.dll"));
            ATLTRACE("[Thread %d] LoadLibrary=0x%.8x\n", ::GetCurrentThreadId());
            CallTestDLL();
            const int MAX_THREAD = 2;
            HANDLE hThread[MAX_THREAD];
            for (int i=0; i < MAX_THREAD; i++)
            {
            ?? hThread[i] = ::CreateThread(NULL, 0, ThreadProc, 0, 0, NULL);
            }
            Sleep(MAX_THREAD * 1000);
            ::FreeLibrary(g_hDLLModule);

            輸出結果1:

            [Thread 4976] DLL_PROCESS_ATTACH??????????????? //主線程
            [Thread 4976] LoadLibrary=0x0ecbf9d4
            [Thread 4976] DoSleep() in DLL
            [Thread 7860] DLL_THREAD_ATTACH????????????????? //CreateThread 產生的線程
            [Thread 736] DLL_THREAD_ATTACH??????????????????? //CreateThread 產生的線程
            [Thread 736] DoSleep() in DLL
            [Thread 7860] DoSleep() in DLL
            [Thread 736] DLL_THREAD_DETACH
            [Thread 7860] DLL_THREAD_DETACH
            [Thread 4976] DLL_PROCESS_DETACH??????????????? //主線程

            以上輸入結果我們看到每個Thread 調用DLL函數DoSleep 立即結束,這時候DLL_THREAD_DETACH 被正常調用。 這時只要候稍微改一下代碼,會看到完全不同的結果。

            DWORD WINAPI ThreadProc( LPVOID lpParam)
            {
            ??? CallTestDLL();

            ??? DoSomethingElse(); // 延遲線程結束
            ??? return 0;
            }??


            輸出結果2:


            [Thread 7448] DLL_PROCESS_ATTACH????????????? //主線程
            [Thread 7448] LoadLibrary=0x0b1cf9d4
            [Thread 7448] DoSleep() in DLL
            [Thread 6872] DLL_THREAD_ATTACH
            [Thread 6556] DLL_THREAD_ATTACH
            [Thread 6556] DoSleep() in DLL
            [Thread 6872] DoSleep() in DLL
            [Thread 7448] DLL_PROCESS_DETACH???????????? //主線程


            我們發現,CreateThread 產生的線程并沒有調用DLL_THREAD_DETACH 。

            結論:

            如果是線程在DLL被卸載(調用FreeLibrary) 之前結束,則DLL_THREAD_DETACH 會被調用。 如果線程在DLL卸載之后結束,則DLL_THREAD_DETACH 不會被調用。

            色偷偷91久久综合噜噜噜噜| 久久青青草原精品国产软件| 女人高潮久久久叫人喷水| 97精品国产97久久久久久免费 | 久久国产视频网| 99久久精品午夜一区二区| 国产精品一区二区久久精品涩爱| 久久香蕉综合色一综合色88| 亚洲国产精品成人久久蜜臀| 久久久久亚洲爆乳少妇无| 青青草原综合久久大伊人精品| 伊人久久大香线蕉综合热线| 狠狠色婷婷久久一区二区| 国产精品久久久久久一区二区三区| 一级做a爱片久久毛片| 中文精品久久久久人妻不卡| 久久久久国产亚洲AV麻豆| 国产精品99久久久久久宅男小说| 亚洲精品白浆高清久久久久久| 久久精品国产只有精品2020| 思思久久99热只有频精品66| 99久久做夜夜爱天天做精品| 中文字幕人妻色偷偷久久| 亚洲国产成人久久综合一 | 女同久久| 久久人妻少妇嫩草AV无码蜜桃| 99精品久久精品一区二区| 国产三级久久久精品麻豆三级| 精品水蜜桃久久久久久久| 国内精品伊人久久久久AV影院| 婷婷久久久亚洲欧洲日产国码AV| 亚洲人成无码久久电影网站| 久久精品国产亚洲av高清漫画| 99久久精品无码一区二区毛片| 久久精品国产只有精品66| 久久99毛片免费观看不卡| 久久午夜免费视频| 亚洲国产精品无码久久SM| 久久99国产精品久久| 99精品国产在热久久| 久久免费线看线看|