亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關(guān)于我們
? 蟲蟲下載站

??

?? 兼容內(nèi)核漫談 適合想將Windows上的程序移植到其它平臺上的朋友研究查看
??
?? 第 1 頁 / 共 5 頁
字號:
        KeReleaseDispatcherDatabaseLock(OldIrql);
    } else {
        /* Return Locked and with a Wait */
        KTHREAD *Thread = KeGetCurrentThread();
        Thread->WaitNext = TRUE;
        Thread->WaitIrql = OldIrql;
    }
    /* Return the previous State */
    return PreviousState;
}[/code]
    參數(shù)Increment和Wait所起的作用與前面KeReleaseSemaphore()中的相同。所謂“設(shè)置事件”其實就是一種特殊的、變通的V操作。具體的操作分兩種情況:
    1. 如果IsListEmpty()為真,即等待隊列是空的、沒有線程在睡眠等待,就把Event->Header.SignalState設(shè)置成1。這樣,如果此后有線程對此事件對象執(zhí)行P操作、即NtWaitForSingleObject(),就因此而不必睡眠等待。這對于通知型和同步型的事件對象都是一樣。
    2. 已經(jīng)有線程在這個對象上睡眠等待,那就要從中喚醒一個或所有線程。這時候的處理取決于事件對象的類型以及等待的方式:
    l 對于通知型的事件對象,或者等待者的等待方式是WaitAll,而且此前SignalState為0,就將SignalState置1,并通過KiWaitTest()喚醒這個線程,以及等待隊列中所有符合條件的線程。可是,要是SignalState本來就已經(jīng)是1,則沒有任何影響。
    l 否則,對于同步型的事件對象,并且等待者的等待方式是WaitAny,就通過KiAbortWaitThread()喚醒等待隊列中的第一個線程。此時并不改變SignalState的值。因為既然喚醒了一個線程,就已經(jīng)把這籌碼消耗掉了。
    這里KiWaitTest()和KiAbortWaitThread()的區(qū)別在于:KiWaitTest()是在一個while循環(huán)中對等待隊列中的所有進程執(zhí)行KiAbortWaitThread(),條件是SignalState大于0。對于信號量,由于每喚醒一個線程就使SignalState減1,這循環(huán)很快就停止了,一般是只喚醒一個線程。但是如前所述,通知型事件對象在喚醒一個線程的時候不改變SignalState的值。于是,這個while循環(huán)就會喚醒等待隊列中的所有進程。不過這里也有例外,如果其中的某個線程是在多個“可等待對象”上等待,而且等待方式是WaitAll,那就還要看是否別的條件也滿足了,不然就只好把它跳過,這也是KiWaitTest()的代碼中按排好了的。

    前面講過,一旦將通知型事件對象的SignalState設(shè)置成1,它就一直保持為1,P操作不會改變它的值。即使再對其執(zhí)行一次KeSetEvent(),也不會改變它的值,因為本來就已經(jīng)是1了。為了使其變成0,以便再次使用這個事件對象,就需要對其執(zhí)行另一個系統(tǒng)調(diào)用NtResetEvent()。同樣,NtResetEvent()的主體是KeResetEvent()。

[code][NtSetEvent() > KeResetEvent()]

LONG  STDCALL
KeResetEvent(PKEVENT Event)
{
    KIRQL OldIrql;
    LONG PreviousState;
   
    DPRINT("KeResetEvent(Event %x)\n",Event);
   
    /* Lock the Dispatcher Database */
    OldIrql = KeAcquireDispatcherDatabaseLock();
   
    /* Save the Previous State */
    PreviousState = Event->Header.SignalState;
   
    /* Set it to zero */
    Event->Header.SignalState = 0;
   
    /* Release Dispatcher Database and return previous state */   
    KeReleaseDispatcherDatabaseLock(OldIrql);
    return PreviousState;
}[/code]
    解釋就沒有必要了。注意調(diào)用NtResetEvent()的不必就是NtSetEvent()的調(diào)用者,而可以是別的線程或內(nèi)核模塊。不過有些內(nèi)核模塊只能調(diào)用KeResetEvent(),而不是NtResetEvent()。
    Windows還有個系統(tǒng)調(diào)用NtPulseEvent(),這是把NtSetEvent()和NtResetEvent()組合在了一起,相當(dāng)于先NtSetEvent()、然后馬上就NtResetEvent()。這樣,其效果就是喚醒已經(jīng)在通知型事件對象上等待的所有線程,但是下不為例。而對于同步型事件對象則大致等同于NtSetEvent()。
    可見,雖然“事件”實質(zhì)上是“信號量”的一種特例和變種,但是在使用上卻有著明顯的差別。信號量的“正宗”的用途是構(gòu)筑臨界區(qū)。在這種應(yīng)用中,一個線程得以通過P操作進入臨界區(qū)的原因可能是有另一個線程執(zhí)行了V操作,但是既然進了臨界區(qū)就總有從臨界區(qū)退出而執(zhí)行V操作的時候。這樣,一個線程在P操作以后總是有個V操作。從總體上看,每個線程的P操作和V操作是平衡的、即數(shù)量相等的。但是“事件”則不同,“事件”并不是用來構(gòu)筑臨界區(qū)、而純粹是用于線程間同步的。在這里,等待事件發(fā)生的一方總是執(zhí)行P操作,而發(fā)出事件通知的一方則總是執(zhí)行V操作。在前面對于“信號量”的比喻中把P操作比作領(lǐng)取通行證,把V操作比作交還通行證。相比之下,對于“事件”則相當(dāng)于領(lǐng)取的通行證從來不交還,而另有供應(yīng)者在不時地提供新的通行證。而且,特別有意義的是,發(fā)出事件通知的一方還不必非得是一個線程,也可以是內(nèi)核中的某些子系統(tǒng),例如設(shè)備驅(qū)動,所以也可以用于線程與內(nèi)核之間的同步,特別是廣泛地應(yīng)用于設(shè)備驅(qū)動。當(dāng)然,發(fā)出事件通知的一方更不必局限于某一個特定的線程,而是任何一個線程都可以。
    為了幫助讀者加深對事件機制的理解,下面是一個內(nèi)核線程DebugLogThreadMain的代碼:

[code]VOID STDCALL
DebugLogThreadMain(PVOID Context)
{
  KIRQL oldIrql;
  IO_STATUS_BLOCK Iosb;
  static CHAR Buffer[256];
  ULONG WLen;

  for (;;)
    {
      LARGE_INTEGER TimeOut;
      TimeOut.QuadPart = -5000000; /* Half a second. */
      KeWaitForSingleObject(&DebugLogEvent, 0, KernelMode, FALSE, &TimeOut);
      KeAcquireSpinLock(&DebugLogLock, &oldIrql);
      while (DebugLogCount > 0)
      {
        if (DebugLogStart > DebugLogEnd)
        {
            WLen = min(256, DEBUGLOG_SIZE - DebugLogStart);
            memcpy(Buffer, &DebugLog[DebugLogStart], WLen);
            Buffer[WLen + 1] = '\n';
            DebugLogStart =  (DebugLogStart + WLen) % DEBUGLOG_SIZE;
            DebugLogCount = DebugLogCount - WLen;
            KeReleaseSpinLock(&DebugLogLock, oldIrql);
            NtWriteFile(DebugLogFile, NULL, NULL, NULL, &Iosb, Buffer, WLen + 1,
                        NULL, NULL);
        }
        else
        {
            WLen = min(255, DebugLogEnd - DebugLogStart);
            memcpy(Buffer, &DebugLog[DebugLogStart], WLen);
            DebugLogStart =
                   (DebugLogStart + WLen) % DEBUGLOG_SIZE;
            DebugLogCount = DebugLogCount - WLen;
            KeReleaseSpinLock(&DebugLogLock, oldIrql);
            NtWriteFile(DebugLogFile, NULL, NULL, NULL, &Iosb, Buffer, WLen,
                        NULL, NULL);
          }
        KeAcquireSpinLock(&DebugLogLock, &oldIrql);
      }
      KeResetEvent(&DebugLogEvent);
      KeReleaseSpinLock(&DebugLogLock, oldIrql);
    }
}[/code]

    這個內(nèi)核線程是為內(nèi)核調(diào)試日志(Log)服務(wù)的。內(nèi)核中有個環(huán)形緩沖區(qū)DebugLog[],以及用作該數(shù)組下標(biāo)的變量DebugLogStart和DebugLogEnd,還有表示環(huán)形緩沖區(qū)中數(shù)據(jù)長度的變量DebugLogCount。不管是哪一個線程,只要是進入了內(nèi)核,如果需要在日志中寫上一筆,就可以把字符串拷貝到這個環(huán)形緩沖區(qū)中,然后要求這個內(nèi)核線程把內(nèi)容寫到一個日志文件中。為此當(dāng)然需要同步,這是通過一個(同步型)事件對象DebugLogEvent達成的。由于是在內(nèi)核中,所以這里對事件對象的操作都直接調(diào)用其內(nèi)核版本,例如KeResetEvent()、而不是NtResetEvent()。此外,對于環(huán)形緩沖區(qū)的使用當(dāng)然還需要互鎖,這是通過“空轉(zhuǎn)鎖”DebugLogLock實現(xiàn)的,不過那不是我們此刻所關(guān)心的。
    每當(dāng)需要生成一項日志時,可以調(diào)用DebugLogWrite():

[code]VOID
DebugLogWrite(PCH String)
{
  KIRQL oldIrql;

   . . . . . .
   KeAcquireSpinLock(&DebugLogLock, &oldIrql);

   if (DebugLogCount == DEBUGLOG_SIZE)
   {
      DebugLogOverflow++;
      KeReleaseSpinLock(&DebugLogLock, oldIrql);
      if (oldIrql < DISPATCH_LEVEL)
      {
        KeSetEvent(&DebugLogEvent, IO_NO_INCREMENT, FALSE);
      }
       return;
   }

   while ((*String) != 0)
   {
      DebugLog[DebugLogEnd] = *String;
      String++;
      DebugLogCount++;

      if (DebugLogCount == DEBUGLOG_SIZE)
      {   
        DebugLogOverflow++;
        KeReleaseSpinLock(&DebugLogLock, oldIrql);
        if (oldIrql < DISPATCH_LEVEL)
        {
           KeSetEvent(&DebugLogEvent, IO_NO_INCREMENT, FALSE);
        }
        return;
      }
      DebugLogEnd = (DebugLogEnd + 1) % DEBUGLOG_SIZE;
   }

    KeReleaseSpinLock(&DebugLogLock, oldIrql);

    if (oldIrql < DISPATCH_LEVEL)
    {
      KeSetEvent(&DebugLogEvent, IO_NO_INCREMENT, FALSE);
    }
}[/code]

    對于這段代碼,以及對于DebugLogWrite()和DebugLogThreadMain()之間怎樣互動,這里就不作解釋了。只是要指出:DebugLogWrite()的每次執(zhí)行可能都在不同線程的上下文里、代表著不同的線程,因為任何線程都可以調(diào)用DebugLogWrite()。另外,想必讀者已經(jīng)注意到,KeResetEvent()是由DebugLogThreadMain()自己調(diào)用、而不是由別的線程調(diào)用的。

    介紹完事件對象,還應(yīng)該提一下,Windows還有一種特殊的“事件對(EventPair)”對象。與此有關(guān)的系統(tǒng)調(diào)用有這么一些:
[code]   NtCreateEventPair()
    NtOpenEventPair()
    NtWaitHighEventPair()
    NtWaitLowEventPair()
    NtSetHighWaitLowEventPair()
    NtSetLowWaitHighEventPair()
    NtSetHighEventPair()
    NtSetLowEventPair()[/code]

    顧名思義,“事件對”就是把兩個事件對象緊密地組合在一起。事實上也正是如此,一個事件對由“高”、“低”兩個事件對象組合構(gòu)成,其設(shè)計意圖是用于“點對點”的雙向進程間通信。實際上這是為提高Windows進程與服務(wù)進程Csrss之間的通信效率而設(shè)置的(Csrss是Windows子系統(tǒng)的管理/服務(wù)進程)。早期的csrss承擔(dān)著許多操作,Windows進程與Csrss之間的通信非常頻繁,所以其效率至關(guān)重要。這種進程間通信的典型情景就是一方喚醒另一方、自身卻又進入睡眠,反過來等待被對方喚醒,就像打乒乓球一樣,為此就專門設(shè)計了NtSetHighWaitLowEventPair()和NtSetLowWaitHighEventPair()兩個系統(tǒng)調(diào)用。不僅如此,為了盡可能地提高效率(在這種情況下的優(yōu)化甚至是以CPU的時鐘周期數(shù)計算的),還專門單獨分配了兩個中斷向量0x2B和0x2C,而不跟別的系統(tǒng)調(diào)用合用0x2E。不過,后來Csrss的許多操作被移到了內(nèi)核中,不再需要那么頻繁的進程間通信了,因而在效率上的容忍度也寬松了一些,所以現(xiàn)在又回到了0x2E,而不再使用0x2B和0x2C這兩個中斷向量。

5. 命名管道(Named Pipe)和信箱(Mail Slot)
    前面提到,如果從字面上理解,那么進程間通信也可以通過磁盤文件而實現(xiàn)。但是,把信息寫入某個磁盤文件,再由另一個進程從磁盤文件讀出,在速度上是很慢的。固然,由于文件緩沖區(qū)(Cache)的存在,對磁盤文件的寫和讀未必都經(jīng)過磁盤,但是那并沒有保證。再說,普通的文件操作也沒有提供進程間同步的手段。所以通過普通的磁盤文件實現(xiàn)進程間通信是不太現(xiàn)實的。但是這也提示我們,如果能實現(xiàn)一種特殊文件,使得對文件的讀寫只在緩沖區(qū)中進行(而不寫入磁盤),并且實現(xiàn)進程間的同步,那倒是個不壞的主意。命名管道就是這樣一種特殊文件。實際上,命名管道還不僅是這樣的特殊文件,它還是一種網(wǎng)絡(luò)通信的機制,只是當(dāng)通信的雙方存在于同一臺機器上時,才落入本文所說的進程間通信的范疇。
    既然命名管道是一種特殊文件,它的創(chuàng)建、打開、讀寫等等操作就基本上都可以利用文件系統(tǒng)中的有關(guān)資源加以實現(xiàn)。當(dāng)然,這畢竟是一種特殊文件,對于使用者來說,最大的特殊之處在于這是一個“先進先出”的字節(jié)流,不能對其執(zhí)行l(wèi)seek()一類的操作。
    先看命名管道的創(chuàng)建,Windows的Win32 API上提供了一對庫函數(shù)CreateNamedPipeA()和CreateNamedPipeW(),前者用于ASCII碼字符串,后者用于“寬字符”即Unicode的字符串,實際上前者只是把8位字符轉(zhuǎn)換成Unicode以后再調(diào)用后者。對CreateNamedPipeW()的調(diào)用大致如下:

[code]  Handle = CreateNamedPipeW(L"[A]\\\\.\\pipe\\MyCont

?? 快捷鍵說明

復(fù)制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
国产精品欧美经典| 中文字幕巨乱亚洲| 久久综合国产精品| 色综合久久天天| 青青草一区二区三区| 中文字幕一区二区三| 日韩视频一区在线观看| 91丨九色丨国产丨porny| 看片的网站亚洲| 亚洲成人资源网| 国产精品久久久久久久第一福利| 在线综合视频播放| 日本韩国欧美在线| 成人福利在线看| 国产一区二区三区最好精华液| 午夜欧美大尺度福利影院在线看| 国产精品国产精品国产专区不蜜| 日韩美女在线视频| 欧美精品久久一区二区三区| av电影天堂一区二区在线观看| 九九国产精品视频| 丝袜美腿亚洲综合| 夜夜嗨av一区二区三区四季av| 国产丝袜欧美中文另类| 日韩美女主播在线视频一区二区三区| 欧美日韩欧美一区二区| 一本久道中文字幕精品亚洲嫩| 成人的网站免费观看| 国产精品99久久久久久宅男| 欧美日韩国产美| 色婷婷久久久亚洲一区二区三区 | 在线看日韩精品电影| 国产精品77777竹菊影视小说| 欧美aaaaaa午夜精品| 日韩精品久久理论片| 午夜成人免费视频| 婷婷激情综合网| 日韩激情一二三区| 日本aⅴ精品一区二区三区| 天天色天天爱天天射综合| 亚洲成人久久影院| 午夜国产精品一区| 青草av.久久免费一区| 日本不卡一区二区三区| 日韩激情一二三区| 久久精品99国产精品| 久久97超碰国产精品超碰| 久久精品999| 狠狠色狠狠色综合日日91app| 激情图区综合网| 国产一区二区精品久久99| 久久精品99国产精品日本| 狠狠色丁香婷综合久久| 激情深爱一区二区| 风间由美一区二区三区在线观看| 国产成人综合网站| 不卡的电影网站| 在线视频国内自拍亚洲视频| 欧美日韩午夜在线视频| 日韩一区二区三区视频| 久久夜色精品国产噜噜av| 国产精品久久久久久一区二区三区 | 99精品久久只有精品| 中文字幕制服丝袜成人av| 中文字幕日韩一区| 亚洲国产综合在线| 免费成人结看片| 国产成人免费视频| 色哟哟日韩精品| 91精品欧美久久久久久动漫| 久久久亚洲国产美女国产盗摄 | 亚洲乱码一区二区三区在线观看| 亚洲影视在线播放| 久久成人久久爱| 99精品热视频| 欧美一级午夜免费电影| 中文久久乱码一区二区| 伊人色综合久久天天人手人婷| 日本一道高清亚洲日美韩| 国产精品91一区二区| 在线精品观看国产| 亚洲精品在线电影| 亚洲精品成人a在线观看| 喷水一区二区三区| 99re8在线精品视频免费播放| 欧美电影影音先锋| 国产精品不卡在线| 欧美一区二区成人6969| 中文在线资源观看网站视频免费不卡 | 美腿丝袜亚洲三区| 99久久精品免费看| 日韩午夜三级在线| 国产精品乱码一区二区三区软件| 视频一区中文字幕| www.亚洲激情.com| 日韩欧美精品在线视频| 一区二区三区在线观看视频| 韩国v欧美v亚洲v日本v| 欧美综合在线视频| 国产亚洲精品bt天堂精选| 亚洲影院理伦片| 成人ar影院免费观看视频| 欧美一区二区三区视频免费| 国产精品久久久久久久久免费相片| 日韩国产在线一| 91色综合久久久久婷婷| 久久久国产精品不卡| 视频精品一区二区| 色综合网色综合| 国产亚洲欧美一级| 麻豆久久一区二区| 欧美日韩国产成人在线免费| 日韩一区在线看| 国产精品 日产精品 欧美精品| 在线播放中文一区| 亚洲一区二区在线免费看| fc2成人免费人成在线观看播放| 日韩一级视频免费观看在线| 一区二区三区欧美视频| 99久久精品费精品国产一区二区| 久久免费看少妇高潮| 蜜芽一区二区三区| 91超碰这里只有精品国产| 亚洲国产精品视频| 欧美亚州韩日在线看免费版国语版| 国产精品狼人久久影院观看方式| 国产精品综合在线视频| 精品99一区二区| 狠狠狠色丁香婷婷综合激情| 欧美一区二区私人影院日本| 欧美日韩一区二区不卡| 亚洲欧美一区二区三区孕妇| 成人高清免费在线播放| 国产午夜三级一区二区三| 国产一区二区在线观看免费| 亚洲精品在线免费观看视频| 免费人成网站在线观看欧美高清| 欧美三级乱人伦电影| 亚洲高清视频中文字幕| 欧美亚洲国产怡红院影院| 一区二区三区在线视频免费| 欧美亚洲精品一区| 日韩和欧美一区二区| 日韩无一区二区| 精品影视av免费| 久久精品亚洲麻豆av一区二区| 国精品**一区二区三区在线蜜桃| www国产精品av| 国产精品综合二区| 国产精品乱码人人做人人爱| 91视频在线观看| 亚洲一区免费在线观看| 欧美精品日韩一本| 韩国三级在线一区| 欧美高清一级片在线观看| 99精品视频一区二区| 亚洲高清在线视频| 欧美大黄免费观看| 国产精品1区二区.| |精品福利一区二区三区| 欧美最新大片在线看| 奇米在线7777在线精品| 国产日韩三级在线| 91豆麻精品91久久久久久| 日韩影视精彩在线| 久久久综合精品| 色噜噜狠狠成人中文综合| 日韩av在线发布| 中文字幕av在线一区二区三区| 色天使久久综合网天天| 天涯成人国产亚洲精品一区av| 精品国产乱码91久久久久久网站| 成人免费三级在线| 亚洲国产中文字幕在线视频综合 | 久久青草国产手机看片福利盒子 | 2020日本不卡一区二区视频| 成人午夜伦理影院| 亚洲国产一区二区a毛片| 精品欧美一区二区在线观看| 白白色 亚洲乱淫| 午夜视频在线观看一区二区三区 | av影院午夜一区| 丝袜国产日韩另类美女| 国产日韩精品一区| 欧美日韩性生活| 成人一区二区三区中文字幕| 亚洲va欧美va人人爽午夜| 久久久久国色av免费看影院| 91九色02白丝porn| 国产专区欧美精品| 亚洲午夜日本在线观看| 欧美经典一区二区| 欧美一卡二卡三卡| 在线影院国内精品| 国产精品亚洲一区二区三区在线 | 麻豆91在线观看| 亚洲美女淫视频| 国产欧美一区二区三区在线看蜜臀 | 欧美人妇做爰xxxⅹ性高电影| 成人黄页在线观看|