青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

春暖花開(kāi)
雪化了,花開(kāi)了,春天來(lái)了
posts - 149,comments - 125,trackbacks - 0

摘自:http://www.diybl.com/course/3_program/c++/cppjs/2008426/111619.html
  
      去年11月的MSDN雜志曾刊登過(guò)一篇文章 Break Free of Code Deadlocks in Critical Sections Under Windows ,Matt Pietrek 和 Russ Osterlund 兩位對(duì)臨界區(qū)(Critical Section)的內(nèi)部實(shí)現(xiàn)做了一次簡(jiǎn)短的介紹,但點(diǎn)到為止,沒(méi)有繼續(xù)深入下去,當(dāng)時(shí)給我的感覺(jué)就是癢癢的,呵呵,于是用IDA和SoftIce大致分析了一下臨界區(qū)的實(shí)現(xiàn),大致弄明白了原理后也就沒(méi)有深究。現(xiàn)在乘著Win2k源碼的東風(fēng),重新分析一下這塊的內(nèi)容,做個(gè)小小的總結(jié)吧 :P
     臨界區(qū)(Critical Section)是Win32中提供的一種輕量級(jí)的同步機(jī)制,與互斥(Mutex)和事件(Event)等內(nèi)核同步對(duì)象相比,臨界區(qū)是完全在用戶態(tài)維護(hù)的,所以僅能在同一進(jìn)程內(nèi)供線程同步使用,但也因此無(wú)需在使用時(shí)進(jìn)行用戶態(tài)和核心態(tài)之間的切換,工作效率大大高于其它同步機(jī)制。
     臨界區(qū)的使用方法非常簡(jiǎn)單,使用 InitializeCriticalSection 或 InitializeCriticalSectionAndSpinCount 函數(shù)初始化一個(gè) CRITICAL_SECTION 結(jié)構(gòu);使用 SetCriticalSectionSpinCount 函數(shù)設(shè)置臨界區(qū)的Spin計(jì)數(shù)器;然后使用 EnterCriticalSection 或 TryEnterCriticalSection 獲取臨界區(qū)的所有權(quán);完成需要同步的操作后,使用 LeaveCriticalSection 函數(shù)釋放臨界區(qū);最后使用 DeleteCriticalSection 函數(shù)析構(gòu)臨界區(qū)結(jié)構(gòu)。
     以下是MSDN中提供的一個(gè)簡(jiǎn)單的例子:

    以下為引用:

 // Global variable
 CRITICAL_SECTION CriticalSection;

 void main()
 {
     ...

     // Initialize the critical section one time only.
     if (!InitializeCriticalSectionAndSpinCount(&CriticalSection, 0x80000400) )
         return;
     ...

     // Release resources used by the critical section object.
     DeleteCriticalSection(&CriticalSection)
 }

 DWORD WINAPI ThreadProc( LPVOID lpParameter )
 {
     ...

     // Request ownership of the critical section.
     EnterCriticalSection(&CriticalSection);

     // Access the shared resource.

     // Release ownership of the critical section.
     LeaveCriticalSection(&CriticalSection);

     ...
 }

     首先看看構(gòu)造和析構(gòu)臨界區(qū)結(jié)構(gòu)的函數(shù)。
     InitializeCriticalSection 函數(shù)(ntosdll esource.c:1210)實(shí)際上是調(diào)用 InitializeCriticalSectionAndSpinCount 函數(shù)(resource.c:1266)完成功能的,只不過(guò)傳入一個(gè)值為0的初始Spin計(jì)數(shù)器;InitializeCriticalSectionAndSpinCount 函數(shù)主要完成兩部分工作:初始化 RTL_CRITICAL_SECTION 結(jié)構(gòu)和 RTL_CRITICAL_SECTION_DEBUG 結(jié)構(gòu)。前者是臨界區(qū)的核心結(jié)構(gòu),下面將著重討論;后者是調(diào)試用結(jié)構(gòu),Matt 那篇文章里面分析的很清楚了,我這兒就不羅嗦了 :P
     RTL_CRITICAL_SECTION結(jié)構(gòu)在winnt.h中定義如下:

以下為引用:

 typedef struct _RTL_CRITICAL_SECTION {
     PRTL_CRITICAL_SECTION_DEBUG DebugInfo;

     //
     //  The following three fields control entering and exiting the critical
     //  section for the resource
     //

     LONG LockCount;
     LONG RecursionCount;
     HANDLE OwningThread;        // from the thread''s

ClientId->UniqueThread
     HANDLE LockSemaphore;
     ULONG_PTR SpinCount;        // force size on 64-bit systems when packed
 } RTL_CRITICAL_SECTION, *PRTL_CRITICAL_SECTION;

     InitializeCriticalSectionAndSpinCount 函數(shù)中首先對(duì)臨界區(qū)結(jié)構(gòu)進(jìn)行了初始化

     DebugInfo 字段指向初始化臨界區(qū)時(shí)分配的RTL_CRITICAL_SECTION_DEBUG結(jié)構(gòu);
     LockCount 字段是臨界區(qū)中最重要的字段,初始值為-1,當(dāng)臨界區(qū)被獲取(Hold)時(shí)此字段大于等于0;
     RecursionCount 字段保存當(dāng)前臨界區(qū)所有者線程的獲取緩沖區(qū)嵌套層數(shù),初始值為0;
     OwningThread 字段保存當(dāng)前臨界區(qū)所有者線程的句柄,初始值為0;
     LockSemaphore 字段實(shí)際上是一個(gè)auto-reset的事件句柄,用于喚醒等待獲取臨界區(qū)的阻塞線程,初始值為0;
     SpinCount 字段用于在多處理器環(huán)境下完成輕量級(jí)的CPU見(jiàn)同步,單處理器時(shí)沒(méi)有使用(初始值為0),多處理器時(shí)設(shè)置為SpinCount參數(shù)值(最大為MAX_SPIN_COUNT=0x00ffffff)。此外 RtlSetCriticalSectionSpinCount 函數(shù)(resource.c:1374)的代碼與這兒設(shè)置SpinCount的代碼完全一樣。

     初始化臨界區(qū)結(jié)構(gòu)后,函數(shù)會(huì)根據(jù)SpinCount參數(shù)的一個(gè)標(biāo)志位判斷是否需要預(yù)先初始化 LockSemaphore 字段,如果需要?jiǎng)t使用NtCreateEvent創(chuàng)建一個(gè)具有訪問(wèn)權(quán)限的同步用事件核心對(duì)象,初始狀態(tài)為沒(méi)有激發(fā)。這一初始化本來(lái)是在 EnterCriticalSection 函數(shù)中完成的,將之顯式提前可以進(jìn)一步優(yōu)化 EnterCriticalSection 函數(shù)的性能。
     值得注意的是,這一特性僅對(duì)Win2k有效。MSDN里面說(shuō)明如下:

以下為引用:

 Windows 2000:  If the high-order bit is set, the function preallocates the event used by the EnterCriticalSection function. Do not set this bit if you are creating a large number of critical section objects, because it will consume a significant amount of nonpaged pool. This flag is not necessary on Windows XP and later, and it is ignored.

     與之對(duì)應(yīng)的 DeleteCriticalSection 函數(shù)完成關(guān)閉事件句柄和是否調(diào)試結(jié)構(gòu)的功能。

     臨界區(qū)真正的核心代碼在win2kprivate tosdlli386critsect.asm里面,包括_RtlEnterCriticalSection、_RtlTryEnterCriticalSection和_RtlLeaveCriticalSection三個(gè)函數(shù)。

     _RtlEnterCriticalSection 函數(shù) (critsect.asm:85) 首先檢查SpinCount是否為0,如果不為0則處理多處理器架構(gòu)下的問(wèn)題[分支1];如為0則使用原子操作給LockCount加一,并判斷是否其值為0。如果加一后LockCount大于0,則此臨界區(qū)已經(jīng)被獲取[分支2];如為0則獲取當(dāng)前線程TEB中的線程句柄,保存在臨界區(qū)的OwningThread字段中,并將RecursionCount設(shè)置為1。調(diào)試版本則調(diào)用RtlpCriticalSectionIsOwned函數(shù)在調(diào)試模式下驗(yàn)證此緩沖區(qū)是被當(dāng)前線程獲取的,否則在調(diào)試模式下激活調(diào)試器。最后還會(huì)更新TEB的CountOfOwnedCriticalSections計(jì)數(shù)器,以及臨界區(qū)調(diào)試結(jié)構(gòu)中的EntryCount字段。
     如果此臨界區(qū)已經(jīng)被獲取[分支2],則判斷獲取臨界區(qū)的線程句柄是否與當(dāng)前線程相符。如果是同一線程則直接將RecursionCount和調(diào)試結(jié)構(gòu)的EntryCount字段加一;如果不是當(dāng)前線程,則調(diào)用RtlpWaitForCriticalSection函數(shù)等待此臨界區(qū),并從頭開(kāi)始執(zhí)行獲取臨界區(qū)的程序。
     多CPU情況的分支處理方式類似,只是多了對(duì)SpinCount的雙重檢查處理。

     接著的_RtlTryEnterCriticalSection(critsect.asm:343)函數(shù)是一個(gè)輕量級(jí)的嘗試獲取臨界區(qū)的函數(shù)。偽代碼如下:

以下為引用:

 if(CriticalSection->LockCount == -1)
 {
   // 臨界區(qū)可用
   CriticalSection->LockCount = 0;
   CriticalSection->OwningThread = TEB->ClientID;
   CriticalSection->RecursionCount = 1;

   return TRUE;
 }
 else
 {
   if(CriticalSection->OwningThread == TEB->ClientID)
   {
     // 臨界區(qū)是當(dāng)前線程獲取
     CriticalSection->LockCount++;
     CriticalSection->RecursionCount++;

     return TRUE;
   }
   else
   {
     // 臨界區(qū)已被其它線程獲取
     return FALSE;
   }
 }
 
 

 

 

     最后的_RtlLeaveCriticalSection(critsect.asm:271)函數(shù)釋放已獲取的臨界區(qū),實(shí)現(xiàn)就比較簡(jiǎn)單了,實(shí)際上就是對(duì)嵌套計(jì)數(shù)器和鎖定計(jì)數(shù)器進(jìn)行操作。偽代碼如下:

 

以下為引用:

 if(--CriticalSection->RecursionCount == 0)
 {
bsp;  // 臨界區(qū)已不再被使用
   CriticalSection->OwningThread = 0;
 

   if(--CriticalSection->LockCount)
   {
     // 仍有線程鎖定在臨界區(qū)上
     _RtlpUnWaitCriticalSection(CriticalSection)
   }
 }
 else
 {
   --CriticalSection->LockCount
 }

posted on 2008-12-01 14:05 Sandy 閱讀(1832) 評(píng)論(0)  編輯 收藏 引用 所屬分類: c++學(xué)習(xí)
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            一区二区91| 老牛影视一区二区三区| 亚洲国产精品va在线看黑人| 午夜视黄欧洲亚洲| 国产欧美日韩精品在线| 欧美有码在线视频| 欧美在线观看网站| 尤物精品在线| 亚洲激情在线播放| 欧美午夜精品伦理| 欧美在线free| 久久精品导航| 亚洲伦理中文字幕| 一区二区三区 在线观看视频| 欧美日韩在线免费观看| 午夜一级在线看亚洲| 久久www成人_看片免费不卡| 激情综合色综合久久| 久久亚洲精品视频| 欧美区高清在线| 欧美一区二区三区视频免费播放| 欧美在线观看视频| 亚洲激情在线观看视频免费| 一本久道久久综合婷婷鲸鱼| 国内精品国产成人| 亚洲美女视频| 伊甸园精品99久久久久久| 亚洲精品小视频在线观看| 国产视频一区三区| 亚洲伦理精品| 在线观看一区二区视频| 中文欧美日韩| 亚洲九九精品| 久久大香伊蕉在人线观看热2| 日韩视频在线永久播放| 欧美一区二区三区免费观看| 99视频精品免费观看| 午夜日韩在线| 亚洲午夜精品网| 欧美r片在线| 久久久久国内| 国产精品sm| 亚洲精品一品区二品区三品区| 国产午夜精品久久久| 亚洲免费成人av电影| 亚洲国产成人tv| 久久9热精品视频| 午夜精品福利一区二区蜜股av| 久久亚裔精品欧美| 久久人人97超碰人人澡爱香蕉| 欧美午夜精品理论片a级按摩 | 亚洲午夜免费福利视频| 你懂的视频欧美| 麻豆精品网站| 国模叶桐国产精品一区| 亚洲免费在线视频一区 二区| 一本色道久久综合亚洲精品小说 | 欧美黄网免费在线观看| 国产一区二区三区免费不卡 | 欧美成人午夜激情在线| 国产真实久久| 久久精品道一区二区三区| 欧美与黑人午夜性猛交久久久| 国产精品激情偷乱一区二区∴| 亚洲国产精品第一区二区| 在线不卡a资源高清| 久久精品午夜| 免费国产自线拍一欧美视频| 伊人伊人伊人久久| 久久三级福利| 欧美大片免费看| 亚洲国产精品专区久久| 卡通动漫国产精品| 亚洲国产高清自拍| 亚洲精品乱码久久久久久| 欧美va亚洲va日韩∨a综合色| 噜噜噜91成人网| 亚洲人成网站精品片在线观看| 免费观看在线综合色| 欧美韩日一区二区三区| 亚洲高清一区二| 欧美精选一区| 亚洲先锋成人| 久久精品一区蜜桃臀影院| 国产有码在线一区二区视频| 久久在线免费观看| 亚洲欧洲一区| 亚洲欧美日韩直播| 国产一区二区三区免费在线观看| 久久久久久久久久看片| 亚洲激情在线观看| 亚洲欧美日韩电影| 韩日精品视频一区| 欧美精品www| 亚洲无亚洲人成网站77777| 欧美中文字幕在线观看| 亚洲国产mv| 欧美四级在线观看| 久久精品国产99精品国产亚洲性色| 欧美成人精品一区| 亚洲午夜免费福利视频| 国语自产偷拍精品视频偷 | 欧美日韩国产美女| 欧美一区二区三区视频| 国产精品视频一二三| 欧美一区二区三区在线| 欧美激情一区二区三区在线| 亚洲一区二区三区精品在线| 国产一在线精品一区在线观看| 欧美精品日韩一区| 校园激情久久| 亚洲精品网站在线播放gif| 久久蜜桃香蕉精品一区二区三区| 亚洲国产精品激情在线观看| 国产精品一区二区三区成人| 你懂的国产精品| 欧美中日韩免费视频| 一区二区三区四区国产精品| 欧美高清视频一区| 欧美综合二区| 亚洲自拍偷拍麻豆| 亚洲人体1000| 久久久久高清| 性做久久久久久| 亚洲六月丁香色婷婷综合久久| 欧美刺激午夜性久久久久久久| 欧美亚洲免费高清在线观看| 亚洲精品美女| 亚洲福利免费| 国产中文一区二区三区| 国产精品日韩欧美一区| 欧美日韩免费在线| 欧美激情第六页| 男女精品网站| 久久手机免费观看| 久久激情五月丁香伊人| 午夜性色一区二区三区免费视频 | 亚洲福利视频一区| 免费不卡在线视频| 久久夜色精品| 噜噜噜91成人网| 久久综合亚洲社区| 久久免费视频网| 久久九九免费| 久久久久久婷| 久久久久久有精品国产| 久久久久9999亚洲精品| 久久aⅴ国产紧身牛仔裤| 亚洲欧美日韩国产中文| 欧美一区二区免费观在线| 香蕉尹人综合在线观看| 性做久久久久久| 欧美伊人精品成人久久综合97| 性一交一乱一区二区洋洋av| 欧美一区永久视频免费观看| 久久成人综合视频| 久久婷婷人人澡人人喊人人爽| 久久免费视频一区| 国产视频在线观看一区二区| 国产女主播在线一区二区| 国产精品人人做人人爽人人添 | 久久噜噜噜精品国产亚洲综合| 久久久久99| 欧美激情一区二区三区在线视频观看 | 亚洲国产裸拍裸体视频在线观看乱了| 亚洲大胆美女视频| 日韩亚洲精品视频| 亚洲性线免费观看视频成熟| 欧美亚洲网站| 免费不卡中文字幕视频| 欧美日韩成人在线观看| 欧美日韩亚洲精品内裤| 国产乱人伦精品一区二区| 国产热re99久久6国产精品| 亚洲电影自拍| 99av国产精品欲麻豆| 午夜精品一区二区三区电影天堂| 篠田优中文在线播放第一区| 久久gogo国模裸体人体| 欧美亚洲在线| 久久精品国产77777蜜臀| 欧美午夜激情在线| 国产亚洲综合在线| 国产午夜精品久久久久久久| 欧美日韩国产二区| 欧美日韩理论| 欧美视频免费在线观看| 亚洲午夜久久久久久久久电影网| 欧美在线视频网站| 久久久蜜桃一区二区人| 久久综合网络一区二区| 国产精品免费看| 国产精品久久久免费| 国产精品久久97| 国产精品久久久久9999| 国产精品羞羞答答| 在线观看视频欧美| 国产一区二区精品久久91| 亚洲成人影音| 国产日本欧美在线观看|