日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > windows >内容正文

windows

windows 内核情景分析

發布時間:2025/4/14 windows 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 windows 内核情景分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

原文很長;先轉部分過來,有時間看一下;

?

一 windows 內核情景分析---說明

說明

本文結合《Windows內核情景分析》(毛德操著)、《軟件調試》(張銀奎著)、《Windows核心編程》
、《寒江獨釣-Windows內核安全編程》、《Windows PE權威指南》、《C++反匯編與逆向分析揭秘》以及
ReactOS操作系統 (V0.3.12)源碼,以《Windows內核情景分析》為藍本,對Windows內核重要框架、函數
、結構體進行解析

由于工程龐大,我能理解到的只是冰山一角,但本文力求做到讓每個讀者都能從整體上理解Windows內核
的架構,并大量解釋一些關鍵細節。

本文解讀方式:1、源碼、偽碼結合,展示主流程,很多時候忽略權限、錯誤檢查,多線程互斥等旁枝末

2、函數的參數沒有嚴格排序,很多不重要的參數也省略了,要注意

3、結構體內的成員沒有嚴格排序,成員名稱也不嚴格對應,并只列出一些重要成員

4、一些清理工作,如關閉句柄、釋放內存、釋放互斥對象等工作省略

5、很多時候,函數體開頭聲明的那些沒有初始值的局部變量我都略去了

翻看了毛老師的大作,受益匪淺,在基本理清了原理與細節后,特此做了一番總結,
,ReactOS本來就與Windows有一些小差別,

分析的部分項目截圖:

本文術語約定:

描述符:指用來描述一件事物的“結構體”。如緩沖描述符,描述了一個緩沖的基址、長度等信息。中
斷描述符,描述了那個中斷向量對應的分配狀態、isr等信息

Entry:指表中的表項、條目,有時也指函數入口

SSDT:基本系統服務表(其實全稱應叫系統服務派遣表)

Shadow SSDT:GUI/GDI系統服務函數表,這是第二張SSDT

SSDTDT:系統服務表描述符表,表中每個元素是一個SSDT描述符(注意內核中有兩張SSDT和兩張SSDTDT

IDT:中斷描述符表,每個cpu一個。(每個表項是一個描述符,可以簡單視為isr)

ISR:中斷服務例程,IDT表中的中斷描述符所描述的中斷處理函數

EPR:異常處理例程,IDT表中的異常描述符所描述的異常處理函數

VA:虛擬地址, ? ?PA:物理地址, ? LA:線性地址, ? RVA:相對虛擬地址 ? ?foa:文件偏移

PDE:頁目錄中的表項,保存著對應二級頁表的物理地址,又叫“二級頁表描述符”

PTE:二級頁表中的表項,真正記錄著每個虛擬頁面的映射情況以及其他信息,又叫“映射描述符”

頁目錄:(又叫一級頁表、總頁表),一個PDE數組,這個數組的大小剛好占據一個頁面

二級頁表:一個PTE數組,這個數組的大小也剛好占據一個頁面(進程有一個總頁表+1024個二級頁表)

AREA:地址空間中的一塊連續的區段,VirtualAlloc分配內存都是以區段為單位

內存分配:表示從地址空間中用VirtualAlloc預定或者提交映射一塊內存,不是指malloc、new、
HeapAlloc

PID:進程ID、進程號。(其實也是個句柄)

TID:線程ID、線程號。(其實也是個句柄)

PDO:物理設備對象,相對于fdo而言。Pdo并不一定是最底層的那個硬件pdo

FDO:功能設備對象,相對于pdo而言。Fdo也可能直接訪問硬件芯片。fdo與pdo只是一種相對概念。

棧底pdo:又叫‘基石pdo’,‘硬件pdo’,指用作堆?;哪莻€pdo,它是由相應的總線驅動內部創
建的 。?

端口設備對象:端口驅動或者小端口驅動中創建的設備對象(他下面是硬件pdo)

總線驅動:用來驅動總線的驅動(總線本身也是一種特殊的設備),如pci.sys總線驅動

端口驅動:由廠家提供的真正用來直接訪問硬件芯片的驅動,位于總線驅動上層

功能驅動:指類驅動。如鼠標類驅動mouseclass.sys,磁盤類驅動disk.sys

上層過濾驅動:位于功能類驅動上面的驅動

下層過濾驅動:位于功能驅動下面,端口驅動上面的驅動

頂層驅動:指位于棧頂的驅動

中間驅動:intermediate drivers,凡是夾在頂層驅動與端口驅動之間的那些驅動都叫中間驅動

設備樹:由PnP管理器構造的一顆用來反映物理總線布局的‘硬件設備樹’。

設備節點:設備樹中的節點。每個節點都表示一個真正的‘硬件pdo’

老式驅動:即NT式驅動,指不提供AddDevice或通過NtLoadDriver加載的驅動

WDM驅動:指提供了AddDevice并且不是通過NtLoadDriver加載的驅動

IRP派遣例程:又叫分發例程、派遣函數。驅動程序中用來響應處理irp的函數。(Dispatch)

設備綁定:指將設備‘堆棧’到原棧頂設備上面,成為新的棧頂設備。

文件:指物理介質上的文件(磁盤、光盤、U盤)

文件對象:每次打開設備時生成一個文件對象(文件對象不是文件,僅僅表示對設備的一次打開上下文
,因此文件對象又叫打開者)

套接字驅動:afd.sys

套接字設備:\Device\Afd\Endpoint

套接字文件對象:每打開一次套接字設備生成一個套接字文件對象

套接字FCB:每個套接字文件對象關聯的FCB,用來描述套接字的其他信息

地址文件對象:每次打開傳輸層的tdi設備時生成的一個文件對象,用于套接字綁定

地址對象:傳輸層中為每個地址文件對象創建一個地址對象,用來描述一個地址(IP、端口號、協議等

Socket irp:發往afd套接字設備(即\Device\Afd\Endpoint)的irp

Tdi irp:發往傳輸層設備(即\Device\Tcp,\Device\Udp,\Device\RawIp)的irp

物理卷設備:指磁盤卷、光盤卷、磁帶卷等物理卷設備,由相應類型的硬件驅動創建

磁盤卷設備:指磁盤分區,設備對象名為\Device\HarddiskN\PartitionN 形式(N從0開始)

文件卷設備:由文件系統內部創建的掛載(即綁定)在物理卷上的匿名設備

Cdo:控制設備對象。一個驅動通常創建有一個cdo,用來與外界通信。

FSD:文件系統驅動,File System Driver縮寫。

簇:文件以簇為分配單位。一個文件包含N個簇,簇之間不必物理連續,一個簇一般為4KB

扇區:系統以扇區為單位進行磁盤IO。一個簇包含N個扇區,一個扇區一般為512B

文件塊:磁盤文件中的文件塊,對應于內核中的文件緩沖段

緩沖段:文件塊在內核中的緩沖

ACL:訪問控制表。每個Ntfs文件、內核對象都有一份ACL,記錄了各用戶、組的訪問權限

Token:訪問令牌。每個線程、進程都有一個Token,記錄了包含的特權、用戶、組等信息

SID:指用戶ID、組ID、機器ID,用來唯一標識。

主令牌:進程自己的令牌

客戶令牌:也即模擬令牌。每個線程默認使用進程的令牌,但也可模式使用其他進程的令牌

二 windows內核情景分析--系統調用

Windows的地址空間分用戶模式與內核模式,低2GB的部分叫用戶模式,高2G的部分叫內核模式,位于用戶空
間的代碼不能訪問內核空間,位于內核空間的代碼卻可以訪問用戶空間

一個線程的運行狀態分內核態與用戶態,當指令位于用戶空間時,就表示當前處于內核態,當指令位于內核
空間時,就處于內核態.

一個線程由用戶態進入內核態的途徑有3種典型的方式:

1、 主動通過int 2e(軟中斷自陷方式)或sysenter指令(快速系統調用方式)調用系統服務函數,主
動進入內核

2、 發生異常,被迫進入內核

3、 發生硬件中斷,被迫進入內核

現在討論第一種進入內核的方式:(又分為兩種方式)

1、 通過老式的int 2e指令方式調用系統服務(因為老式cpu沒提供sysenter指令)

如ReadFile函數調用系統服務函數NtReadFile

Kernel32.ReadFile() ?//點號前面表示該函數的所在模塊

{

//所有Win32 API通過NTDLL中的系統服務存根函數調用系統服務進入內核
NTDLL.NtReadFile();

}

NTDLL.NtReadFile()

{

? ?Mov eax,152 ? //我們要調用的系統服務函數號,也即SSDT表中的索引,記錄在eax中
? ?If(cpu不支持sysenter指令)
? ?{
? ? ? Lea edx,[esp+4] //用戶空間中的參數區基地址,記錄在edx中
? ? ? Int 2e ?//通過該自陷指令方式進入KiSystemService,‘調用’對應的系統服務
? ?}
? ?Else
? ?{
? ? ? Lea edx,[esp +4] //用戶空間中的參數區基地址,記錄在edx中
? ? ? Sysenter //通過sysenter方式進入KiFastCallEntry,‘調用’對應的系統服務
? ?}
? ?Ret 36 //不管是從int 2e方式還是sysenter方式,系統調用都會返回到此條指令處
}

Int 2e的內部實現原理:

該指令是一條自陷指令,執行該條指令后,cpu會自動將當前線程的當前棧切換為本線程的內核棧(棧分
用戶棧、內核棧),保存中斷現場,也即那5個寄存器。然后從該cpu的中斷描述符表(簡稱IDT)中找到
這個2e中斷號對應的函數(也即中斷服務例程,簡稱ISR),jmp 到對應的isr處繼續執行,此時這個ISR
本身就處于內核空間了,當前線程就進入內核空間了

Int 2e指令可以把它理解為intel提供的一個內部函數,它內部所做的工作如下

Int 2e
{
? ?Cli ?//cpu一中斷,立馬自動關中斷
? ?Mov esp, TSS.內核棧地址 //切換為內核棧,TSS中記錄了當前線程的內核棧地址
? ?Push SS
? ?Push esp
? ?Push eflags
? ?Push cs
Push eip ?//這5項工作保存了中斷現場【標志、ip、esp】
Jmp ?IDT[中斷號] ?//跳轉到對應本中斷號的isr
}
?
IDT的整體布局:【異常->空白->5系->硬】(推薦采用7字口訣的方式重點記憶)

異常:前20個表項存放著各個異常的描述符(IDT表不僅可以放中斷描述符,還放置了所有異常的異常處
理描述符,0x00-0x13)

保留:0x14-0x1F,忽略這塊號段

空白:接下來存放一組空閑的保留項(0x20-0x29),供系統和程序員自己分配注冊使用

5系:然后是系統自己注冊的5個預定義的軟中斷向量(軟中斷指手動的INT指令)

? ? ?(0x2A-0x2E ?5個系統預注冊的中斷向量,0x2A:KiGetTickCount, 0x2B:KiCallbaclReturn

0x2C:KiRaiseAssertion, ?0x2D:KiDebugService, ?0x2E:KiSystemService)

硬: ?最后的表項供驅動程序注冊硬件中斷使用和自定義注冊其他軟中斷使用(0x30-0xFF)
......
......
參見《寒江獨釣》一書P93頁注冊鍵盤中斷時,搜索空閑未用表項是從0x20開始,到0x29結束的,就知道
為什么寒江獨釣是在這段范圍內搜索空白表項了(其實我們也完全可以從0x14開始搜索)
......
明白了IDT,就可以看到0x2e號中斷的isr為KiSystemService,顧名思義,這個中斷號專用于提供系統服
務。

在正式分析KiSystemService,前,先看下幾個輔助函數

SaveTrap() ?//這個函數用來保存寄存器現場和其他狀態信息

{
Push 0 ? //LastError
Push ebp
Push ebx
Push esi
Push edi
Push fs ? //此時的fs若是從用戶空間自陷進來的就指著TEB,反之指著kpcr
Push kpcr.ExceptionList
Push kthread.PreviousMode
Sub esp,0x48 //騰給調式寄存器保存用
-----------至此,上面的這些語句連同int 2e中的語句在棧上構造了一個trap幀-----------------

Mov CurTrapFrame,esp ?//當前Trap幀的地址
Mov CurTrapFrame.edx, kthread.TrapFrame //將上次的trap幀地址記錄到edx成員中
Mov kthread.TrapFrame, CurTrapFrame, //修改本線程當前trap幀的地址
Mov kthread.PreviousMode,GetMode(進入內核前的CS) ?//根據CS自動確定上次模式
Mov kpcr.ExceptionList,-1 ?//表示剛進入內核時,尚未安裝seh
Mov fs,kpcr ? //一進入內核就讓fs改指向當前cpu的描述符kpcr,不再指向TEB
If(當前線程處于調試狀態)
? ?保存DR0-DR7到trap幀中
}

FindTableCall() //這個函數用來查表,拷貝參數,調用系統服務
{
Mov edi,eax ?//系統函數號,低12位為索引,第13為表示是哪張系統服務表中的索引
Mov eax, edi.低12位 //eax=真正的服務號
If(edi.第13位=1) ?//if這是shadow SSDT中的系統函數號
{
? ?If(當前線程.服務描述符表!=shadow)
? ? ? 當前線程.服務描述符表=shadow ?//換用另外一張描述符表
}

服務表描述符=當前線程.服務描述符表[edi.第13位]
Mod edi=服務表描述符.base //這個系統服務表的地址
Mov ebx,[edi+eax*4] ?//查表獲得這個函數的地址
Mov ecx=服務表描述符.Number[eax] ?//查表獲得的這個系統函數的參數大小
Mov esi,edx ? //esi=用戶空間中的參數地址
Mov edi,esp ?//esp已經為內核棧的棧頂地址
Rep movsb ?//將所有參數從用戶空間復制到內核空間,相當于N個連續push壓參
Call ?ebx ?//調用對應的系統服務函數
}

......
Struct KSERVICE_TABLE_DESCRIPTOR

{
? ?ULONG* base;//系統服務表的地址
? ?ULONG* CountTable;//該系統服務表中每個函數的歷史調用次數統計表
? ?ULONG limit;//該系統服務表的大小,也即容量
? ?BYTE* ArgSizeTable;//記錄該系統服務表中每個函數參數大小的表 ??
}

2、 通過快速調用指令(Intel的是sysenter,AMD的是syscall)調用系統服務

老式的cpu不支持、不提供sysenter指令,只能由int 2e模擬中斷方式進入內核,調用系統服務,

但是,那種方式有一個明顯的缺點,就是速度慢!(如int 2e內部本身要保存5個寄存器的現場,然后還
要去IDT中查找isr,這個過程消耗的時間太多),因此x86系列從奔騰2代開始為系統調用專門增設了一
條sysenter指令以及相應的寄存器msr。同樣,sysenter指令也可看做intel提供的一個內部函數,它做
的工作如下:

Sysenter()
{
? ?Mov ss,msr_ss
? ?Mov esp,msr_esp //關鍵
? ?Mov cs,msr_cs
? ?Mov eip,msr_eip //關鍵
}
系統在啟動初始化過程中,會將上面四個msr寄存器設為固定的值,其中msr_esp為DPC函數專用堆棧,

Msr_eip則固定為KiFastCallEntry
......
......
KeGetPreviosMode()
{
Return kthread.PreviousMode;
}
這樣:內核API KeGetPreviosMode的返回值就是內核模式了
......

三 windows內核情景分析--內存管理

32位系統中有4GB的虛擬地址空間

每個進程有一個地址空間,共4GB,(具體分為低2GB的用戶地址空間+高2GB的內核地址空間)

各個進程的用戶地址空間不同,屬于各進程專有,內核地址空間部分則幾乎完全相同

虛擬地址如0x11111111, ?看似這8個數字是一個整體,其實是由三部分組成的,是一個三維地址,將這
個32位的值拆開,高10位表示二級頁表號,中間10位表示二級頁表中的頁號,最后12位表示頁內偏移
(2^12=4kb),因此,一個虛擬地址實際上是一個三維地址,指明了本虛擬地址在哪個二級頁表,又在哪
個頁以及頁內偏移是多少 ?這三樣信息!

【虛擬地址 = 二級頁表號.頁號.頁內偏移】:口訣【頁表、頁號、頁偏移】

Cpu訪問物理內存的原理介紹:

如高級語言

DWORD ?g_var; ?//假設這個全局變量被編譯器編譯為0x00000004

g_var=100;?

那么這條賦值語句編譯后對應的匯編語句為:mov DWORD PTR[0x00000004],100

這里0x00000004就是一個虛擬地址,簡稱VA,那么這條mov 指令究竟是如何尋址的呢?

尋址過程為:CPU中的虛擬地址轉換器也即MMU,將虛擬地址0x00000004轉換為物理地址

具體轉換過程為:

根據CR3寄存器中記錄的當前進程頁表的物理地址,找到總頁表也即頁目錄,再根據虛擬地址中的頁表號
,以頁表號為索引,找到總頁表中對應的PDE,再根據PDE,找到對應的二級頁表,再以虛擬地址中的頁
號部分為索引,找到二級頁表中的對應PTE,再根據這個PTE記錄的映射關系,找到這個虛擬頁面對應的
物理頁面,最后加上虛擬地址中的頁內偏移部分,加上這個偏移值,就得出最后的物理地址。具體用下
面的函數可以形象表達尋址轉換過程:

mov DWORD PTR[0x00000004],100 //這條指令的內部原理(沒考慮二級緩沖情況)
{
va=0x00000004;//頁表號=0,頁號=0,頁內偏移=4
? ? ? 總頁表=CR3; ?//本進程的總頁表的物理地址固定保存在cr3寄存器中
? ? ? PDE=總頁表[va.頁表號]; ?//PDE為對應的二級頁表描述符
? ? ? 二級頁表=PDE.PageAddr; ?//得出本二級頁表的地址
? ? ? PTE=二級頁表[va.頁號]; ? //得出到該虛擬地址所在頁面的PTE映射描述符
? ? ? If(PTE空白) ?//PTE為空表示該虛擬頁面尚未建立映射
? ? ? ? ?觸發0x0e號頁面訪問異常(具體為缺頁異常)
? ? ? Else
? ? ? If(PTE.bPresent==false) //PTE的這個字段表示該虛擬頁面當前是否映射到了物理內存
? ? ? ? ?觸發0x0e號頁面訪問異常(具體為缺頁異常)
? ? ? Else
? ? ? If(CR0.wp==1 ?&& ?PTE.Writable==false) //已開啟頁面寫保護功能,就檢查這個頁面是否可寫
? ? ? ? ?觸發0x0e號頁面訪問異常(具體為頁面訪問保護越權異常)
? ? ? Else
? ? ? ? ?物理地址pa =cs.base + PTE.PageAddr + va.頁內偏移 ?//得出對應的物理地址
? ? ? 將得到的pa放到地址總線上,100放在數據總線上,經由FSB->北橋->內存總線->內存條 寫入內存
}
PTE是二級頁表中的表項,記錄了對應虛擬頁面的映射情況,這個PTE實際上可以看做一個描述符。

上面的過程比較簡單,由于每次訪問內存都要先訪問一次PTE獲取該虛擬頁面對應的物理頁面,再訪問物
理頁面讀得對應的數據,因此實際上訪問了兩次物理內存,如果類似于每條這樣的Mov指令都要訪問物理
內存兩次,才能獲得數據,效率就很低。因此,cpu芯片中專門開辟了一個二級緩沖,用來保存那些頻繁
訪問的PTE,這樣,cpu每次去查物理頁面時,就先嘗試在二級緩沖中查找對應的PTE,如果找不到,再才
去訪問內存中的PTE。這樣,效率就比較高,實際上絕大數情況就可以在二級緩沖中一次性找到對應的
PTE。

另外有一個問題需要說明下:va---->pa的轉換過程實際上是va->la->pa,實際上PTE.PageAddr表示的是
相對于cs段的偏移,加上cs段的base基址,就得到了該頁面的la線性地址。

(線性地址=段.基地址 + 段內偏移),但是由于Windows采取了Flat也即所謂的平坦分段機制,使得每
個段的基地址都在0x00000000處,長度為4GB,也即相當于Windows沒有采取分段機制。前面講過,cs是
GDT表中的索引,指向GDT表中的cs段描述符,由于Windows不分段,因此GDT中每個段描述符的基址=0,
長度=4GB,是固定的!這樣一來,由于不分段,線性地址就剛好是物理地址,所以本來是由虛擬地址->
線性地址->物理地址的轉換就可以直接看做虛擬地址->物理地址。

(注:在做SSDT hook、IDT hook時,由于SSDT與IDT這兩張表各自所在的頁面都是只讀的,也即他們的
PTE中標志位標示了該頁面不可寫。因此,一修改SSDT、IDT就會報異常,一個簡單的處理方法是是關閉
CRO中的wp即寫保護位,這樣就可以修改了)

前文說了,每個進程有兩個地址空間,一個用戶地址空間,一個內核地址空間,該地址空間的內核結構
體定義為:

Struct ?MADDRESS_SPACE ?//地址空間描述符
{
? ?MEMORY_AREA* ?MemoryRoot;//本地址空間的已分配區段表(一個AVL樹的根)
? ?VOID* ?LowestAddress;//本地址空間的最低地址(用戶空間是0,內核空間是0x80000000)
? ?EPROCESS* Process;//本地址空間的所屬進程
/*一個表,表中每個元素記錄了本地址空間中各個二級頁表中的PTE個數,一旦某個二級頁表中的PTE個
數減到了0,就自動釋放該二級頁面表本身,體現為稀疏數組特征*/
? ?USHORT* PageTableRefCountTable;?
? ?ULONG PageTableRefCountTableSize;//上面那個表的大小
}
地址空間中所有已分配的區段都記錄在一張表中,這個表不是簡單的數組,而是一個AVL樹,用來提高查
找效率。每個區段的基址都對齊64KB或4KB(指64KB整倍數),各個區段之間可以有空隙,

區段的分布是很零散的!各個區段之間,夾雜的空隙就是尚未分配的虛擬內存。

注:所謂已分配區段,是指已經過VirtualAlloc預訂(reserve)或提交(commit)后的虛擬內存

區段的描述符如下:

Struct ?MEMORY_AREA ? ?//區段描述符
{
? ?Void* StartingAddress; //開始地址,普通區段對齊64KB,其它類型區段對齊4KB
? ?Void* EndAddress;//結尾地址,EndAddress – StartingAddress就是該區段的大小
? ?MEMORY_AREA* ?Parent;//AVL樹中的父節點
? ?MEMORY_AREA* ?LeftChild;//左邊的子節點
? ?MEMORY_AREA* ?RightChild;//右邊的子節點

//常見的區段類型有:普通型區段、視圖型區段、緩沖型區段(后面文件系統中會講到)等

? ?ULONG type;//本區段的類型
? ?ULONG protect;//本區段的保護權限,可讀、可寫、可執行的組合
? ?ULONG flags;//當初分配本區段時的分配標志
? ?BOOLEAN DeleteInProgress;//本區段是否標記為了‘已刪除’
? ?ULONG PageOpCount;

? Union

{

? ? Struct //這個Struct專用于視圖型區段

? ? {

? ? ? ?//凡是含有ROS字樣的函數與結構體都表示是ReactOS與Windows中不同的實現細節
? ? ? ?ROS_SECTION_OBJECT* ?section;?
? ? ? ?ULONG ViewOffest;//指本視圖型區段在所在Segment內部的偏移
? ? ? ?MM_SECTION_SEGMENT* Segment;//所屬Segment
? ? ? ?BOOLEAN WriteCopyView;//本視圖區段是不是一個寫復制區段 ??
? ? }SectionData;

LIST_ENTRY ?RegionListHead;//本區段內部的所有Region區塊,放在一個鏈表中

}Data;

}//end
淺談區段類型:

MEMORY_AREA_VIRTUAL_MEMORY://普通型區段,由VirtuAlloc應用層用戶分配的區段都是普通區段

MEMORY_AREA_SECTION_VIEW://視圖型區段,用于文件映射、共享內存

MEMORY_AREA_CACHE_SEGMENT://用于文件緩沖的區段(一個簇大小)

MEMORY_AREA_PAGED_POOL://內核分頁池中的區段

MEMORY_AREA_KERNEL_STACK://用于內核棧中的區段

MEMORY_AREA_PEB_OR_TEB://用于PEB、TEB的區段

MEMORY_AREA_MDL_MAPPING://內核中專用于建立MDL映射的區段

MEMORY_AREA_CONTINUOUS_MEMORY://對應的物理頁面也連續的區段

MEMORY_AREA_IO_MAPPING://內核空間中用于映射外設內存(如顯存)的區段

MEMORY_AREA_SHARED_DATA://內核空間中用于與用戶空間共享的區段

Struct ?MM_REGION ?//區塊描述符

{
? ?ULONG type;//指本區塊的分配類型(預定型分配、提交型分配),又叫映射狀態(已映射、尚未映
射)

? ?ULONG protect;//本區塊的訪問保護權限,可讀、可寫、可執行的組合

? ?ULONG length;//區塊長度,對齊頁面大小(4KB)

? ?LIST_ENTRY RegionListEntry;//用來掛入所在區段的區塊鏈表

}
內存以區段為分配單位,一個區段內部,又按分配類型、保護屬性劃分區塊。一個區塊包含一到多個內
存頁面,分配類型相同并且保護權限相同的區域組成一個個的區塊,因此,稱為“同屬性區塊”。一個
區段內部,相鄰區塊之間的屬性肯定是不相同的(分配類型或保護權限不同),若兩個相鄰區塊的屬性
相同了,會自動合并成一個新的區塊。

......
......
......
創建好了section對象后,就可以讓任意進程拿去映射了,不過映射是以視圖為單位進行的

【section. ?segment. ?視圖. 頁面】,這是這四者之間的層級關系,請牢記

NtMapViewOfSection(hSection, ViewOffset, ViewSize, ? AllocType, protect, ?hProcess, void**
BaseAddr )
{
? ?PreviousMode=ExGetPreviousMode();
? ?If(PreviousMode == UserMode)
? ? ? ?參數檢查;
? ?ViewOffset=Align4kb(ViewOffset);
? ?ViewSize=Align4kb(ViewSize);
? ?ObReferenceObjectByHandle(hSection---> Section);//獲得對應的對象
? ?MmMapViewOfSection(Section, ViewOffset,ViewSize, AllocType, protect, hProcess, void**
BaseAddr );
}
MmMapViewOfSection(Section, ViewOffset, ViewSize , AllocType, protect, hProcess, void**
BaseAddr )

{

? ?AddressSpace=process->VadRoot;//那個進程的用戶地址空間

//若是PE文件的section,則加載映射文件中的每個segment,注意此時的ViewOffset和ViewSize參數不

起作用,將自動把每個完整segment當做一個視圖來映射。

? ?If(Section->AllocationAttribute ?& ?SEC_IMAGE)

? ?{

? ? ? ULONG i;

? ? ? ULONG NrSegments;

? ? ? ULONG_PTR ImageBase;

? ? ? ULONG ImageSize;

? ? ? PMM_IMAGE_SECTION_OBJECT ImageSectionObject;

? ? ? PMM_SECTION_SEGMENT SectionSegments;

? ? ? ImageSectionObject = Section->ImageSection;

? ? ? SectionSegments = ImageSectionObject->Segments;//節數組

? ? ? NrSegments = ImageSectionObject->NrSegments;//該pe文件中的節數

? ? ? ImageBase = (ULONG_PTR)*BaseAddress;

? ? ? if (ImageBase == 0)

? ? ? ? ?ImageBase = ImageSectionObject->ImageBase;

? ? ? ImageSize = 0;

? ? ? //下面的循環遍歷該pe文件中所有需要加載的節,計算所有節的大小總和

? ? ? for (i = 0; i < NrSegments; i++)

? ? ? {

? ? ? ? ?if (!(SectionSegments[i].Characteristics & IMAGE_SCN_TYPE_NOLOAD))//所需要加載這個

? ? ? ? ?{

? ? ? ? ? ? ULONG_PTR MaxExtent;

? ? ? ? ? ? //該節的rva+該節的對齊4KB長度

? ? ? ? ? ? MaxExtent=SectionSegments[i].VirtualAddress + SectionSegments[i].Length;

? ? ? ? ? ? ImageSize = max(ImageSize, MaxExtent);

? ? ? ? ?}

? ? ? }

? ? ? ImageSectionObject->ImageSize = ImageSize;

? ? ? //如果該pe文件期望加載的區域中有任何一個地方被占用了,重定位,dll文件一般都會重定位

? ? ? if (MmLocateMemoryAreaByRegion(AddressSpace, ImageBase,PAGE_ROUND_UP(ImageSize)))

? ? ? {

? ? ? ? ?if ((*BaseAddress) != NULL)//如果用戶的要求是必須加載到預期地址處,返回失敗!

? ? ? ? ? ? return(STATUS_UNSUCCESSFUL);

? ? ? ? ?ImageBase = MmFindGap(AddressSpace, ImageSize, PAGE_SIZE, FALSE);//重定位,找空閑

? ? ? }

? ? ? //一次性加載映射該pe文件中的所有節

? ? ? for (i = 0; i < NrSegments; i++)

? ? ? {

? ? //注意pe文件中有的節是不用加載的

? ? ? ? ?if (!(SectionSegments[i].Characteristics & IMAGE_SCN_TYPE_NOLOAD))

? ? ? ? ?{

? ? ? ? ? ? PVOID SBaseAddress = ?((char*)ImageBase + (SectionSegments[i].VirtualAddress);

? ? ? ? ? ? //把該節整體當做一個view進行映射。由此可見,pe文件中的每個節也是一個視圖型區段

? ? ? ? ? ? MmMapViewOfSegment(AddressSpace,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?Section,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?&SectionSegments[i],//該視圖所在的第一個節

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?&SBaseAddress,//該節的預期映射地址

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?SectionSegments[i].Length,//ViewSize=整個節的長度

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?SectionSegments[i].Protection,

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?0,//ViewOffset=0

? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?0);

? ? ? ? ?}

? ? ? }

? ? ? *BaseAddress = (PVOID)ImageBase;//返回該PE文件實際加載映射的地址

? ?}

? ?Else//普通數據文件和頁文件的section,都只有一個segment

? ?{

? ? ? MmMapViewOfSegment(AddressSpace, ?section, section->segmen, ? ViewOffset, ViewSize , ?
AllocType & MEM_TOPDOWN, ?protect, ?hProcess, ?void** BaseAddr);

? ?}

}
MmMapViewOfSegment(AddressSpace, ?section, segmen, ? ViewOffset, ViewSize , ?AllocType, ?
protect, ?hProcess, ?void** BaseAddr);

{

? ?MEMORY_AREA* ?Area;

? ?MmCreateMemoryArea(AddressSpace, ?視圖型區段, ?BaseAddr,ViewSize, ?protect, AllocType,
&Area);

? ?//記錄本視圖區段映射的是哪個section的哪個segment中的哪個位置

? ?Area->Data.SectionData.Section=Section;

? ?Area->Data.SectionData.Segment=segment;

? ?Area->Data.SectionData.ViewOffset=ViewOffset;

? ?Area->Data.SectionData..WriteCopyView=FALSE;//視圖型區段默認是不‘寫復制’的

? ?初始化Area區段中的區塊鏈表;//初始時,整個區段中就一個區塊

}
......
? ?if (NumberOfBytes > PageSize-BlockHeadSize)//超出一個頁面

? ?{

? ? ? ? //大于一個頁面大小的分配特殊處理;

? ? ? ? Retun ?MiAllocatePoolPages(PoolType, NumberOfBytes);;

? ?}

? ?For(遍歷空閑塊表)

? ?{

If(找到了一個合乎大小的空閑塊)

{

? ? ? ? ? 從空閑塊鏈表中摘下一個合乎大小的塊;

? ? ? ? ? 前后合并相鄰塊;//是在一個頁面內分隔、合并

? ? ? ? ? Return 找到的塊地址;

? ? ? }?

? ?}

? ?//如果已有的空閑鏈表找不到這樣一個大小的塊

? ?在池中分配一個新的頁面;

? ?在新頁面中把前面的部分割出來,后面剩余的部分掛入池中的空閑塊表中;

? ?Return 分得的塊地址 ?

}
內核中池的分配原理同用戶空間中的堆一樣,都是先用VirtuAllocate去分配一個頁面,然后在這個頁面

尋找空閑塊,分給用戶。每個池塊的塊頭含有一些附加信息,如這個池塊的大小,池類型,該池塊的tag
標記等信息。用戶空間中的malloc,new堆塊分配函數,都是調用HeapAlloc API函數從堆管理器維護的N
個虛擬頁面中分出一些零散的塊出來,每個堆塊的塊頭、塊尾也含有一些附加信息,如堆塊大小,防止
堆塊溢出的cookie等信息。堆管理器則在底層調用VirtualAlloc ?API分配,增長虛擬頁面,提供底層服
務。

......

四 Windows內核情景分析---內核對象

寫過Windows應用程序的朋友都常常聽說“內核對象”、“句柄”等術語卻無從得知他們的內核實現到底是怎樣的, 本篇文章就揭開這些技術的神秘面紗。

常見的內核對象有:

Job、Directory(對象目錄中的目錄)、SymbolLink(符號鏈接),Section(內存映射文件)、Port(LPC端口)、IoCompletion(Io完成端口)、File(并非專指磁盤文件)、同步對象(Mutex、Event、Semaphore、Timer)、Key(注冊表中的鍵)、Token(用戶/組令牌)、Process、Thread、Pipe、Mailslot、Debug(調試端口)等

內核對象就是一個數據結構,就是一個struct結構體,各種不同類型的對象有不同的定義,本片文章不專門介紹各個具體對象類型的結構體定義,只講述一些公共的對象管理機制。

至于各個具體對象類型的結構體定義,后文逐步會有詳細介紹。
?

所有內核對象都遵循統一的使用模式:
第一步:先創建對象;
第二步:打開對象,得到句柄(可與第一步合并在一起,表示創建時就打開)
第三步:通過API訪問對象;
第四步,關閉句柄,遞減引用計數;
第五步:句柄全部關完并且引用計數降到0后,銷毀對象。

句柄就是用來維系對象的把柄,就好比N名纖夫各拿一條繩,同拉一艘船。每打開一次對象就可拿到一個句柄,表示拿到該對象的一次訪問權。

內核對象是全局的,各個進程都可以訪問,比如兩個進程想要共享某塊內存來進行通信,就可以約定一個對象名,然后一個進程可以用CreatFileMapping(”SectionName”)創建一個section,而另一個進程可以用OpenFileMapping(”SectionName”)打開這個section,這樣這個section就被兩個進程共享了。

(注意:本篇說的都是內核對象的句柄。像什么hWnd、hDC、hFont、hModule、hHeap、hHook等等其他句柄,并不是指內核對象,因為這些句柄值不是指向進程句柄表中的索引,而是另外一種機制)


各個對象的結構體雖然不同,但有一些通用信息記錄在對象頭中,看下面的結構體定義

typedef struct _OBJECT_HEADER
{
? ? LONG PointerCount;//引用計數
? ? union
? ? {
? ? ? ? LONG HandleCount;//本對象的打開句柄計數(每個句柄本身也占用一個對象引用計數)
? ? ? ? volatile VOID* NextToFree;//下一個要延遲刪除的對象
? ? };

? ? OBJECT_TYPE* Type;//本對象的類型,類型本身也是一種內核對象,因此我習慣叫‘類型對象’
? ? UCHAR NameInfoOffset;//對象名的偏移(無名對象沒有Name)
? ? UCHAR HandleInfoOffset;//各進程的打開句柄統計信息數組
? ? UCHAR QuotaInfoOffset;//對象本身實際占用內存配額(當不等于該類對象的默認大小時要用到這個)
? ? UCHAR Flags;//對象的一些屬性標志

? ? union
? ? {
? ? ? ? OBJECT_CREATE_INFORMATION* ObjectCreateInfo;//來源于創建對象時的OBJECT_ATTRIBUTES
? ? ? ? PVOID QuotaBlockCharged;
? ? };

? ? PSECURITY_DESCRIPTOR SecurityDescriptor;//安全描述符(對象的擁有者、ACL等信息)
? ? QUAD Body;//通用對象頭后面緊跟著真正的結構體(這個字段是后面真正結構體中的第一個成員)

} OBJECT_HEADER, *POBJECT_HEADER;

如上,Body就是對象體中的第一個字段,頭部后面緊跟具體對象類型的結構體定義

typedef struct _OBJECT_HEADER_NAME_INFO
{
? ? POBJECT_DIRECTORY Directory;//對象目錄中的父目錄(不一定是文件系統中的目錄)
? ? UNICODE_STRING Name;//相對于Directory的路徑或者全路徑
ULONG QueryReferences;//對象名查詢操作計數

} OBJECT_HEADER_NAME_INFO, *POBJECT_HEADER_NAME_INFO;

typedef struct _OBJECT_HEADER_CREATOR_INFO
{
? ? LIST_ENTRY TypeList;//用來掛入所屬‘對象類型’中的鏈表(也即類型對象內部的對象鏈表)
PVOID CreatorUniqueProcess;//表示本對象是由哪個進程創建的

} OBJECT_HEADER_CREATOR_INFO, *POBJECT_HEADER_CREATOR_INFO;

對象頭中記錄了NameInfo、HandleInfo、QuotaInfo、CreatorInfo這4種可選信息。如果這4種可選信息全部都有的話,整個對象的布局從低地址到高地址的內存布局為:

QuotaInfo-> HandleInfo->NameInfo->CreatorInfo->對象頭->對象體;這4種可選信息的相對位置倒不重要,但是必須記住,他們都是在對象頭中的上方(也即對象頭上面的低地址端)。以下為了方便,不妨叫做“對象頭中的可選信息”、“頭部中的可選信息”。

于是有宏定義:

//由對象體的地址得到對象頭的地址

#define OBJECT_TO_OBJECT_HEADER(pBody) ? ?CONTAINING(pBody,OBJECT_HEADER,Body)

//得到對象的名字

#define OBJECT_HEADER_TO_NAME_INFO(h)

? ?h->NameInfoOffset?(h - h->NameInfoOffset):NULL

//得到對象的創建者信息

#define OBJECT_HEADER_TO_CREATOR_INFO(h)

h->Flags & OB_FLAG_CREATOR_INFO?h-sizeof(OBJECT_HEADER_CREATOR_INFO):NULL

所有有名字的對象都會進入內核中的‘對象目錄’中,對象目錄就是一棵樹。內核中有一個全局指針變量ObpRootDirectoryObject,就指向對象目錄樹的根節點,根節點是一個根目錄。

對象目錄的作用就是用來將對象路徑解析為對象地址。給定一個對象路徑,就可以直接在對象目錄中找到對應的對象。就好比給定一個文件的全路徑,一定能從磁盤的根目錄中向下一直搜索找到對應的文件。

如某個設備對象的對象名(全路徑)是”\Device\MyCdo”,那么從根目錄到這個對象的路徑中:

Device是根目錄中的子目錄,MyDevice則是Device目錄中的子節點。

對象有了名字,應用程序就可以直接調用CreateFile打開這個對象,獲得句柄,沒有名字的對象無法記錄到對象目錄中,應用層看不到,只能由內核自己使用。

內核中各種類型的對象在對象目錄中的位置:

目錄對象:最常見,就是對象目錄中的目錄節點(可以作為葉節點)?

普通對象:只能作為葉節點

符號鏈接對象:只能作為葉節點

注意文件對象和注冊表中的鍵對象看似有文件名、鍵名,但此名非對象名。因此,文件對象與鍵對象是無名的,無法進入對象目錄中

根目錄也是一種目錄對象,符號鏈接對象可以鏈接到對象目錄中的任何節點,包括又鏈向另一個符號鏈接對象。

對象目錄中,每個目錄節點下面的子節點可以是

1、 普通對象節點

2、 子目錄

3、 符號鏈接

該目錄中的所有子節點對象都保存在該目錄內部的目錄項列表中。不過,這個列表不是一個簡單的數組,而是一個開式hash表,用來方便查找。根據該目錄中各個子對象名的hash值,將對應的子對象掛入對應的hash鏈表中,用hash方式存儲這些子對象以提高查找效率

目錄本身也是一種內核對象,其類型就叫“目錄類型”,現在就可以看一下這種對象的結構體定義:

typedef struct _OBJECT_DIRECTORY
{
? ? struct _OBJECT_DIRECTORY_ENTRY* ?HashBuckets[37];//37條hash鏈
? ? EX_PUSH_LOCK Lock;
? ? struct _DEVICE_MAP *DeviceMap;
? ? …
} OBJECT_DIRECTORY, *POBJECT_DIRECTORY;

如上,目錄對象中的所有子對象按hash值分門別類的安放在該目錄內部不同的hash鏈中

其中每個目錄項的結構體定義為:

typedef struct _OBJECT_DIRECTORY_ENTRY
{
? ? struct _OBJECT_DIRECTORY_ENTRY * ChainLink;//下一個目錄項(即下一個子節點)
? ? PVOID Object;//對象體的地址
? ? ULONG HashValue;//所在hash鏈
} OBJECT_DIRECTORY_ENTRY, *POBJECT_DIRECTORY_ENTRY;

看到沒,每個目錄項記錄了指向的對象的地址,同時間接記錄了對象名信息

下面這個函數用來在指定的目錄中查找指定名稱的子對象

VOID*

ObpLookupEntryDirectory(IN POBJECT_DIRECTORY Directory,

? ? ? ? ? ? ? ? ? ? ? ? IN PUNICODE_STRING Name,

? ? ? ? ? ? ? ? ? ? ? ? IN ULONG Attributes,

? ? ? ? ? ? ? ? ? ? ? ? IN POBP_LOOKUP_CONTEXT Context)

{

? ? BOOLEAN CaseInsensitive = FALSE;
? ? PVOID FoundObject = NULL;?

? ? //表示對象名是否嚴格大小寫匹配查找

? ? if (Attributes & OBJ_CASE_INSENSITIVE) CaseInsensitive = TRUE;

HashValue=CalcHash(Name->Buffer);//計算對象名的hash值

? ? HashIndex = HashValue % 37;//獲得對應的hash鏈索引

? ? //記錄本次是在那條hash中查找

? ? Context->HashValue = HashValue;

? ? Context->HashIndex = (USHORT)HashIndex;

? ? if (!Context->DirectoryLocked)

? ? ? ? ObpAcquireDirectoryLockShared(Directory, Context);//鎖定目錄,以便在其中進行查找操作

? ??
? ? //遍歷對應hash鏈中的所有對象

? ? AllocatedEntry = &Directory->HashBuckets[HashIndex];

? ? LookupBucket = AllocatedEntry;

? ? while ((CurrentEntry = *AllocatedEntry))

? ? {

? ? ? ? if (CurrentEntry->HashValue == HashValue)

? ? ? ? {
? ? ? ? ? ? ObjectHeader = OBJECT_TO_OBJECT_HEADER(CurrentEntry->Object);

? ? ? ? ? ? HeaderNameInfo = OBJECT_HEADER_TO_NAME_INFO(ObjectHeader);

? ? ? ? ? ? if ((Name->Length == HeaderNameInfo->Name.Length) &&

? ? ? ? ? ? ? ? (RtlEqualUnicodeString(Name, &HeaderNameInfo->Name, CaseInsensitive)))

? ? ? ? ? ? {
? ? ? ? ? ? ? ? break;//找到對應的子對象
? ? ? ? ? ? }

? ? ? ? }

? ? ? ? AllocatedEntry = &CurrentEntry->ChainLink;

? ? }

? ? if (CurrentEntry)//如果找到了子對象
? ? {

? ? ? ? if (AllocatedEntry != LookupBucket)

? ? ? ? ? ? 將找到的子對象掛入鏈表的開頭,方便下次再次查找同一對象時直接找到;

? ? ? ? FoundObject = CurrentEntry->Object;

? ? }

? ? if (FoundObject) //如果找到了子對象

? ? {

? ? ? ? ObjectHeader = OBJECT_TO_OBJECT_HEADER(FoundObject);

? ? ? ? ObpReferenceNameInfo(ObjectHeader);//遞增對象名字的引用計數

? ? ? ? ObReferenceObject(FoundObject);//注意遞增了對象本身的引用計數

? ? ? ? if (!Context->DirectoryLocked)

? ? ? ? ? ? ObpReleaseDirectoryLock(Directory, Context); ? ??

? ? }

? ? //檢查本次函數調用前,查找上下文中是否已有一個先前的中間節點對象,若有就釋放

? ? if (Context->Object)

? ? {

? ? ? ? ObjectHeader = OBJECT_TO_OBJECT_HEADER(Context->Object);

? ? ? ? HeaderNameInfo = OBJECT_HEADER_TO_NAME_INFO(ObjectHeader);

? ? ? ? ObpDereferenceNameInfo(HeaderNameInfo);

? ? ? ? ObDereferenceObject(Context->Object);

? ? }

? ? Context->Object = FoundObject;

? ? return FoundObject;//返回找到的子對象

}

如上,hash查找子對象,找不到就返回NULL。

注意由于這個函數是在遍歷路徑的過程中逐節逐節的調用的,所以會臨時查找中間的目錄節點,記錄到Context中。

......

五 windows內核情景分析---進程線程

本篇主要講述進程的啟動過程、線程的調度與切換、進程掛靠

進程的啟動過程:

BOOL CreateProcess

(

? LPCTSTR lpApplicationName, ? ? ? ? ? ? ? ? //

? LPTSTR lpCommandLine, ? ? ? ? ? ? ? ? ? ? ?// command line string

? LPSECURITY_ATTRIBUTES lpProcessAttributes, // SD

? LPSECURITY_ATTRIBUTES lpThreadAttributes, ?// SD

? BOOL bInheritHandles, ? ? ? ? ? ? ? ? ? ? //?

? DWORD dwCreationFlags, ? ? ? ? ? ? ? ? ? ?// creation flags

? LPVOID lpEnvironment, ? ? ? ? ? ? ? ? ? ? // new environment block

? LPCTSTR lpCurrentDirectory, ? ? ? ? ? ? ? // current directory name

? LPSTARTUPINFO lpStartupInfo, ? ? ? ? ? ? ? // startup information

? LPPROCESS_INFORMATION lpProcessInformation // process information

);

這個Win32API在內部最終調用如下:

CreateProcess(…)

{

? ?…

? ?NtCreateProcess(…);//間接調用這個系統服務,先創建進程

? ?NtCreateThread(…);//間接調用這個系統服務,再創建該進程的第一個線程(也即主線程)

? ?…

}

進程的4GB地址空間分兩部分,內核空間+用戶空間

看下面幾個定義:

#define MmSystemRangeStart ?0x80000000 //系統空間的起點

#define MM_USER_PROB_ADDRESS ?MmSystemRangeStart-64kb ?//除去高端的64kb隔離區

#define MM_HIGHEST_USER_ADDRESS ? MmUserProbAddress-1 //實際的用戶空間中最高可訪問地址

#define MM_LOWEST_USER_ADDRESS ?64kb ?//實際的用戶空間中最低可訪問地址

#define KI_USER_SHARED_DATA ?0xffdf0000 ? //內核空間與用戶空間共享的一塊區域

由此可見,用戶地址空間的范圍實際上是從 ?64kb---->0x80000000-64kb 這塊區域。

(訪問NULL指針報異常的原因就是NULL(0)落在了最前面的64kb保留區中)

內核中提供了一個全局結構變量,該結構的類型是KUSER_SHARED_DATA。內核中的那個結構體變量所在的虛擬頁面起始地址為:0xffdf0000,大小為一個頁面大小。這個內核頁面對應的物理內存頁面也映射到了每個進程的用戶地址空間中,而且是固定映在同一處:0x7ffe0000。這樣,用戶空間的程序直接訪問用戶空間中的這個虛擬地址,就相當于直接訪問了內核空間中的那個公共頁面。所以,那個內核頁面稱之為內核空間提供給各個進程的一塊共享之地。(事實上,這個公共頁面非常有用,可以在這個頁面中放置代碼,應用程序直接在r3層運行這些代碼,如在內核中進行IAT hook)
......

六 Windows內核情景分析 --APC

明白了APC大致原理后,現在詳細看一下APC的工作原理。

APC分兩種,用戶APC、內核APC。前者指在用戶空間執行的APC,后者指在內核空間執行的APC。

先看一下內核為支持APC機制提供的一些基礎結構設施。

Typedef struct _KTHREAD

{

? ?…

? ?KAPC_STATE ?ApcState;//表示本線程當前使用的APC狀態(即apc隊列的狀態)

? ?KAPC_STATE ?SavedApcState;//表示保存的原apc狀態,備份用

? ?KAPC_STATE* ApcStatePointer[2];//狀態數組,包含兩個指向APC狀態的指針

? ?UCHAR ApcStateIndex;//0或1,指當前的ApcState在ApcStatePointer數組中的索引位置

? ?UCHAR ApcQueueable;//指本線程的APC隊列是否可插入apc

? ?ULONG KernelApcDisable;//禁用標志

//專用于掛起操作的APC(這個函數在線程一得到調度就重新進入等待態,等待掛起計數減到0)

? ?KAPC SuspendApc;

? ?… ??

}KTHREAD;

?

Typedef struct _KAPC_STATE //APC隊列的狀態描述符

{

? ?LIST_EBTRY ?ApcListHead[2];//每個線程有兩個apc隊列

? ?PKPROCESS Process;//當前線程所在的進程

? ?BOOL KernelApcInProgress;//指示本線程是否當前正在 內核apc

? ?BOOL KernelApcPending;//表示內核apc隊列中是否有apc

? ?BOOL UserApcPending;//表示用戶apc隊列中是否apc

}

Typedef enum _KAPC_ENVIRONMENT

{

? ?OriginalApcEnvironment,//0,狀態數組索引

? ?AttachedApcEnvironment;//1,狀態數組索引

? ?CurrentApc Environment;//2,表示使用當前apc狀態

? ?CurrentApc Environment;//3,表示使用插入apc時那時的線程的apc狀態

}

七 Windows內核情景分析---線程同步

基于同步對象的等待、喚醒機制:

一個線程可以等待一個對象或多個對象而進入等待狀態(也叫睡眠狀態),另一個線程可以觸發那個等待對象,喚醒在那個對象上等待的所有線程。

一個線程可以等待一個對象或多個對象,而一個對象也可以同時被N個線程等待。這樣,線程與等待對象之間是多對多的關系。他們之間的等待關系由一個隊列和一個‘等待塊’來控制,等待塊就是線程與等待目標對象之間的紐帶。

WaitForSingleObject可以等待那些“可等待對象”,哪些對象是‘可等待’的呢?進程、線程、作業、文件對象、IO完成端口、可等待定時器、互斥、事件、信號量等,這些都是‘可等待’對象,可用于WaitForSingleObject等函數。

‘可等待’對象又分為‘可直接等待對象’和‘可間接等待對象’

互斥、事件、信號量、進程、線程這些對象由于內部結構中的自第一個字段是DISPATCHER_HEADER結構(可以看成是繼承了DISPATCHER_HEADER),因此是可直接等待的。而文件對象不帶這個結構,但文件對象內部有一個事件對象,因此,文件對象是‘可間接等待對象’。

比如:信號量就是一種可直接等待對象,它的結構如下:

Struct KSEMAPHORE
{
? ?DISPATCHER_HEADER Header;//公共頭
? ?LONG Limit;//最大信號量個數
}

Struct DISPATCHER_HEADER
{

? ?…
? ?LONG SignalState;//信號狀態量(>0表示有信號,<=0表示無信號)
? ?LIST_ENTRY WaitListHead;//等待塊隊列
? ?…
}

WaitForSingleObject內部最終調用下面的系統服務

NTSTATUS
NtWaitForSingleObject(IN HANDLE ObjectHandle,//直接或間接可等待對象的句柄
? ? ? ? ? ? ? ? ? ? ? IN BOOLEAN Alertable,//表示本次等待操作是否可被吵醒(即被強制喚醒)
? ? ? ? ? ? ? ? ? ? ? IN PLARGE_INTEGER TimeOut ?OPTIONAL)//超時
{

? ? PVOID Object, WaitableObject;
? ? KPROCESSOR_MODE PreviousMode = ExGetPreviousMode();
? ? LARGE_INTEGER SafeTimeOut;

? ? NTSTATUS Status;

? ? if ((TimeOut) && (PreviousMode != KernelMode))
? ? {
? ? ? ? _SEH2_TRY
? ? ? ? {
? ? ? ? ? ? SafeTimeOut = ProbeForReadLargeInteger(TimeOut);
? ? ? ? ? ? TimeOut = &SafeTimeOut;
? ? ? ? }

? ? ? ? _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
? ? ? ? {
? ? ? ? ? ? _SEH2_YIELD(return _SEH2_GetExceptionCode());
? ? ? ? }

? ? ? ? _SEH2_END;

? ? }

? ? Status = ObReferenceObjectByHandle(ObjectHandle,SYNCHRONIZE,NULL,PreviousMode,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?&Object,NULL);

? ? if (NT_SUCCESS(Status))
{

? ? //得到那個對象的‘可直接等待對象’DefaultObject
? ? ? ? WaitableObject = OBJECT_TO_OBJECT_HEADER(Object)->Type->DefaultObject;

? ? ? ? if (IsPointerOffset(WaitableObject))//if DefaultObject是個偏移,不是指針
? ? ? ? {
? ? ? ? ? ? //加上偏移值,獲得內部的‘可直接等待對象’
? ? ? ? ? ? WaitableObject = (PVOID)((ULONG_PTR)Object + (ULONG_PTR)WaitableObject);
? ? ? ? }

? ? ? ? _SEH2_TRY
? ? ? ? {
? ? ? ? ? ? Status = KeWaitForSingleObject(WaitableObject,//這個函數只能等待‘直接等待對象’
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ?UserRequest,PreviousMode,Alertable,TimeOut);
? ? ? ? }

? ? ? ? _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
? ? ? ? {
? ? ? ? ? ? Status = _SEH2_GetExceptionCode();
? ? ? ? }

? ? ? ? _SEH2_END;
? ? ? ? ObDereferenceObject(Object);
? ? }
? ? return Status;
}

#define IsPointerOffset(Ptr) ((LONG)(Ptr) >= 0)
如上,每個對象的對象類型都有一個默認的可直接等待對象,要么直接指向對象,要么是個偏移值。

如果是個偏移值,那么DefaultObject值的最高位為0,否則為1。
......

八 windows內核情景分析--窗口消息

消息與鉤子

眾所周知,Windows系統是消息驅動的,現在我們就來看Windows的消息機制.

早期的Windows的窗口圖形機制是在用戶空間實現的,后來為了提高圖形處理效率,將這部分移入內核空間,在Win32k.sys模塊中實現。這個模塊作為一個擴展的內核模塊,提高了一個擴展額系統服務表,專用于窗口圖形操作,相應的,這個模塊中添加了一個擴展系統調用服務表Shadow SSDT,以及一個擴展的系統調用服務表描述符表:KeServiceDescriptorTableShadow.(系統中 不僅有兩張SSDT,還有兩張系統服務表描述符表)。當一個線程首次調用這個模塊中的系統服務函數時,這個線程就自然變成了GUI線程。GUI線程結構的ServiceTable指向的就是這個shadow描述符表。

指向這個表的系統服務號的bit12位(也即第13位)為1,如0x1XXX表示使用的是shadow服務表。

每個線程創建時都是普通線程,但是只要那個線程在運行的過程中發起了一次對win32k.sys模塊中的系統調用,就會轉變成GUI線程,下面的函數就是這個用途。

NTSTATUS ?PsConvertToGuiThread(VOID)
{
? ? ULONG_PTR NewStack;
? ? PVOID OldStack;
? ? PETHREAD Thread = PsGetCurrentThread();
? ? PEPROCESS Process = PsGetCurrentProcess();
? ? NTSTATUS Status;

? ? if (KeGetPreviousMode() == KernelMode) return STATUS_INVALID_PARAMETER;

? ? ASSERT(PspW32ProcessCallout != NULL);//確保win32k.sys模塊已加載到內存

? ? if (Thread->Tcb.ServiceTable != KeServiceDescriptorTable)
? ? ? ? return STATUS_ALREADY_WIN32;//表示先前已經轉換為GUI線程了

? ? if (!Thread->Tcb.LargeStack)//if 尚未換成大內核棧
? ? {

? ? ? ? NewStack = (ULONG_PTR)MmCreateKernelStack(TRUE, 0);//分配一個64KB的大內核棧
? ? ? ? //更為大內核棧
? ? ? ? OldStack = KeSwitchKernelStack(NewStack, (NewStack - KERNEL_STACK_SIZE));
? ? ? ? MmDeleteKernelStack(OldStack, FALSE);//銷毀原來的普通內核棧
? ? }

? ? if (!Process->Win32Process)//if 尚未分配W32PROCESS結構(也即if是該進程中的第一個GUI線程)
? ? ? ? Status = PspW32ProcessCallout(Process, TRUE);//分配Win32Process結構(表示GUI進程)

Thread->Tcb.ServiceTable = KeServiceDescriptorTableShadow;//關鍵。更改描述符表

//為當前線程分配一個W32THREAD結構
? ? Status = PspW32ThreadCallout(Thread, PsW32ThreadCalloutInitialize);
? ? if (!NT_SUCCESS(Status)) Thread->Tcb.ServiceTable = KeServiceDescriptorTable;//改為原來的
? ? return Status;

}
如上,每個線程在轉換為GUI線程時,必須換用64KB的大內核棧,因為普通的內核棧只有12KB大小,不能支持開銷大的圖形任務。然后分配一個W32PROCESS結構,將進程轉換為GUI進程,然后分配W32THREAD結構,更改系統服務表描述符表。上面的PspW32ProcessCallout和PspW32ThreadCallout函數都是回調函數,分別指向win32k.sys模塊中的Win32kProcessCallback、Win32kThreadCallback函數。
......
?

總結

以上是生活随笔為你收集整理的windows 内核情景分析的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

一级α片免费看 | 一二三区视频在线 | 欧美日韩中文在线视频 | 四虎在线观看 | 日韩精品一区在线播放 | 91桃色在线观看视频 | 狠狠地操| www.xxxx欧美 | 国产爽妇网 | 久久久免费看视频 | 日韩视频免费 | 伊人视频| 一区二区三区精品久久久 | 亚洲黄网站 | 久久激情五月丁香伊人 | 天天操天天插 | 精品福利在线观看 | 丰满少妇在线观看 | 天天久久综合 | 久久精品伊人 | 在线观看你懂的网址 | 成人久久久久久久久久 | 日韩在线不卡视频 | 久操视频在线观看 | 国产91在线观 | 成人久久毛片 | 国产中文字幕视频 | 国产美女精品视频免费观看 | 欧美三人交 | 天天综合在线观看 | 在线免费观看不卡av | 国产成人精品午夜在线播放 | 午夜精品福利一区二区 | 中文字幕 在线看 | 在线观看一区 | 午夜色大片在线观看 | 日韩黄色免费看 | 国产成人精品网站 | 成人av高清在线观看 | 成人午夜电影在线 | 亚洲一区二区黄色 | 国产一二三四在线视频 | 九色在线 | 色综合久久综合网 | 91av99| 久草免费福利在线观看 | www.91成人 | 97精品欧美91久久久久久 | 国产高清在线 | 国产精品中文字幕在线播放 | 91成年人在线观看 | 久久久www成人免费毛片麻豆 | 97视频在线免费 | 日日干天天 | 99精品欧美一区二区 | 日本h视频在线观看 | 99久久99| 综合久久精品 | 香蕉网在线 | 狠狠干婷婷 | 国产精品青青 | 日韩免费不卡av | 91色在线观看视频 | 色婷五月| 草免费视频 | 亚洲国产精品999 | 久久免费视频7 | 国产精品一区二区三区99 | 亚洲激情视频在线观看 | 国产自产高清不卡 | av成年人电影 | 在线小视频国产 | 欧美一区二区三区在线看 | 日本精品视频网站 | 夜夜澡人模人人添人人看 | 丁香久久 | 亚洲精品国产高清 | 国产精品欧美久久久久三级 | 亚洲91av| www.午夜色.com | 国产成人av在线 | 久久精品2 | 国产婷婷色 | 国产一区在线视频 | 国产精品女教师 | 国产视频在线播放 | 婷婷免费视频 | 外国av网| 日韩欧美精品一区二区三区经典 | 国产探花视频在线播放 | 日韩色视频在线观看 | 国产不卡在线观看视频 | 久久午夜视频 | 超碰在线免费福利 | 久久精精品视频 | 免费看成人 | 日本电影久久 | 天天射天天色天天干 | 日日草夜夜操 | 亚洲午夜av电影 | 久久国产精品99精国产 | 在线性视频日韩欧美 | 成人91在线观看 | 成人黄视频 | 日韩欧美一区二区在线播放 | 中日韩男男gay无套 日韩精品一区二区三区高清免费 | 亚洲人成人在线 | 天天射射天天 | 在线免费中文字幕 | 在线不卡的av | 国产免费av一区二区三区 | 五月天久久狠狠 | 国产片免费在线观看视频 | 日韩在线观看小视频 | 久久久久国产成人免费精品免费 | 日本精品在线看 | 国产日韩欧美在线免费观看 | 一区二区av | 五月综合激情婷婷 | 激情av资源 | 国产精品女同一区二区三区久久夜 | 午夜在线免费观看视频 | 久久精品亚洲国产 | 91黄视频在线 | 友田真希x88av | 日韩城人在线 | 亚洲成av人片 | 亚洲视频播放 | 午夜精品一区二区三区在线观看 | 不卡的av片| 久久精品一区八戒影视 | 久久精品国产一区二区电影 | 日韩一区精品 | 久久精品国产久精国产 | 中文字幕在线精品 | 国产精品久久久亚洲 | 青青河边草观看完整版高清 | 亚洲影视资源 | 在线免费观看黄色小说 | 美女久久一区 | av一级二级 | 亚洲国产精品成人女人久久 | 成年人在线观看免费视频 | 成人在线免费看 | 国产精品毛片久久久久久 | 国产一级视屏 | 国产精品欧美日韩在线观看 | 麻豆视频大全 | 成人av影视观看 | 亚洲精品www.| 免费手机黄色网址 | 亚洲永久精品在线观看 | 激情久久久| 在线 国产 日韩 | 免费观看www小视频的软件 | 91视频免费网站 | 99久久精品久久久久久动态片 | 伊人久久在线观看 | 国语久久 | 日日爱网站 | 久久免费在线观看 | 国产高清免费在线观看 | 日韩在线观看的 | 日韩免费一区二区在线观看 | 狠狠色噜噜狠狠狠狠 | 中国美女一级看片 | 色就是色综合 | 亚洲日韩精品欧美一区二区 | 日韩v在线91成人自拍 | 日韩欧美在线国产 | 97国产在线播放 | 欧美一级性生活视频 | 黄色av免费电影 | 成人av在线影院 | 成年人免费在线观看网站 | 精品夜夜嗨av一区二区三区 | 欧美日韩视频在线观看一区二区 | 亚洲国产美女久久久久 | 国产成人久久 | 亚洲精品久久久蜜桃直播 | av再线观看 | 精品国产乱码久久久久久天美 | 九九视频在线观看视频6 | 久久影视一区 | 91精品国产高清自在线观看 | 国产精品21区 | 欧美巨乳波霸 | 97在线视频免费 | 久草网站| 国产一级在线视频 | 天天草天天插 | 正在播放 国产精品 | 五月花激情 | 伊人五月天综合 | av九九| 欧美一级日韩免费不卡 | 中文字幕日韩免费视频 | 成人免费视频在线观看 | 久久综合婷婷国产二区高清 | 狠狠的操狠狠的干 | 成人动漫一区二区三区 | 精品国产乱子伦一区二区 | 久久激情五月婷婷 | 日本精品一区二区 | 亚洲精品视频二区 | 国产一线二线三线性视频 | 天天操天天怕 | 一级黄色在线免费观看 | 亚洲国产中文字幕在线视频综合 | a级国产片| 国产一区二区免费看 | 在线观看日韩免费视频 | 国产剧情一区二区 | 亚洲播放一区 | 婷婷新五月 | 蜜臀久久99精品久久久无需会员 | 免费h视频 | 99久久精品久久久久久动态片 | 人人澡人摸人人添学生av | 97热久久免费频精品99 | 欧美久久久久久久久久 | 国产高清精品在线观看 | 特级黄色视频毛片 | 九九久久国产精品 | 午夜精品一区二区三区在线 | 狠狠色丁香婷婷综合久小说久 | 午夜精品久久久久99热app | 久久a国产 | 亚洲黄色av| 欧美精品久久久久久久亚洲调教 | 97精品国产97久久久久久春色 | 亚洲精品久久久蜜桃 | 丁香婷婷色综合亚洲电影 | 一区久久久 | 久久影视一区二区 | 1024手机看片国产 | 久久无码精品一区二区三区 | 天天曰天天曰 | 免费在线一区二区三区 | www.久久久.cum | 韩国在线视频一区 | 992tv又爽又黄的免费视频 | 国产精品乱码久久久久 | 黄色看片 | 亚洲国产色一区 | 国产中年夫妇高潮精品视频 | 亚洲视频精品在线 | 成人午夜片av在线看 | 成年人免费观看在线视频 | 国模吧一区 | www.亚洲视频 | 超碰免费97 | 欧美在线观看视频一区二区三区 | 99精品视频在线看 | 欧美一级电影在线观看 | 最近中文字幕完整视频高清1 | 国产精品99久久久久久人免费 | 欧美analxxxx| www视频在线免费观看 | 日本久草电影 | 日日色综合 | 在线免费观看国产视频 | 久草在线免费在线观看 | 欧美在线观看视频免费 | 亚洲综合视频在线 | 五月激情丁香图片 | 国产福利资源 | 91中文字幕网 | 在线视频app | 国产精彩视频一区二区 | 国产黄网在线 | 在线观看黄色国产 | 激情婷婷在线观看 | 色视频网站在线 | 久久免费看a级毛毛片 | 成人免费视频网址 | 99免在线观看免费视频高清 | 丁香高清视频在线看看 | 福利一区二区在线 | 在线日韩中文 | 欧美日韩中文在线 | 久草资源在线 | 91精品啪在线观看国产81旧版 | 国产五码一区 | 国产美女主播精品一区二区三区 | 亚洲视频1区2区 | 2024av| 免费观看日韩 | 国产人成精品一区二区三 | 热re99久久精品国产66热 | a天堂在线看 | 激情五月看片 | 久久99热精品 | 五月视频 | 欧美日韩首页 | 狠狠狠狠狠狠干 | 夜夜高潮夜夜爽国产伦精品 | 亚洲乱码久久 | 成年人在线播放视频 | 天天色播| 99热在线看 | 五月花婷婷 | 一区二精品 | 超碰97在线人人 | 日韩网站在线免费观看 | 伊甸园av在线 | 色婷婷丁香 | 五月开心网 | 伊人中文在线 | 久久久久久久久久久久久影院 | 亚洲精品视频中文字幕 | 亚洲欧美国产精品18p | 亚洲精品国 | 91精品视频免费在线观看 | 亚洲日本va午夜在线电影 | 欧美一二在线 | 久久精品国产亚洲aⅴ | 日韩三级视频在线观看 | 深夜成人av| 在线观看av国产 | 美女久久视频 | 久久婷婷丁香 | 91精品国产91久久久久福利 | 婷婷狠狠操 | 国产免费三级在线观看 | 制服丝袜在线91 | 色噜噜日韩精品一区二区三区视频 | 久久99欧美| 99精品热视频只有精品10 | 中文字幕久久久精品 | 综合天堂av久久久久久久 | 热99久久精品 | 狠狠狠狠狠狠狠 | 日韩a级免费视频 | 午夜精品久久久久99热app | 国语麻豆| 国产乱码精品一区二区蜜臀 | 久久久久久久久久久高潮一区二区 | 国产一级不卡毛片 | 99视频在线精品国自产拍免费观看 | 欧美日韩国语 | 日韩欧美一区二区三区在线观看 | 日韩资源在线观看 | 中文字幕在线观看免费高清电影 | 国产成人精品一区在线 | 亚洲91中文字幕无线码三区 | 亚洲激情久久 | 人人澡澡人人 | 黄色特级毛片 | 精品女同一区二区三区在线观看 | 日本3级在线观看 | 最近中文字幕完整视频高清1 | av网站大全免费 | 伊人开心激情 | 综合网五月天 | 精品中文字幕在线观看 | 99视频精品在线 | 在线观看你懂的网址 | 久久综合加勒比 | 亚洲日本va午夜在线电影 | 国产 欧美 在线 | 国产精品久久久久永久免费 | 久久蜜臀av | 中日韩免费视频 | 免费黄av| 日韩视频中文字幕在线观看 | 天天爱天天操 | 久久不射电影院 | 91av在线免费播放 | 国产精品久久精品国产 | 激情综合啪 | 久久久久久网址 | 亚洲女在线| 久久国精品 | 欧美精品久久久久久久久免 | 久久免费在线 | 亚洲激情久久 | 亚洲最大成人网4388xx | 久久久久久久国产精品影院 | 狠狠色丁香久久综合网 | 国产第一页在线播放 | 三级在线播放视频 | 日韩在线视频播放 | 最近中文字幕免费大全 | 午夜精品一区二区三区免费视频 | 国产一级性生活视频 | 一级免费观看 | 在线观看av免费观看 | av成人免费 | 日本少妇久久久 | 婷婷激情五月 | 91香蕉视频 mp4| 97在线看片 | 国产最新在线观看 | 欧美夫妻生活视频 | 欧美日韩精品久久久 | 国产永久免费 | 久草久草在线观看 | 激情文学综合丁香 | 黄色在线观看污 | 久青草视频| 一区三区视频在线观看 | 亚洲精品视频免费在线观看 | 综合天天网| 亚洲91av| 久久久久国产精品厨房 | 狠狠色丁香婷婷综合久小说久 | 麻豆国产精品视频 | 成人av影视在线 | 中国一级片在线观看 | 欧美日韩国产一区二区在线观看 | 国产精品美女久久久久久久 | 国产一级三级 | 国产精品久久久777 成人手机在线视频 | 久久曰视频 | 91天天视频| 成人在线一区二区三区 | 久久综合久久综合久久综合 | 欧美久草视频 | 九九热在线观看 | 亚洲一级免费观看 | 中文伊人 | 日日操日日 | 亚洲少妇久久 | 久久久久久久久久久久久9999 | 久久精品波多野结衣 | 久久精品视频免费观看 | 亚洲精品男女 | 亚洲另类视频在线观看 | 美国三级黄色大片 | 成人一区二区三区中文字幕 | 欧美在线一二 | 伊人开心激情 | 国产剧情一区二区 | 国产韩国日本高清视频 | 成人一级免费电影 | 人人干人人干人人干 | 天天干天天干天天干天天干天天干天天干 | 午夜免费电影院 | 91精品视频免费观看 | 天天人人综合 | 97在线视频免费看 | 国产97在线看 | 久久综合久色欧美综合狠狠 | 中文字幕在线免费97 | 欧美精品二区 | 国产精品九九九九九 | 最新高清无码专区 | 国产精品成久久久久 | 2019免费中文字幕 | 在线观看免费国产小视频 | 六月丁香婷婷网 | 欧美a级在线免费观看 | 91av视频| 亚洲天天草 | 久久精品欧美 | 91中文字幕 | 久久久久久久久久久久久9999 | 午夜精品一区二区三区免费 | 国产精品一码二码三码在线 | 99国产在线观看 | 免费视频一级片 | 99超碰在线观看 | www.夜色321.com| 久久国产精品免费看 | 日韩av一区二区在线影视 | 久久日韩精品 | 色综合久久久 | 欧美精品在线视频 | 最近最新中文字幕视频 | 精品色综合 | 久久爱资源网 | 毛片99 | www.天天操.com| 天天干天天操天天操 | 日韩av电影国产 | 99免费在线观看视频 | 九色porny真实丨国产18 | 亚洲va韩国va欧美va精四季 | 亚洲精品视频大全 | 久久久久久久久久久免费视频 | 午夜av日韩| 亚洲精品1区2区3区 超碰成人网 | 九九交易行官网 | 欧美性天天 | 久久专区| 亚洲精品字幕在线 | 五月天激情视频在线观看 | 亚洲第二色 | 天天天色| 亚洲综合视频在线 | 欧美一级电影在线观看 | 久久99九九99精品 | 天天操天天摸天天射 | 久久久精品 | 日本在线观看黄色 | 久操中文字幕在线观看 | 人人藻人人澡人人爽 | 久久久免费精品视频 | 麻豆91精品91久久久 | 在线精品一区二区 | 色天天综合久久久久综合片 | 麻豆视频网址 | 涩涩成人在线 | a视频免费看 | 亚洲91中文字幕无线码三区 | 国产一级在线免费观看 | 国产精品网站一区二区三区 | 97免费| www.成人sex | 国产精品手机视频 | 久久99亚洲热视 | 日韩视频www | 日韩在线视| 亚洲日韩欧美一区二区在线 | 91av在线播放视频 | 亚洲精品在线一区二区三区 | 成人h电影在线观看 | 日韩午夜高清 | 伊色综合久久之综合久久 | 日韩电影中文,亚洲精品乱码 | 97视频播放| 国产精品久久久久久久久婷婷 | 日韩精品一区二区三区在线播放 | 成人免费在线视频观看 | 天堂av影院 | 久久午夜网 | 黄色小视频在线观看免费 | 在线观看91视频 | 色丁香色婷婷 | 久久观看最新视频 | 99精品久久久久 | 国产老太婆免费交性大片 | 99视频久| 亚洲视频网站在线观看 | 综合网天天色 | 超碰在线9 | 中文字幕av免费 | 欧美精品三级在线观看 | 中国一级片在线观看 | 亚洲aⅴ免费在线观看 | 男女啪啪免费网站 | 99视屏| 久久久久欠精品国产毛片国产毛生 | 久久精品一二三区白丝高潮 | 日本黄色免费播放 | 在线观看理论 | 免费看三级黄色片 | 色偷偷av男人天堂 | 中文字幕首页 | 久久精品视频免费观看 | 日韩国产精品毛片 | 中文字幕制服丝袜av久久 | 久久久久五月天 | www.狠狠操.com | 亚洲国产精品激情在线观看 | 永久免费av在线播放 | 中文一二区 | 中文字幕久久精品 | 亚洲黄色一级视频 | 免费美女久久99 | 人人看人人草 | 久久国内精品视频 | 国产中文字幕在线播放 | 最新婷婷色 | 久久精品99国产 | 精品一区二区免费在线观看 | 欧美最猛性xxx | 天天干,天天射,天天操,天天摸 | www黄免费| 久久在线免费观看视频 | 日批视频在线播放 | 久久久天堂 | 日韩综合视频在线观看 | 日韩欧美网站 | www.久久91| 久久国产一区二区三区 | 在线观看一级视频 | 成人h在线观看 | 欧美在线1区 | 中文字幕资源在线 | 中文在线免费看视频 | 麻豆免费观看视频 | 91麻豆精品国产自产在线 | 日韩一区正在播放 | 99re久久资源最新地址 | 国产自在线观看 | 久久超级碰视频 | 综合成人在线 | 久久y | 欧美伦理一区二区三区 | 午夜精品久久久久久中宇69 | 91精品秘密在线观看 | 国产精品久久久网站 | 国产精品美乳一区二区免费 | 精品国产一二三四区 | 欧美另类交在线观看 | 久久综合久久鬼 | 日本久久久影视 | 久久成人国产精品免费软件 | 91九色视频观看 | 亚洲国产日韩一区 | 99这里精品 | 国产精品情侣视频 | 国产涩涩在线观看 | 国产精品福利在线播放 | 黄色av一级片 | 99精品视频99 | 日韩影片在线观看 | 不卡国产视频 | 成年人免费看片网站 | 青草视频在线看 | 国产精品视频在线看 | 久久精品1区 | 中文字幕色网站 | 国产一区二区在线播放 | 91亚洲精品久久久蜜桃借种 | 五月婷影院 | 国产视 | 久99久精品 | 亚洲免费观看视频 | 新版资源中文在线观看 | www色 | 国产在线 一区二区三区 | 狠狠干网址 | 久久久久久久久久久影院 | 亚洲午夜精品在线观看 | 992tv人人网tv亚洲精品 | 婷婷深爱五月 | 中文字幕在线视频免费播放 | 91成品视频 | 在线观看一级片 | 色综合久久久 | 视频成人免费 | 国产精品一区二区久久精品爱微奶 | 国产成人在线观看 | 国产成人精品久久亚洲高清不卡 | 婷婷在线观看视频 | 91麻豆精品国产午夜天堂 | 最新国产精品视频 | 成人禁用看黄a在线 | 色婷婷av一区 | 少妇高潮冒白浆 | 亚洲精品视频网 | 日韩精品久久中文字幕 | 成年人黄色av| 国产中出在线观看 | 久久视频精品 | www.色就是色 | 欧美资源 | 国产涩涩网站 | 黄色小网站在线观看 | 日韩高清免费在线 | 欧美一区二区精美视频 | 激情综合中文娱乐网 | 成人午夜影院 | 欧美在线观看视频 | 91探花国产综合在线精品 | 丁香在线视频 | 久久精品国产免费看久久精品 | 国产裸体永久免费视频网站 | 亚洲黄色成人网 | 久久九九久久 | 免费观看www小视频的软件 | 久久精品xxx | 欧美日韩高清不卡 | 黄色高清视频在线观看 | 亚洲午夜av久久乱码 | 丁香色综合 | 亚洲午夜精品电影 | 亚洲三级网站 | 亚洲视频h | 有码视频在线观看 | 欧美日韩国语 | 在线观看91精品国产网站 | 麻豆91在线| 999在线视频 | 免费在线激情电影 | 久久国色夜色精品国产 | 婷婷伊人综合亚洲综合网 | 一级性视频 | 成人h视频在线 | 在线观看视频黄 | 特级a毛片| 91精选在线观看 | 国产精品一区二区久久精品爱涩 | www久草 | 91人人爱 | 91在线精品观看 | 国产成人久久精品亚洲 | 久久精美视频 | 国产黄网在线 | 黄色免费网站 | 狂野欧美激情性xxxx欧美 | 国产精品久久久久婷婷二区次 | 伊人久在线 | 欧美福利网站 | 久久99国产精品二区护士 | 日韩大陆欧美高清视频区 | 99久久精品免费看国产 | 综合网成人 | 日韩精品中文字幕av | 国产亚洲精品久久19p | 97在线视频网站 | 波多野结衣一区二区 | 在线看片a| 激情偷乱人伦小说视频在线观看 | 日韩有码中文字幕在线 | 精品久久久久久久久久久院品网 | 天天天天干 | av亚洲产国偷v产偷v自拍小说 | 久久伦理影院 | 日韩中文字幕在线 | 91九色视频网站 | 97爱 | 97av在线视频 | 久久久2o19精品 | 日韩在线免费视频 | 97av视频在线| 18国产精品福利片久久婷 | 国产精品久久久久永久免费看 | 一本一本久久a久久 | 黄色高清视频在线观看 | 性色xxxxhd| 在线观看的黄色 | 国产a高清 | 91精品爽啪蜜夜国产在线播放 | 精品久久久久久亚洲综合网 | 亚洲成人xxx | 天天操天天是 | 91视频91色 | 国产成人亚洲在线观看 | 91看片淫黄大片一级在线观看 | 欧美性春潮 | 婷婷九九| 国产精品99久久久久 | 久久久91精品国产一区二区精品 | 97在线观 | 成在人线av | 中文字幕色播 | 国产视频1 | 天天操夜夜曰 | 国产精品永久免费 | 九九热在线视频免费观看 | 热九九精品 | 免费网站在线 | 日韩欧美一区二区不卡 | 国产精品久久久免费看 | 国内视频在线观看 | 九九久久精品 | 玖玖在线看| 久久这里只有精品9 | 国产精品美女免费看 | 97色视频在线 | 日韩午夜在线 | 麻豆视频成人 | 日本特黄一级片 | 亚洲精品免费在线视频 | 超碰在线公开免费 | 日韩在线色| 人人爽影院 | 丁香婷婷深情五月亚洲 | 国产在线视频导航 | 亚洲精品视频中文字幕 | www五月天com | 深爱婷婷久久综合 | 亚洲高清在线精品 | 国产黄av | 天天色天天操综合网 | 国产免费高清视频 | av资源在线看 | 久久久五月天 | 免费看一级特黄a大片 | 最新成人在线 | 美女性爽视频国产免费app | 亚洲精品在线网站 | 在线亚洲人成电影网站色www | 黄色三几片| 四虎在线视频免费观看 | 91精品综合在线观看 | 99精彩视频在线观看免费 | 麻豆av电影 | 久草视频播放 | 丁香色婷 | 天天操夜夜看 | 国产视频在线播放 | 天天综合久久综合 | 亚洲国产天堂av | 久久成人在线 | 精品国产成人在线 | 欧美最猛性xxxxx免费 | 成人综合免费 | 色多多视频在线 | 日本韩国中文字幕 | a级片在线播放 | 亚洲干视频在线观看 | 久久综合操 | 欧美久久成人 | 亚洲精品国产麻豆 | 精品久久久久_ | 欧美精品在线一区 | 在线观看视频黄色 | 在线一区二区三区 | av电影 一区二区 | 中文字幕丝袜美腿 | 国产精品久久久久久久av电影 | 六月婷婷久香在线视频 | 中文字幕在线看 | 人人澡人摸人人添学生av | 1区2区视频 | 在线成人性视频 | 2021av在线| 欧美日韩精品网站 | 亚洲国产经典视频 | 热久在线| 四川bbb搡bbb爽爽视频 | 91精品国自产在线观看欧美 | 精品在线一区二区三区 | 韩日精品视频 | 国产一区在线看 | 欧美黑人xxxx猛性大交 | 男女日麻批 | 久久精品免视看 | 日韩最新中文字幕 | 欧美日韩不卡一区二区三区 | 综合久久久久久久 | 成年人精品 | www五月天 | 国产免费视频在线 | 欧美日韩三级在线观看 | 日韩在线不卡 | 久久精品99国产国产精 | 国产91免费在线观看 | 久久免费a| 国产中文字幕三区 | 深夜视频久久 | 国产精品久久久久久a | 国产亚洲视频在线免费观看 | 日日干天天爽 | 亚洲精品视频在线免费播放 | 国产精品永久久久久久久www | 日本精品一二区 | 久精品视频在线观看 | 久久久久久久久久久高潮一区二区 | 久久在线免费观看 | 91av视频在线免费观看 | 久久超碰在线 | 黄色软件在线观看视频 | 免费观看v片在线观看 | 国产小视频在线免费观看 | 午夜国产福利在线 | 国产小视频免费在线观看 | www久久99 | 五月激情在线 | 一 级 黄 色 片免费看的 | 91av免费在线观看 | 国产一区二区不卡在线 | 超碰在线1 | 国产精品网红直播 | 伊人中文字幕在线 | 樱空桃av | 992tv成人免费看片 | 中文字幕在线观看日本 | 91人人射 | 亚洲成人一区 | 久久超碰在线 | av在线免费观看不卡 | 香蕉97视频观看在线观看 | 日韩在线观看视频网站 | 欧美精品v国产精品 | 这里只有精品视频在线观看 | 久久综合久久综合久久综合 | 五月天婷婷在线观看视频 | 成人黄色电影在线播放 | 天天干天天干天天干天天干天天干天天干 | 欧美成人91| www.五月天 | 波多野结衣一区二区 | 91伊人久久大香线蕉蜜芽人口 | 久久五月天综合 | 日本婷婷色 | av中文资源在线 | 成人一级在线 | 国产精品美女久久久久久久 | 久久综合欧美精品亚洲一区 | 成人看片 | 在线免费三级 | 欧美综合在线观看 | 欧美精品乱码久久久久久按摩 | 成人毛片在线视频 | 69av久久| 久久人人爽人人爽 | 国内精品美女在线观看 | 久久成人精品视频 | 蜜桃视频成人在线观看 | 亚洲每日更新 | 欧洲激情在线 | 在线免费试看 | 91精品一区二区三区久久久久久 | 麻豆系列在线观看 | 中文字幕av在线不卡 | 91丨九色丨国产在线 | 日黄网站 | 久久久 精品 | 国产精品ssss在线亚洲 | 亚洲精品午夜视频 | 国产大片免费久久 | 久久综合婷婷综合 | 色婷婷中文 | 日韩精品一区二区三区三炮视频 | 国产精品手机在线观看 | 麻豆精品视频在线观看免费 | 久久97精品| 国产精品欧美久久久久无广告 | 五月天激情视频 | 99这里只有精品99 | 亚洲国内在线 | 狠狠综合 | 一本一本久久a久久精品综合妖精 | 亚洲 欧美 国产 va在线影院 | av中文在线观看 | 色综合天天视频在线观看 | 日韩精品免费在线视频 | 国产日本高清 | 久久精品精品电影网 | 久久精品aaa| 最新日本中文字幕 | 久久人人爽人人人人片 | 亚洲视屏一区 | 亚洲精品视频免费在线观看 | 亚洲资源网| 蜜桃视频日韩 | 亚洲国产成人在线观看 | 婷婷丁香六月天 | 国产精品9999久久久久仙踪林 | 狠狠色香婷婷久久亚洲精品 | 欧美色图亚洲图片 | 中文字幕高清在线 | 亚洲天堂网视频 | 亚洲欧洲精品一区二区精品久久久 | 在线高清av | 国产91区| 久艹在线免费观看 | 中文字幕黄网 | 国产精品免费大片视频 | 国产黄色视| 爱爱av网站 | 久久成人综合视频 | 午夜精品av在线 | 成人精品在线 | 色视频网站在线 | 国产欧美精品一区二区三区 | 日韩免费观看一区二区三区 | 国产亚洲在线观看 | 狠狠色噜噜狠狠狠狠 | 久久网页| 美女国产精品 | 日韩一区二区三区免费电影 | 波多野结衣在线观看一区二区三区 | 免费av成人在线 | 日日麻批40分钟视频免费观看 | 色婷婷www | 人人澡超碰碰97碰碰碰软件 | av资源网在线播放 | av成人在线播放 | 成人久久亚洲 | 久久久久国产免费免费 | 97电影院在线观看 | 色婷婷国产在线 | 永久免费毛片在线观看 | 亚洲国产婷婷 | 8x成人免费视频 | 日日草视频| 国产精品一区二区三区在线 | 精品免费久久久久 | 日日干夜夜爱 | 日日爽天天爽 | 久久久精品亚洲 | 久久国产热| 在线a视频 | 91精品国产自产在线观看永久 | av一级网站 | 91色亚洲| 国产成人久久77777精品 | 色久天| 少妇18xxxx性xxxx片 | 激情视频一区二区三区 | 日本久久综合视频 | 在线v片免费观看视频 | 蜜臀av性久久久久蜜臀aⅴ四虎 | 欧美在线你懂的 | 视频国产在线 | 国产精品久久久久久超碰 | 日韩欧美在线免费观看 | 成人av资源站 | 国产视频一区在线免费观看 | 亚洲精品美女 | 热久久在线视频 |