• <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>

            大熊的口袋

             

            TRACE改進(jìn)版(靜態(tài)tls在dll中使用的問題)

            前面的文章里自己搗鼓了一個(gè)vc下的TraceX的實(shí)現(xiàn)。主要就是基于調(diào)用棧的共享。
            http://www.shnenglu.com/zwp/archive/2009/11/04/100134.html
            先是調(diào)用sprintf格式化輸出到一段靜態(tài)線程局部存儲(chǔ)空間,然后輸出到OutputDebugString。從而可以被調(diào)試器捕獲。
            之所以選擇靜態(tài)tls而不選擇全局變量,主要是防止多線程環(huán)境下可能出現(xiàn)的讀寫沖突。至于為何不使用棧變量(局部自動(dòng)變量)是因?yàn)楸仨毨脳I犀F(xiàn)有的參數(shù),且參數(shù)個(gè)數(shù)是不確定的,所以無法移動(dòng)和拷貝這些參數(shù)。
            然而,昨天在一個(gè)測(cè)試工程中動(dòng)態(tài)加載了一個(gè)dll,在該dll里調(diào)用了Trace時(shí)卻出錯(cuò)了。跟下來一看是在訪問tls變量時(shí)出錯(cuò)。而靜態(tài)加載該dll則不會(huì)出錯(cuò)。
            后來從網(wǎng)上搜索才得知。靜態(tài)tls在dll被動(dòng)態(tài)加載時(shí)并不會(huì)被初始化。這也是靜態(tài)tls最大的一個(gè)缺點(diǎn)了。缺點(diǎn)歸缺點(diǎn),不能因?yàn)椴荒苡镁头艞塗race啊,所以后來經(jīng)過思考決定用動(dòng)態(tài)tls來重新實(shí)現(xiàn)相關(guān)線程局部存儲(chǔ)部分。
            代碼如下:
            #ifndef __DEBUG_INFO_H__
            #define __DEBUG_INFO_H__

            #ifndef NDEBUG
            // 最多512個(gè)字符
            VOID _cdecl TraceA(const char *msgFmt, );
            VOID _cdecl TraceW(
            const wchar_t *msgFmt, );

            #define TRACEA TraceA
            #define TRACEW TraceW

            // 確保所有的TRACE都在這個(gè)范圍之內(nèi)。
            bool TraceStartUp();
            void TraceShutDown();

            #define TRACESTARTUP TraceStartUp
            #define TRACESHUTDOWN TraceShutDown

            #else

            #define TRACEA ((void)0)
            #define TRACEW ((void)0)

            #define TRACESTARTUP ((void)0)
            #define TRACESHUTDOWN ((void)0)
            #endif



            #endif


            #include 
            "debuginfo.h"
            #include 
            <stdio.h>
            #include 
            <Windows.h>

            #ifndef NDEBUG

            const UINT g_cInvalidTlsIdx = 0xFFFFFFFF;
            UINT g_unTlsSlotIdx 
            = g_cInvalidTlsIdx;

            bool TraceStartUp()
            {
                
            if(g_cInvalidTlsIdx == g_unTlsSlotIdx)
                
            {
                    g_unTlsSlotIdx 
            = TlsAlloc();
                    
            if (g_cInvalidTlsIdx == g_unTlsSlotIdx)
                    
            {
                        
            return false;
                    }

                }

                
            return true;
            }


            void TraceShutDown()
            {
                
            if (g_cInvalidTlsIdx != g_unTlsSlotIdx)
                
            {
                    TlsFree(g_unTlsSlotIdx);
                }

                g_unTlsSlotIdx 
            = g_cInvalidTlsIdx;
            }


            static void __stdcall SaveRetAddr(LPVOID pRet)
            {
                TlsSetValue(g_unTlsSlotIdx, pRet);
            }


            static LPVOID __stdcall GetRetAddr()
            {
                
            return TlsGetValue(g_unTlsSlotIdx);
            }



            const UINT g_cBufSize = 512;
            static char * __stdcall GetAddrOftszBuf()
            {
                
            return new char[g_cBufSize];
            }


            static wchar_t * __stdcall GetAddrOftswzBuf()
            {
                
            return new wchar_t[g_cBufSize];
            }


            static void __stdcall DestroyStrBuf(LPVOID lpBuf)
            {
                delete [] lpBuf;
            }



            _declspec(naked) VOID TraceA(
            const char *msgFmt, )
            {
                __asm 
            {
                    call SaveRetAddr;
                    call GetAddrOftszBuf;
                    push eax;
                    call dword ptr [sprintf];
                    pop eax;
                    push eax;
                    push eax;
                    call dword ptr [OutputDebugStringA];
                    call DestroyStrBuf;
                    call GetRetAddr;
                    push eax;
                    ret;
                }

            }


            _declspec(naked) VOID TraceW(
            const wchar_t *msgFmt, )
            {
                __asm 
            {
                    call SaveRetAddr;
                    call GetAddrOftszBuf;
                    push eax;
                    call dword ptr [_swprintf];
                    pop eax;
                    push eax;
                    push eax;
                    call dword ptr [OutputDebugStringW];
                    call DestroyStrBuf;
                    call GetRetAddr;
                    push eax;
                    ret;
                }

            }


            #endif

            這里多增加了兩個(gè)函數(shù)StartTrace和ShutdownTrace。用來從tls中獲取一個(gè)可用索引和釋放該索引。這就要求所有的Trace都必須在這兩個(gè)調(diào)用之間工作,否則就會(huì)出問題。

            posted on 2009-11-14 12:54 大熊的口袋 閱讀(2213) 評(píng)論(5)  編輯 收藏 引用 所屬分類: cppwin32

            評(píng)論

            # re: TRACE改進(jìn)版(靜態(tài)tls在dll中使用的問題) 2009-11-14 14:28 OwnWaterloo

            別搞了兄弟,用vsprintf之類的函數(shù)就可以了。

              回復(fù)  更多評(píng)論   

            # re: TRACE改進(jìn)版(靜態(tài)tls在dll中使用的問題) 2009-11-14 14:47 李佳

            額... 大部分都用匯編做的... 支持一下吧   回復(fù)  更多評(píng)論   

            # re: TRACE改進(jìn)版(靜態(tài)tls在dll中使用的問題) 2009-11-14 15:02 OwnWaterloo

            @李佳
            如果只是練習(xí)著玩玩,機(jī)器碼寫都沒問題。

            如果是實(shí)際應(yīng)用上,這么實(shí)現(xiàn)這么簡單的需求 —— 10行代碼了事 ——用到匯編和線程局部存儲(chǔ),那……
            有點(diǎn)過了……

              回復(fù)  更多評(píng)論   

            # re: TRACE改進(jìn)版(靜態(tài)tls在dll中使用的問題)[未登錄] 2009-11-14 15:27 q

            這位仁兄,不錯(cuò)!精神可佳,支持一下.

              回復(fù)  更多評(píng)論   

            # re: TRACE改進(jìn)版(靜態(tài)tls在dll中使用的問題) 2009-11-16 08:57 大熊的口袋

            之前也聽說過有v系列的函數(shù),然后又看到過一些基于棧共享實(shí)現(xiàn)一些東東的思想。這次遇到這個(gè)問題,首先想到的就是棧共享。完全沒有考慮到v系類函數(shù),不想在棧共享實(shí)現(xiàn)過程中又遇到線程同步問題,用了靜態(tài)tls后又出現(xiàn)dll被動(dòng)態(tài)加載后無法正常使用的問題。遂出現(xiàn)現(xiàn)在的解決方案。雖然不是最簡單的,但是確實(shí)學(xué)到一些東西:)  回復(fù)  更多評(píng)論   

            導(dǎo)航

            統(tǒng)計(jì)

            公告

            常用鏈接

            留言簿(2)

            隨筆分類

            隨筆檔案

            win32 & debug

            搜索

            積分與排名

            最新評(píng)論

            閱讀排行榜

            評(píng)論排行榜

            欧美国产精品久久高清| 久久精品国产清自在天天线 | 久久国产高清一区二区三区| 99国产精品久久| 久久国产综合精品五月天| 精品人妻伦九区久久AAA片69| 久久人爽人人爽人人片AV| 久久国产成人| 99久久久精品| 少妇内射兰兰久久| 人人狠狠综合久久亚洲高清| 狠狠狠色丁香婷婷综合久久俺| 亚洲国产香蕉人人爽成AV片久久 | 欧洲性大片xxxxx久久久| 久久天天躁狠狠躁夜夜96流白浆| 精品熟女少妇aⅴ免费久久| 久久久国产乱子伦精品作者| 精品国产日韩久久亚洲 | 伊人久久精品无码二区麻豆| 久久久精品视频免费观看| 久久久久久久亚洲Av无码| 18禁黄久久久AAA片| 亚洲国产成人精品女人久久久| 91精品国产色综合久久| 久久亚洲精品国产精品| AV无码久久久久不卡蜜桃| 性做久久久久久久久浪潮| 久久精品国产清自在天天线| 国产免费久久久久久无码| 国产福利电影一区二区三区久久久久成人精品综合 | 狠狠综合久久综合88亚洲| 日日狠狠久久偷偷色综合96蜜桃| 狠狠综合久久综合中文88| 国产99久久九九精品无码| 曰曰摸天天摸人人看久久久| 一本久久久久久久| 国产精品激情综合久久| 久久99国产一区二区三区| 久久夜色精品国产www| 日韩AV毛片精品久久久| 狠狠色丁香久久婷婷综合_中|