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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

磁盘分区表恢复原理

發(fā)布時間:2023/12/18 编程问答 53 豆豆
生活随笔 收集整理的這篇文章主要介紹了 磁盘分区表恢复原理 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

purpleroc · 2016/01/13 19:36

Author: [email?protected]

Email: [email?protected]

0x00 引子


實在是詞窮想不到要怎么寫題目了,就把vs中的工程名當題目吧。

這篇文,主要講講MBR、DBR、NTFS、FAT32結構等等諸如此類的東西,以及在數據恢復中,我們可以從現有的被破壞了的磁盤中獲取到哪些有利于我們進行數據恢復的信息。

不知道是最近沒休息好還是其他原因,總覺得靜不下心、集中不了注意力,也不知道從什么時候開始,瀏覽網頁只需要幾秒鐘,查找資料也從來不會耐心看完文章,總是一翻到底,用最快的速度去搜索、定位自己要找的內容。但通常來說,網上有的與你問題相同的解決方案并不多。而人們寫文章往往不是奔著主題去的,而是和寫論文、寫書一樣,先把種種后面要用到的概念堆砌起來,然后再來慢慢的說解決方案。當然,其實這樣也挺好的,但作為一個目的驅動者,我更喜歡看需要用到的時候再講的內容。

于是,這篇文就這么來寫吧,免得你看完NTFS和FAT32文件系統(tǒng)就不想看下去了。

0x01 背景


事情是這樣的,這幾天在測試TrueCrypt解密的時候,碰到這么一種情況:

用TrueCrypt做整盤加密系統(tǒng)時,TrueCrypt會重寫磁盤的MBR區(qū)域,將原本的MBR加密保存到其他位置,啟動過程中通過了TrueCrypt的密碼驗證后再在內存中恢復原先的MBR并引導進入系統(tǒng)

這么做沒問題,用我的解密程序解完后也能夠正常掛載并訪問,可另一種情況來了:

如果加密的磁盤中,并不是只有一個c盤,而還有其他分區(qū)

這時,我的解密工具便不能直接掛載了,雙擊出現:

圖1:

Winhex打開看了看,解密后的數據都是正確的,也看到了DBR:

圖2:

可還是打不開,為啥呢?猜測是分區(qū)表被破壞了。那怎么修復呢?瞬間想到了大一那會兒幫別人修電腦,搞壞分區(qū)表,花了一晚上找數據(而且還沒找全)的黑歷史,畢竟那會兒不知道用重建分區(qū)表的功能,也不理解原理,也就那次之后,呆圖書館看了蠻多數據恢復的書籍,去了解原理。

于是對解密后的文件做了個鏡像,用數據恢復軟件Diskgenuis打開,搜索并重建分區(qū)表,得到結果如下:

圖3:

發(fā)現他很神奇的把兩個分區(qū)找回來了,如果點保存,再去winhex中看,就能當磁盤來分析了。

所以,我的目的是想知道他是怎么恢復分區(qū)表的!

想自己寫個程序,能修復鏡像文件中損壞了的分區(qū)表,并且能夠當做vhd文件形式被win7以上的系統(tǒng)直接加載,并且在加載完之后,恢復到原有形態(tài)。

于是,憋了一周,邊寫邊找資料,終于把程序寫完來寫這文章了。

要恢復分區(qū)表,首先得從損壞了分區(qū)表的磁盤里找出分區(qū)信息,再用這些信息來生成分區(qū)表。

那問題是,磁盤中會有哪些信息呢?

0x02 尋址方式(CHS/LBA)


在用正常磁盤做講解前先來了解下磁盤的兩種尋址方式,一種是CHS(cylinder head sector)尋址方式、一種是LBA(Logical Block Addressing)邏輯塊尋址方式。其中CHS(尋址方式)在分區(qū)表中使用24個bit位(前10位表示cylinder、中間8位表示head、后6位表示sector),其最大尋址空間為8GB,后因為其滿足不了要求了,人們給他擴充到28bit,但尋址空間也只有128G,面對現有的動輒上TB的硬盤還是無能為力。LBA是用來取代CHS的,LBA是一個整數,常見有32bit和64bit,32bit最大尋址空間達到2TB。

不管CHS(尋址方式)也好,還是LBA(尋址方式)也好。磁盤存儲尋址都需要通過cylinder、head、sector這三個變量來實現。

從上面我們了解到的信息就是,有兩種尋址方式,而且可以相互轉換,再然后呢,歸根到底其實就是用的CHS方式。

這一塊的詳細的介紹以及轉換方式我就不說了,有興趣的可以百度百度,這里也提供一個鏈接:

blog.csdn.net/haiross/art…

0x03 主引導記錄MBR


接著,來看看正常的磁盤中的MBR,所謂MBR即Main Boot Record 主引導記錄區(qū),位于整個硬盤的0磁道0柱面1扇區(qū)(也可以用LBA描述成0扇區(qū))。總共占512字節(jié),通常也就是1個扇區(qū)的大小。其重要作用就是負責從BIOS手中接過引導權,再去找可引導的分區(qū),并將權限交給可引導的DBR(Dos Boot Record),完成系統(tǒng)啟動。

MBR雖然占了一個扇區(qū),但其Boot_Code部分只占了446個字節(jié)。其余64個字節(jié)為DPT(Disk Partition Table硬盤分區(qū)表),對就是我們要恢復的東西,最后2個字節(jié)就是傳說中的標志位55AA了。于是,他的結構體大體如下:

#!cpp typedef struct MBR_T {UCHAR boot_code[446];PartTableRecord partition[4];UCHAR sign[2]; }MBR; 復制代碼

對照著結構體,我們看看winhex中的截圖:

圖4:

黃色區(qū)域就是boot_code所占用的446個字節(jié),紅色部分就是DPT,藍色就是標志位了。

0x04 磁盤分區(qū)表DPT


既然找到了DPT,那肯定是要分析清楚,它是干嘛用的,里面都有些什么信息呢?

直接用winhex的模板看看先:

圖5:

桌面太小,截圖不完全,但也大體知道了里面會有些什么信息,順便翻出結構體如下:

#!cpp typedef struct PartTableRecord_t {BYTE byIsBoot; //引導分區(qū) 1B 1B 80(引導分區(qū)),00(非引導分區(qū))BYTE byStartHead; //起始磁頭 1B 2BBYTE byStartSector; //起始扇區(qū) 1B 3B BYTE byStartCylinder; //起始柱面 1B 4BBYTE byPartType; //分區(qū)類型 1B 5B 07(NTFS),05(擴展分區(qū)),0B(FAT32)BYTE byEndHead; //結束磁頭 1B 6BBYTE byEndSector; //結束扇區(qū) 1B 7BBYTE byEndCylinder; //結束柱面 1B 8BDWORD dwStartSector; //開始扇區(qū) 4B 12B DWORD dwTotalSector; //分區(qū)扇區(qū)數 4B 16B 最大2T Byte } PartTableRecord; 復制代碼

恩,每個分區(qū)表占用16個字節(jié),而MBR中只留了64個字節(jié),這也是為什么一塊硬盤最多只能創(chuàng)建4個主分區(qū)的原因了。多了放不下。那,為啥我們可以看到比四個還多的分區(qū)呢?因為擴展分區(qū)里面可以創(chuàng)建邏輯分區(qū),這里個數不定,只要你盤符夠,想創(chuàng)建多少就創(chuàng)建多少。

從結構體后的注釋語句也可以知道16字節(jié)中每一位分別代表什么含義。這里需要注意的是,表示分區(qū)位置和大小的地方有兩個,我們可以通過起始磁頭、扇區(qū)、柱面和結束磁頭、扇區(qū)、柱面來得到分區(qū)位置和大小,也可以直接通過LBA模式記錄的開始扇區(qū)和分區(qū)扇區(qū)數來獲取到分區(qū)的位置和大小。

那,問題來了,他們哪個是有用的?

在第二節(jié)中尋址方式里講過,CHS能記錄的最大的分區(qū)是8.4GB,超過這個大小,就無力了。那這時候32位的LBA自然就派上用場了。

我做了個實驗,把所有分區(qū)表中的CHS記錄全部清零,再用winhex加載,還是能夠正常識別。于是,我決定了,后面所有尋址方式均以LBA方式來說。

再回到winhex中看分區(qū)信息,來對照DPT一一理解。

圖6:

上圖可以看出,這塊硬盤總共有5個分區(qū)。其中主分區(qū)三個,擴展分區(qū)1個,邏輯分區(qū)2個(邏輯分區(qū)是在擴展分區(qū)里面的)。也就是MBR中64個字節(jié)除了主分區(qū)就是擴展分區(qū)。

根據DPT中的起始扇區(qū)以及扇區(qū)大小,就可以得到上圖中每個分區(qū)的大小、1st sector(起始扇區(qū))了。主分區(qū)好說,我們對照著DPT中的其實位置和大小都能看得出,那擴展分區(qū)是怎么個形態(tài)呢?

我們直接看第四個DPT的信息:007A300B05FE3F1880D0020000600300,忽略掉CHS部分,并對照結構體來看,它告訴了我們這些信息:

#!bash byIsBoot = 0x00 // 非引導分區(qū) byPartType = 0x05 // 擴展分區(qū) dwStartSector = 0x0002D080 //起始扇區(qū) 184448 dwTotalSector = 0x00036000 //分區(qū)大小 221184 復制代碼

這里可以看到,擴展分區(qū)的起始位置其實也就是三個主分區(qū)的總大小了,再加上自身的分區(qū)大小,就是整個磁盤的大小了。例如我這個磁盤是200MB的,現在大小應該是184448 + 221184 = 405632,注意單位是扇區(qū),所以換算成MB應該是 405632/512 * 1024 * 1024 = 198.0625MB。為什么與文件總大小相比少了呢?因為,在分區(qū)表后面還有一些未被使用的空間。好奇的是,這個擴展分區(qū)中到底放了些什么呢?

在winhex中Ctrl+G輸入扇區(qū)號184448,跟隨過去:

圖7:

還是和剛才一樣,黃色部分為前446字節(jié),這里全為0,因為不需要boot_code,而后64字節(jié)為擴展分區(qū)的分區(qū)表信息。在圖的左右下方分別標示出了現在所在的偏移扇區(qū)位置,以及總扇區(qū)個數和偏移位置(字節(jié)數表示)。還是來看這里的DPT信息吧,有兩個分區(qū)有信息,這次直接用winhex來看:

圖8:

第一個分區(qū)起始扇區(qū)是128,總大小為122880,類型是ntfs;第二個分區(qū)起始扇區(qū)是123008,總大小為94338,類型是擴展分區(qū)。需要注意的是,擴展分區(qū)的起始扇區(qū)是需要加上基地址(擴展分區(qū)偏移扇區(qū)位置)的。也就是說,我們看到的第一個分區(qū),實際起始地址為:184448 + 128 = 184576,與圖6的partition4的起始位置是一樣的,那下一個呢?再來一個擴展分區(qū)的類型是怎么個意思,也還是算一下實際的起始地址:184448 + 123008 = 307456。

再次Ctrl + G跟隨過去:

圖9:

同樣的,再次找到一個DPT信息,這里面只有一項,也就是圖6中的第五個分區(qū)了,也來算一下吧:

起始扇區(qū) = 307456 + 128 = 307584,與圖6中第五個分區(qū)起始位置一致。

從上面的實例中可以得出,整個磁盤大概是這么分布的:

圖10:

再看擴展分區(qū)的鏈接圖示:

圖11:

于是,回到背景里提到的目標,我們要做的就是,根據磁盤中存有的信息,來重建出這么一個分區(qū)表。

自然的,我們需要去知道分區(qū)表指向的內容是什么!

0x05 操作系統(tǒng)引導分區(qū)DBR


在上一節(jié)里面,提到了DPT,也提到了分區(qū)表的結構體,從結構體里我們可以看到偏移5的位置有鍵值byPartType,分區(qū)類型,去找了找資料,這里的取值非常多,常見類型大致如下:

#!bash 00H DOS或WINDOWS不允許使用,視為非法 01H FAT12 04H FAT16小于32MB 05H Extended 06H FAT16大于32MB 07H HPFS/NTFS OBH WINDOWS95 FAT32 OCH WINDOWS95 FAT32 0EH WINDOWS FAT16 0FH WINDOWS95 Extended(大于8G) 82H Linux swap 83H Linux 85H Linux extended 86H NTFS volume set 87H NTFS volume set 復制代碼

在結合上一節(jié)的分區(qū)表,這里主要關注05、07、0B,即擴展分區(qū)、NTFS、FAT32三種。而05的在上一節(jié)介紹過了,那么,我們將目光投向NTFS與FAT32兩種類型。

5.1 NTFS(New Technology File System)

首先,從MBR中看到分區(qū)信息能知道,分區(qū)1是NTFS分區(qū),在winhex中跟隨上一節(jié)中分區(qū)1的起始位置,可以看到如下信息,圖12:

已將每個數據對應的結構和數據都著色了,然后也是時候拿出NTFS文件系統(tǒng)中DBR的數據結構了:

#!cpp typedef struct ntfs_boot_sector_t {BYTE ignored[3]; /* 0x00 Boot strap short or near jump */char system_id[8]; /* 0x03 Name : NTFS */BYTE sector_size[2]; /* 0x0B bytes per logical sector */BYTE sectors_per_cluster; /* 0x0D sectors/cluster */WORD reserved; /* 0x0E reserved sectors = 0 */BYTE fats; /* 0x10 number of FATs = 0 */BYTE dir_entries[2]; /* 0x11 root directory entries = 0 */BYTE sectors[2]; /* 0x13 number of sectors = 0 */BYTE media; /* 0x15 media code (unused) */WORD fat_length; /* 0x16 sectors/FAT = 0 */WORD secs_track; /* 0x18 sectors per track */WORD heads; /* 0x1A number of heads */DWORD hidden; /* 0x1C hidden sectors (unused) */DWORD total_sect; /* 0x20 number of sectors = 0 */BYTE physical_drive; /* 0x24 physical drive number */BYTE unused; /* 0x25 */WORD reserved2; /* 0x26 usually 0x80 */LCN sectors_nbr; /* 0x28 total sectors nbr */QWORD mft_lcn; /* 0x30 Cluster location of mft data.*/QWORD mftmirr_lcn; /* 0x38 Cluster location of copy of mft.*/char clusters_per_mft_record; /* 0x40 */BYTE reserved0[3]; /* zero */char clusters_per_index_record; /* 0x44 clusters per index block */BYTE reserved1[3]; /* zero */LCN volume_serial_number; /* 0x48 Irrelevant (serial number). */DWORD checksum; /* 0x50 Boot sector checksum. */BYTE bootstrap[426]; /* 0x54 Irrelevant (boot up code). */WORD marker; /* 0x1FE */ }ntfs_boot_sector ; 復制代碼

再次想想我們的目的,是要重構分區(qū)表,而且,我們可以看出,分區(qū)表中最重要的就是各分區(qū)的分區(qū)類型、起始位置以及

分區(qū)大小,三個信息了。當然,還有是否為活動分區(qū),但這里我們主要是做數據恢復,所以,暫不考慮其是否作為引導分區(qū)了。是的,我們只要獲取到每一個分區(qū)的分區(qū)類型、起始位置、分區(qū)大小三個信息。

我們來看看這個結構體中能提供給我們一些什么:

  • sectors_nbr:總扇區(qū)數,即分區(qū)大小
  • system_id:文件系統(tǒng)類型,即分區(qū)類型

而在剛才的試驗中,我們也可以看到,分區(qū)表中每個起始位置對應的信息,都是指向DBR(Dos Boot Record操作系統(tǒng)引導分區(qū))結構體的。于是,分區(qū)的起始位置也是可以獲取到的。

于是,我們的思路是從磁盤中去搜索各個DBR,然后獲取到分區(qū)表所需信息,再建立分區(qū)表。

既然聊到了NTFS,我們再來看看上面結構體中,還有哪些信息是我們需要的吧:

  • sector_size:每扇區(qū)字節(jié)數,一般情況下是固定512字節(jié)的
  • sectors_per_cluster:這個也挺重要的,需要引入一個新概念,“簇”。他在后面要用的MFT中起作用。先看看這個鍵值是干嘛用的,它表示的是每簇的扇區(qū)數量。
  • hidden:字面意思是未使用的扇區(qū)數。這里表示的是從分區(qū)表到DBR的扇區(qū)數量。若為主分區(qū),則hidden表示的是扇區(qū)的起始位置。若是擴展分區(qū),則hidden表示的是從擴展分區(qū)表到該分區(qū)的扇區(qū)數。
  • mft_lcn:$MFT(主文件表)的偏移位置,這里的單位是簇,他LBA位置 = 分區(qū)offset + mft_lcn * sectors_per_cluster。以 圖:12 為例,分區(qū)起始位置為128,sectors_per_cluster = 8,mft_lcn = 853,所以LBA位置為:128 + 8 * 853 = 6952。

    為了證實,我們在winhex中跟隨到6952扇區(qū),如下:

    圖13:

    在這里,我們也可以用MFT作為判斷某個搜索到的DBR是否正確的條件。有想更詳細了解MFT的可以去找找資料繼續(xù)看文件管理方式,這里就不展開了,畢竟,我們這次主要目的是恢復分區(qū)表。

  • mftmirr_lcn:每個MFT表所占用的扇區(qū)數量。一般為2.

對我們有用的信息大概也就是上面所提到的了。所以對于恢復NTFS型分區(qū),我們的策略是,全盤搜索,找到NTFS型DBR,之后通過MFT信息來判斷該分區(qū)是否正確。同時對于NTFS還需要提醒的是,其DBR扇區(qū)不僅僅在分區(qū)頭部存在,在分區(qū)的最后一個扇區(qū)中也有一個備份。所以,就算分區(qū)首部的DBR被破壞了,我們也可以通過分區(qū)尾部的DBR來恢復出分區(qū)表。

5.2 FAT32(32位文件分配表)

上一小節(jié)中,我們知道了怎么重建NTFS型分區(qū),接著,來看看FAT32。回到圖6中,看看第三個分區(qū)。winhex跟過去,得到截圖如下,這次就不上色了,太累~~~

圖14:

找了找資料,翻出FAT32的結構體表示:

#!cpp typedef struct fat_boot_sector_t {BYTE ignored[3]; /* 0x00 Boot strap short or near jump */char system_id[8]; /* 0x03 Name - can be used to special casepartition manager volumes */BYTE sector_size[2]; /* 0x0B bytes per logical sector */BYTE sectors_per_cluster; /* 0x0D sectors/cluster */WORD reserved; /* 0x0E reserved sectors */BYTE fats; /* 0x10 number of FATs */BYTE dir_entries[2]; /* 0x11 root directory entries */BYTE sectors[2]; /* 0x13 number of sectors */BYTE media; /* 0x15 media code (unused) */WORD fat_length; /* 0x16 sectors/FAT */WORD secs_track; /* 0x18 sectors per track */WORD heads; /* 0x1A number of heads */DWORD hidden; /* 0x1C hidden sectors (unused) */DWORD total_sect; /* 0x20 number of sectors (if sectors == 0) *//* The following fields are only used by FAT32 */DWORD fat32_length; /* 0x24=36 sectors/FAT */WORD flags; /* 0x28 bit 8: fat mirroring, low 4: active fat */BYTE version[2]; /* 0x2A major, minor filesystem version */DWORD root_cluster; /* 0x2C first cluster in root directory */WORD info_sector; /* 0x30 filesystem info sector */WORD backup_boot; /* 0x32 backup boot sector */BYTE BPB_Reserved[12]; /* 0x34 Unused */BYTE BS_DrvNum; /* 0x40 */BYTE BS_Reserved1; /* 0x41 */BYTE BS_BootSig; /* 0x42 */BYTE BS_VolID[4]; /* 0x43 */BYTE BS_VolLab[11]; /* 0x47 */BYTE BS_FilSysType[8]; /* 0x52=82*//* */BYTE nothing[420]; /* 0x5A */WORD marker; }fat_boot_sector; 復制代碼

前面是BIOS Parameter Block結構體的一些信息,是公用的,后面的才是單純的FAT32所需要的。用winhex的模板解析一下,得到下面的內容。

圖15:

首先,我們需要的文件類型是可以通過system_id來獲取到的,起始扇區(qū)也是可以通過DBR所在的位置得到的,相對NTFS來說FAT32中少了sectors_nbr鍵值,那我們應該如何去獲取到FAT32的總大小呢?

我采用的辦法是,用搜索到的它的后一個DBR的位置減去當前位置,如果后面沒有分區(qū)了,則用文件總大小減去當前偏移位置。

這樣,我們也能將分區(qū)表需要的三個信息得到。再然后,我們也要來說說這結構體里還有哪些對我們有用的信息。

  • hidden:字面意思是未使用的扇區(qū)數。這里表示的是從分區(qū)表到DBR的扇區(qū)數量。若為主分區(qū),則hidden表示的是扇區(qū)的起始位置。若是擴展分區(qū),則hidden表示的是從擴展分區(qū)表到該分區(qū)的扇區(qū)數。
  • total_sect:本來可以用來表示分區(qū)大小的,可只是個長整形變量,只能正確表示小分區(qū)的大小,對于大分區(qū)無力了。
  • reserved:保留扇區(qū)數,起始,他指的是第一個FAT1表所在的相對偏移位置,例如,在partition3中,起始地址是82048扇區(qū),reserved扇區(qū)數量是6718扇區(qū),可以得到,FAT1表起始扇區(qū)為82048 + 6718 = 88766。在winhex跟過去,得到信息如下:

    圖16:

  • 途中每一種顏色代表一個目錄表項,綠色為0號FAT項,淡黃色為1號FAT項。通常0號FAT項總為0x0ffffff8。于是,這一特征也可以被我們當做判斷分區(qū)表是否正確的標準。

  • fats:表示的是FAT表的個數,通常為2個,即FAT1和FAT2。

  • fat32_length:每個FAT表的扇區(qū)數。在上面我們定位到了FAT1,那么FAT2就是用FAT1的起始位置加上每個FAT表的大小。

  • info_sector:filesystem_info的起始位置,起始也可以作為FAT32分區(qū)是否正確的判斷標志。

  • backup_boot:FAT32的DBR備份文件存在的位置。上例中,backup_boot = 6,即DBR備份存放在分區(qū)起始扇區(qū)+6的地方。

有關FAT的一些結構和作用:

  • FAT32文件一般有兩份FAT,他們由格式化程序在對分區(qū)進行格式化時創(chuàng)建,FAT1是主,FAT2是備份。
  • FAT1跟在DBR之后,其具體地址由DBR的BPB參數中指定,FAT2跟在FAT1的后面。
  • FAT表由FAT表項構成,我們把FAT表項簡稱FAT項,每個FAT項占用4字節(jié)。
  • 每個FAT項都有一個固定的編號,這個編號從0開始。
  • FAT表項的前兩個FAT項為文件系統(tǒng)保留使用,0號FAT為介質類型,1號FAT為文件系統(tǒng)錯誤標志。
  • 分區(qū)的數據區(qū)中每個簇都會映射到FAT表中的唯一一個FAT項,因為0號FAT和1號FAT被系統(tǒng)占用,用戶的數據從2號FAT開始記錄。
  • 如果某個文件占用很多個簇,則第一個FAT項記錄下一個FAT項的編號(既簇號),如果這個文件結束了,則用“0F FF FF FF”表示。
  • 分區(qū)格式化后,用戶文件以簇為單位存放在數據區(qū)中,一個文件至少占用一個簇。
  • FAT的主要作用是標明分區(qū)存儲的介質以及簇的使用情況。
  • 具體有關FAT的其他信息,還請自行收集資料,因為,有了前面的信息,我們也可以有辦法恢復FAT32分區(qū)對應的分區(qū)表數據了。

    了解了前面的知識之后,我們就可以開始編寫程序來對分區(qū)表進行重建了。

    0x06 FileMapping(文件映射)


    首先需要從磁盤、磁盤鏡像中找到DBR存在的痕跡,就需要對整個磁盤或鏡像文件進行遍歷搜索,由于分區(qū)是線性擴展的,而且DBR所在的位置永遠的是扇區(qū)開頭,并且獨自占有一整個扇區(qū)。于是我們可以遍歷文件或磁盤中的每個扇區(qū)來快速完成搜索。

    對于磁盤或者磁盤鏡像文件,肯定不能是通常的直接fopen、fread讀取整個文件了,因為,你沒那么多內存去讀,這時候就要對文件進行分片讀取,比如,每10M讀一次,如此循環(huán)將其遍歷一次。

    但考慮到吞吐率的問題,這里引用的是FileMapping文件映射的方式,將文件直接映射到內存中進行操作。當文件比較小的時候,我們可以直接全文件映射起來,但通常不太建議這么做。

    我采用的也是上面說的循環(huán)讀取的思路,對大的磁盤文件循環(huán)映射起來,代碼如下:

    #!cpp #define MAPPING_SIZE 67108864 #define BYTE_PER_M 1024*1024// // big_file: 需要映射的文件路徑 // ll_file_size: 需要映射的文件總大小 // int ToMapping(char *big_file, unsigned __int64 ll_file_size) { LCN i = 0;//得到系統(tǒng)分配粒度SYSTEM_INFO sinf;GetSystemInfo(&sinf);DWORD dwAll = sinf.dwAllocationGranularity;printf ("Total %dM.\nSearching...\n",ll_file_size / BYTE_PER_M ;if (ll_file_size <= MAPPING_SIZE) //內存鏡像小于64M時,一次性掛載{Maping_file(big_file, 0, ll_file_size);}else{for (i = 0; i < (ll_file_size / MAPPING_SIZE) ; i++) //否則以64M為一個鏡像映射單位,循環(huán)掛載,直到全部映射完成{ if (i == 0){Maping_file(big_file, (i * (MAPPING_SIZE)) - (i * (MAPPING_SIZE) % dwAll), MAPPING_SIZE);}else {Maping_file(big_file, (i * (MAPPING_SIZE)) - ((i * (MAPPING_SIZE)) % dwAll), MAPPING_SIZE); }}if (ll_file_size > (i * MAPPING_SIZE)){ //最后一次可能并不是64M,需要根據實際大小來映射Maping_file(big_file, i * MAPPING_SIZE - ((i * MAPPING_SIZE) % dwAll), ll_file_size - i * MAPPING_SIZE);}}return 0; } 復制代碼

    ToMapping完成的是將文件分片交給Maping_file函數處理。

    #!cpp int Maping_file(char* big_file, LCN lOffset, long lSize) {char* pPtr_File; //存放指向內存映射文件的首地址HANDLE hFile = CreateFileA(big_file, GENERIC_READ, FILE_SHARE_READ,NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL);if ( hFile == INVALID_HANDLE_VALUE){ //Open the data file.ErrorOut("CreateFile() Error!");}HANDLE hFileMapping = CreateFileMapping(hFile, NULL, //Create the file-mapping object.PAGE_READONLY,0, 0,NULL);if (hFileMapping == INVALID_HANDLE_VALUE){ErrorOut("CreateFileMapping() Error!");}PBYTE pbFile = (PBYTE) MapViewOfFile(hFileMapping, FILE_MAP_READ,lOffset & 0xFFFFFFFF00000000, // Offset highlOffset & 0xFFFFFFFF, // Offset lowlSize); // bytes to mapif (pbFile == INVALID_HANDLE_VALUE){ErrorOut("MapViewOfFile() Error!");}/pPtr_File = (char*)pbFile;ToGetDBR(pPtr_File, lSize, lOffset);//UnmapViewOfFile(pbFile);CloseHandle(hFileMapping);CloseHandle(hFile);return 0; } 復制代碼

    這樣,就能通過CreateFileMapping將大的鏡像文件分塊為64M一小塊映射起來,之后調用ToGetDBR,來對磁盤中殘留的DBR信息進行搜索。

    0x07 搜索DBR


    這里的思路是,遍歷整個磁盤,而后檢測每個扇區(qū),看是否滿足NTFS或FAT32型分區(qū)格式。若滿足,則鍵入到鏈表中。

    #!cpp void ToGetDBR(char* p_file, long l_size, LCN offset) {long i = 0;char *buf = NULL;char *temp = NULL;LCN ll_offset = 0;do {buf = p_file + i * SECTOR_SIZE;if (!test_NTFS((ntfs_boot_sector*)buf, offset + i * SECTOR_SIZE)){ll_offset = offset + (i * SECTOR_SIZE);temp = (char*)malloc(512);memcpy(temp, buf, 512);if (InsertDBRList(g_dbr_list_head, temp, 1, g_n_dbr, ll_offset)) // NTFS type is 1{//printf ("Found NTFS! AT %lld sectors\n", ll_offset / SECTOR_SIZE);}ll_offset = 0;temp = NULL;}if(!test_FAT((fat_boot_sector*)buf, offset + i * SECTOR_SIZE)){ll_offset = offset + (i * SECTOR_SIZE);temp = (char*)malloc(512);memcpy(temp, buf, 512);if (InsertDBRList(g_dbr_list_head, temp, 2, g_n_dbr, ll_offset)) // FAT32 type is 2{//printf("Found FAT! AT %lld sectors\n", ll_offset / SECTOR_SIZE);}ll_offset = 0;temp = NULL;}i++;} while (i * SECTOR_SIZE < l_size); } 復制代碼

    檢測是否為NTFS與FAT32形式的磁盤可根據NTFS及FAT32結構體特征來判斷,這里判斷代碼如下:

    #!cpp int test_NTFS(const ntfs_boot_sector*ntfs_header, LCN l_size) {LCN lba = l_size / SECTOR_SIZE;chs tmp;int verbose = 1;if(ntfs_header->marker!=0xAA55 ||(ntfs_header->reserved)>0 ||ntfs_header->fats>0 ||ntfs_header->dir_entries[0]!=0 || ntfs_header->dir_entries[1]!=0 ||ntfs_header->sectors[0]!=0 || ntfs_header->sectors[1]!=0 ||ntfs_header->fat_length!=0 || (ntfs_header->total_sect)!=0 ||memcmp(ntfs_header->system_id,"NTFS",4)!=0)return 1;switch(ntfs_header->sectors_per_cluster){case 1: case 2: case 4: case 8: case 16: case 32: case 64: case 128:break;default:return 1;}return 0;int test_FAT(const fat_boot_sector* fat_header, LCN l_size) {if(!(fat_header->marker==0xAA55&& (fat_header->ignored[0]==0xeb || fat_header->ignored[0]==0xe9)&& (fat_header->fats==1 || fat_header->fats==2)))return 1; /* Obviously not a FAT */switch(fat_header->sectors_per_cluster){case 1: case 2: case 4: case 8: case 16: case 32: case 64: case 128:break;default:return 1;}switch(fat_header->fats){case 1:break;case 2:break;default:return 1;}return 0; } 復制代碼

    而所用到的DBR鏈表結構體如下:

    #!cpp typedef struct dbr_list_t {char* dbr; // DBR contentint n_type; // DBR type ntfs = 1 fat32 = 2int flag; // Is this DBR a Available?int n_is_org; // Is this DBR a orignal or copy.__int64 ll_offset; // DBR offset.__int64 ll_total_sector; // Partition offset.__int64 ll_start_sector; // Partition Size.dbr_list_t* p_next; // Point to next dbrdbr_list_t(){dbr = NULL;n_type = 0;p_next = 0;ll_offset = 0;n_is_org = 0;flag = 0;ll_total_sector = 0;ll_start_sector = 0;} }dbr_list; 復制代碼

    在搜索過中,對dbr、n_type、ll_offset、p_next四個鍵值進行賦值,得到整個磁盤中可能存在的分區(qū)信息。并將這些搜索到的DBR存放在鏈表中,方便進一步處理。

    0x08 判斷DBR


    在上一小節(jié),我們完成了對整個磁盤中可能存在的分區(qū)信息的搜索。那,搜索到的結果肯定不會全部正確或者可用,為了使重建的分區(qū)表更加可靠,需要對搜索到的分區(qū)信息進行篩選,以及鏈表信息的填充。

    在0x05中談到了NTFS和FAT32類型DBR的一些特性,這里就需要用到這些特性對去進行判斷。

    對于NTFS型的DBR。

    #!cpp if (p_dbr_temp->n_type == 1) // NTFS {p_ntfs_temp = (ntfs_boot_sector*)p_dbr_temp->dbr;if (p_ntfs_temp->sectors_nbr < (g_ll_file_size / SECTOR_SIZE)) // 獲取到的大小不能比總大小還大{flag = 0;flag = JudgeMFT(file_path, p_dbr_temp, p_ntfs_temp);if (flag){p_dbr_temp->flag = 1; // 設置dbr可用標志位p_dbr_temp->ll_total_sector = (LCN)p_ntfs_temp->sectors_nbr; // 設置分區(qū)總大小g_n_part++;}// 輸出信息printf("Type: NTFS.\tOffset: %I64u.\tSize %I64u.\t Hidden: %lu\tMFT at %I64u cluster.\t MFT is %s!\n",p_dbr_temp->ll_offset / SECTOR_SIZE, (LCN)p_ntfs_temp->sectors_nbr, p_ntfs_temp->hidden,p_ntfs_temp->mft_lcn.QuadPart,flag ? "Right" : "Wrong");} } 復制代碼
  • DBR中sectors_nbr不能比整個文件大小還大
  • MFT是否正確
  • 若兩項都滿足,則判定其為正確的NTFS類型DBR,即可能為正確的分區(qū)信息。那該如何判斷MFT是否正確呢?在0x05中也提到過,定位到MFT的方法,定位過去之后,讀取文件看是否為滿足主文件記錄的格式。需要注意的是,對于NTFS搜索到的DBR有可能是分區(qū)起始扇區(qū)的,也有可能是分區(qū)結束扇區(qū)的,兩者都需要考慮,并且,在判斷MFT的同時,需要把dbr_list中的其他信息填充完整。

    #!cpp // //file_path: 鏡像文件路徑 //p_dbr:dbr_list結構體 //ntfs:dbr // int JudgeMFT(char* file_path, dbr_list* p_dbr, ntfs_boot_sector* ntfs) {char sz_temp1[4] = {0};char sz_temp2[4] = {0};DWORD readsize;LARGE_INTEGER tmp1 = {0};LARGE_INTEGER tmp2 = {0};tmp1.QuadPart = p_dbr->ll_offset + (ntfs->mft_lcn.QuadPart * ntfs->sectors_per_cluster * SECTOR_SIZE);tmp2.QuadPart = p_dbr->ll_offset - (ntfs->sectors_nbr * SECTOR_SIZE) + (ntfs->mft_lcn.QuadPart * ntfs->sectors_per_cluster * SECTOR_SIZE);if (!ReadFileOffset(file_path, tmp1.QuadPart, 4, sz_temp1, FILE_BEGIN))ErrorOut("ReadFile Error!\n");if (!memcmp(sz_temp1, "FILE", 4)){p_dbr->ll_start_sector = p_dbr->ll_offset / SECTOR_SIZE;return 1;}else{if (!ReadFileOffset(file_path, tmp2.QuadPart, 4, sz_temp2, FILE_BEGIN))ErrorOut("ReadFile Error!\n");if (!memcmp(sz_temp2, "FILE", 4)){p_dbr->ll_start_sector = p_dbr->ll_offset / SECTOR_SIZE - ntfs->sectors_nbr;p_dbr->n_is_org = 1;return 1;}}return 0; } 復制代碼

    對于FAT32型,需要考慮的似乎只有是否滿足能正確找到FAT表,如0x05中所說,其大小是需要靠后一個分區(qū)的起始扇區(qū)或文件總大小來獲取的。

    #!cpp if (p_dbr_temp->n_type == 2) // FAT {p_fat_temp = (fat_boot_sector*)p_dbr_temp->dbr;if (!memcmp(p_fat_temp->BS_FilSysType, "FAT32", 5)) // 只處理FAT32{flag = 0;flag = JudgeFAT(file_path, p_dbr_temp, p_fat_temp);if (flag){p_dbr_temp->flag = 1;g_n_part++;}} 復制代碼

    在對于FAT32型DBR,同樣,也需要考慮獲取的DBR是backup的情況:

    #!cpp int JudgeFAT(char *file_path, dbr_list* p_dbr, fat_boot_sector* fat) {char sz_temp1[4] = {0};char sz_temp2[4] = {0};LARGE_INTEGER tmp1 = {0};LARGE_INTEGER tmp2 = {0};char flag[4] = {'\xf8', '\xff', '\xff', '\x0f'};DWORD readsize = 0;tmp1.QuadPart = p_dbr->ll_offset + (fat->reserved * SECTOR_SIZE);tmp2.QuadPart = p_dbr->ll_offset - ((fat->backup_boot + fat->reserved) * SECTOR_SIZE);if (!ReadFileOffset(file_path, tmp1.QuadPart, 4, sz_temp1, FILE_BEGIN))ErrorOut("ReadFile Error!\n");if (!memcmp(sz_temp1, flag, 4)){p_dbr->ll_start_sector = p_dbr->ll_offset / SECTOR_SIZE;return 1;}else{if (!ReadFileOffset(file_path, tmp2.QuadPart, 4, sz_temp2, FILE_BEGIN))ErrorOut("ReadFile Error!\n");if (!memcmp(sz_temp2, flag, 4)){p_dbr->ll_start_sector = p_dbr->ll_offset / SECTOR_SIZE - fat->backup_boot;p_dbr->n_is_org = 1; return 1;}}return 0; } 復制代碼

    完成這一步后,我們得到了篩選出了整個磁盤中所有可用的DBR信息,并且獲取到了重建分區(qū)表所需要的分區(qū)類型、起始扇區(qū)、分區(qū)大小三個信息。

    可以將其輸出由用戶選擇需要恢復的分區(qū):

    #!cpp /*顯示DPT*/ int ShowDPT() {dbr_list* p_dbr_temp = NULL; __int64 tmp = 0;printf("\n\nChosse the partition you want to rebuild?\n");for(p_dbr_temp = g_dbr_list_head->p_next; p_dbr_temp != NULL;) {if (p_dbr_temp->flag) // 需要添加{p_dbr_temp->flag = 0; // 清空標志位置if (tmp < p_dbr_temp->ll_start_sector){printf("\nPartition with type %s.\tStart with %lld sectors.\t Size %lld sectors.\t End with %lld sectors.\nIs this partition you want to restore?(y/n)", (p_dbr_temp->n_type == 1?"NTFS":"FAT32"), p_dbr_temp->ll_start_sector,p_dbr_temp->ll_total_sector,p_dbr_temp->ll_start_sector + p_dbr_temp->ll_total_sector);if (getchar() == 'y'){p_dbr_temp->flag = 1;tmp = p_dbr_temp->ll_start_sector + p_dbr_temp->ll_total_sector;g_n_part++;getchar();}elsegetchar();}}p_dbr_temp = p_dbr_temp->p_next;}return 0; } 復制代碼

    0x09 重構DPT


    重構DPT需要考慮的問題有以下幾個:

  • 總分區(qū)數是否大于4個

    a. 若不是,則可全寫入0扇區(qū)MBR中 b. 若是,則需要新建擴展分區(qū)完成擴展

  • 獲取到的DBR是否為分區(qū)起始扇區(qū)DBR

    a. 若是,則無需更改 b. 若不是,則需要將作為backup的DBR復制到分區(qū)起始扇區(qū)

  • 分區(qū)表是線性一次擴展下去的,不存在分區(qū)交叉的情況,即各分區(qū)大小之和為文件總大小,各分區(qū)無公共部分。(這一步在0x08中的showdpt中做了處理)

  • 創(chuàng)建擴展分區(qū)時,主分區(qū)DPT最后一條記錄是指向擴展DPT,擴展DPT的最后一條記錄繼續(xù)指向下一個擴展DPT。

  • MBR中boot_code部分信息可以不用考慮

  • 考慮完了這些問題,就可以來編碼實現了。我的做法是,新建一個用于重構的鏈表,將所有要更改的內容、要更改的內容的大小、以及要更改的位置寫入到鏈表中,方便后面寫文件以及恢復文件。

    #!cpp void ReBuildDPT(char* sz_file_path) {rebuild_content_t* rebuild_list = CreateReBuildHead();if (g_n_part <= 4) // 小于四個分區(qū)時,只需要建立主分區(qū)表{sz_tmp = (char*)malloc(4 * sizeof(PartTableRecord) + 2);memset(sz_tmp, 0, 4 * sizeof(PartTableRecord) + 2);for(p_dbr_temp = g_dbr_list_head->p_next; p_dbr_temp != NULL;) {if (p_dbr_temp->flag) // 是否需可用信息{*(sz_tmp + k * 16 + 4) = (p_dbr_temp->n_type == 1) ? 0x07 : 0x0B; // byPartTypememcpy(sz_tmp + k * 16 + 8, (char *)&(p_dbr_temp->ll_start_sector), sizeof(__int64)); // dwStartSectormemcpy(sz_tmp + k * 16 + 12, (char *)&(p_dbr_temp->ll_total_sector), sizeof(__int64)); // dwTotalSectork++;if (p_dbr_temp->n_is_org) // 是否起始扇區(qū){InsertRebuildList(rebuild_list, p_dbr_temp->dbr, SECTOR_SIZE, p_dbr_temp->ll_start_sector, i++);}}p_dbr_temp = p_dbr_temp->p_next;}memcpy(sz_tmp + 64, sign, 2); InsertRebuildList(rebuild_list, sz_tmp, 4 * sizeof(PartTableRecord) + 2, 446, i++);}else // 否則考慮擴展分區(qū)的情況{sz_tmp = (char*)malloc(4 * sizeof(PartTableRecord) + 2);memset(sz_tmp, 0, 4 * sizeof(PartTableRecord) + 2);for(p_dbr_temp = g_dbr_list_head->p_next; p_dbr_temp != NULL;) {if (p_dbr_temp->flag) // 是否需可用信息{if (k < 3) // 主分區(qū)只能有三個,最后一個為擴展分區(qū){if (k != 2){*(sz_tmp + k * 16 + 4) = (p_dbr_temp->n_type == 1) ? 0x07 : 0x0B; // byPartTypememcpy(sz_tmp + k * 16 + 8, (char *)&(p_dbr_temp->ll_start_sector), sizeof(__int64)); // dwStartSectortmp = p_dbr_temp->ll_total_sector + 1;memcpy(sz_tmp + k * 16 + 12, (char *)&tmp, sizeof(__int64)); // dwTotalSectork++;}else{*(sz_tmp + k * 16 + 4) = (p_dbr_temp->n_type == 1) ? 0x07 : 0x0B; // byPartTypememcpy(sz_tmp + k * 16 + 8, (char *)&(p_dbr_temp->ll_start_sector), sizeof(__int64)); // dwStartSectortmp = p_dbr_temp->ll_total_sector + 1;memcpy(sz_tmp + k * 16 + 12, (char *)&tmp, sizeof(__int64)); // dwTotalSectork++;for (p_dbr_temp_tmp = p_dbr_temp->p_next; p_dbr_temp_tmp != NULL;){if (p_dbr_temp_tmp->flag){*(sz_tmp + k * 16 + 4) = 0x05; // byPartTypetmp = p_dbr_temp_tmp->ll_start_sector - 1;memcpy(sz_tmp + k * 16 + 8, (char *)&tmp, sizeof(__int64)); // dwStartSectortmp = (g_ll_file_size/SECTOR_SIZE) - p_dbr_temp_tmp->ll_start_sector + 1;memcpy(sz_tmp + k * 16 + 12, (char *)&tmp, sizeof(__int64)); // dwTotalSectork++;memcpy(sz_tmp + 64, sign, 2); InsertRebuildList(rebuild_list, sz_tmp, 4 * sizeof(PartTableRecord) + 2, 446, i++);break;}p_dbr_temp_tmp = p_dbr_temp_tmp->p_next;}}}else{sz_tmp = NULL;sz_tmp = (char*)malloc(4 * sizeof(PartTableRecord) + 2);memset(sz_tmp, 0, 4 * sizeof(PartTableRecord) + 2);*(sz_tmp + 4) = (p_dbr_temp->n_type == 1) ? 0x07 : 0x0B; // byPartTypetmp = 1; // 擴展分區(qū)偏移地址從當前地址算起(相對地址)memcpy(sz_tmp + 8, (char *)&tmp, sizeof(__int64)); // dwStartSectormemcpy(sz_tmp + 12, (char *)&(p_dbr_temp->ll_total_sector), sizeof(__int64)); // dwTotalSectorif (p_dbr_temp->p_next != NULL){for (p_dbr_temp_tmp = p_dbr_temp->p_next; p_dbr_temp_tmp != NULL;){if (p_dbr_temp_tmp->flag){*(sz_tmp + 16 + 4) = 0x05; // byPartTypetmp = p_dbr_temp_tmp->ll_start_sector - p_dbr_temp->ll_start_sector;//tmp = 1;memcpy(sz_tmp + 16 + 8, (char *)&tmp, sizeof(__int64)); // dwStartSectortmp = (g_ll_file_size/SECTOR_SIZE) - p_dbr_temp_tmp->ll_start_sector;memcpy(sz_tmp + 16 + 12, (char *)&tmp, sizeof(__int64)); // dwTotalSectorbreak;}p_dbr_temp_tmp = p_dbr_temp_tmp->p_next;}}memcpy(sz_tmp + 64, sign, 2); InsertRebuildList(rebuild_list, sz_tmp, 66, (p_dbr_temp->ll_start_sector - 1) * SECTOR_SIZE + 446, i++);}if (p_dbr_temp->n_is_org) // 是否起始扇區(qū){InsertRebuildList(rebuild_list, p_dbr_temp->dbr, SECTOR_SIZE, p_dbr_temp->ll_start_sector * SECTOR_SIZE, i++);}}p_dbr_temp = p_dbr_temp->p_next;}}HandleFile(sz_file_path, rebuild_list);FreeRebuildList(rebuild_list); } 復制代碼

    0x0A 文件處理


    通過前面的操作,可以得到處理好了的rebuild_list。接下來要做的就是用它來完成重建分區(qū)表,恢復DBR的工作。

    首先,我們的目的是,重建后的分區(qū)表后文件能作為VHD直接被win7以上系統(tǒng)加載。其次,希望能夠在我卸載VHD文件后,仍然恢復到原有狀態(tài)。意味著需要對更改的信息做一個備份,這沒問題,因為我們替換先前rebuild_list中的content就可以完成了。

    之后,需要了解VHD文件格式。找了找資料。發(fā)現,VHD文件僅僅在文件尾部添加了一個扇區(qū)的內容,其結構如下:

    #!cpp /*vhd尾部信息結構*/ typedef struct hd_ftr_t { char cookie[8]; /* Identifies original creator of the disk */ unsigned int features; /* Feature Support -- see below */ unsigned int ff_version; /* (major,minor) version of disk file */ unsigned __int64 data_offset; /* Abs. offset from SOF to next structure */ unsigned int timestamp; /* Creation time. secs since 1/1/2000GMT */ char crtr_app[4]; /* Creator application */ unsigned int crtr_ver; /* Creator version (major,minor) */ unsigned int crtr_os; /* Creator host OS */ unsigned __int64 orig_size; /* Size at creation (bytes) */ unsigned __int64 curr_size; /* Current size of disk (bytes) */ unsigned int geometry; /* Disk geometry */ unsigned int type; /* Disk type */ unsigned int checksum; /* 1's comp sum of this struct. */ unsigned char uu[16]; /* Unique disk ID, used for naming parents */ char saved; /* one-bit -- is this disk/VM in a saved state? */ char hidden; /* tapdisk-specific field: is this vdi hidden? */ char reserved[426]; /* padding */ }hd_ftr; 復制代碼

    這張表中,重要的就是orig_size、curr_size和checksum,通常情況下orig_size與curr_size相同,checksum是最后一個扇區(qū)所有字節(jié)相加后取反的值。我建了個模板來實現VHD標志位的添加。

    所以,最終的文件處理模塊如下:

    #!cpp void HandleFile(char* file_path, rebuild_content_t* p_rebuild_list) {char* sz_vhd_buf = (char*)malloc(SECTOR_SIZE);memset(sz_vhd_buf, 0, SECTOR_SIZE);rebuild_content_t* p_rebuild_tmp = NULL;char tmp[SECTOR_SIZE] = {0}; // Gen VHDhd_ftr* vhd;vhd = (hd_ftr*)data;LARGE_INTEGER offset = {0};DWORD readsize = 0;/*Set hd_ftr struct*/vhd->orig_size = 0; // clearvhd->orig_size = g_ll_file_size - SECTOR_SIZE;vhd->orig_size = INT64_TO_NET(vhd->orig_size);vhd->curr_size = vhd->orig_size;vhd->checksum = 0;/*calc checksum*/unsigned int temp = 0;for (int i = 0; i < 512; i++){temp += data[i];}vhd->checksum = htonl(~temp); //for(p_rebuild_tmp = p_rebuild_list->p_next; p_rebuild_tmp != NULL;) {if (!ReadFileOffset(file_path, p_rebuild_tmp->ll_offset, p_rebuild_tmp->n_size, tmp, FILE_BEGIN))ErrorOut("Backup Read Error!\n");if (!WriteFileOffset(file_path, p_rebuild_tmp->ll_offset, p_rebuild_tmp->n_size, p_rebuild_tmp->content, FILE_BEGIN))ErrorOut("Backup Write Error!\n");memcpy(p_rebuild_tmp->content, tmp, p_rebuild_tmp->n_size); // BackUp SECTORp_rebuild_tmp = p_rebuild_tmp->p_next;}/ BackUp VHDReadFileOffset(file_path, -SECTOR_SIZE, SECTOR_SIZE, sz_vhd_buf, FILE_END);/* */// Write VHDWriteFileOffset(file_path, -SECTOR_SIZE, SECTOR_SIZE, (char*)vhd, FILE_END);printf("WriteFile Success! You can mount it as vhd file now!\n");system("pause");// Restore SECTORfor(p_rebuild_tmp = p_rebuild_list->p_next; p_rebuild_tmp != NULL;) {if (!ReadFileOffset(file_path, p_rebuild_tmp->ll_offset, p_rebuild_tmp->n_size, tmp, FILE_BEGIN))ErrorOut("Restore Read Error!\n");if (!WriteFileOffset(file_path, p_rebuild_tmp->ll_offset, p_rebuild_tmp->n_size, p_rebuild_tmp->content, FILE_BEGIN))ErrorOut("Restore Write Error!\n");memcpy(p_rebuild_tmp->content, tmp, p_rebuild_tmp->n_size); // BackUp SECTORp_rebuild_tmp = p_rebuild_tmp->p_next;}/ Restore VHDWriteFileOffset(file_path, -SECTOR_SIZE, SECTOR_SIZE, sz_vhd_buf, FILE_END);printf("Restore File Success!\n"); } 復制代碼

    這樣,重建分區(qū)表的工作就完成了。

    0x0B 程序效果


    還是用前面例子中用到的VHD文件來做演示,該VHD現在的情況是:先前新建過兩個NTFS的分區(qū),之后刪除了一個NTFS,并重新格式化成FAT32,再然后,我將其分區(qū)全部刪除,并新建了3個NTFS與兩個FAT32,最終效果如圖6所示。現在我們用剛寫好的工具來對它進行搜索分析。

    圖17:

    可以看到它總共搜索到了8個可用的DBR信息,但顯然其中有一些是前幾次分區(qū)留下的內容。選擇保留1.2.3.5.7五個分區(qū)(也就是圖6所示的五個分區(qū)),起始你選擇保留第三個的時候,就不有機會讓你選擇第四個了,因為他們是沖突的。結果如圖18:

    提示寫文件成功,并已經可以直接當做VHD文件掛載了,我們用winhex打開處理過的文件,得到圖19:

    可以看到,重構后的分區(qū)表幾乎與原來的分區(qū)表一致。

    接著,我們試著來恢復前幾次分區(qū)留下的信息,看是否能夠成功。圖20:

    這里我選擇恢復了一個NTFS和一個FAT32,再進winhex中查看,得到圖21:

    可以看到第一個NTFS分區(qū),是并不存在與圖6的五個分區(qū)中的。

    再次,我們用圖1的那個TrueCrypt解密后的打不開的磁盤作為例子,看看這次能否恢復出正確的分區(qū)表,并且,實現加載。找到可用的DBR信息如圖22:

    按照邏輯,選擇了第一個后,就只能選擇1和2。已提示寫文件成功,圖23:

    Winhex解析如圖24:

    正確完整恢復,所有文件也能正確解析,圖25:

    接著,在做只留下尾部備份分區(qū)測試程序需找情況的時候,發(fā)現起始我們程序中并沒有必要將備份分區(qū)復制到起始扇區(qū),因為,也能正常解析。

    于是,完工!(不過,還有個小細節(jié)需要注意,如果起始位置是個DBR頭部的話,系統(tǒng)將不會當做MBR處理,而是當做DBR,所以,清除頭部信息也很重要!)

    0x0C Summarize


    加上寫代碼的時間和些這文章的時間,大概花費了近十天吧。各種找資料、調程序、做樣本,也算是把恢復分區(qū)表的原理弄清楚了。同時,也把忘得差不多的磁盤格式、文件系統(tǒng)什么的再撿起來看了看。當然文章其實并不詳細,因為關注點不一樣,就沒講太多關于文件系統(tǒng)的內容了。再往下,可以詳細到文件的恢復等等,當然,起始也沒必要去做,畢竟現有的工具一大堆,我只是突然感興趣就拿來實現了一把。

    再者,本程序暫時只支持NTFS和FAT32兩種類型,若有其他類型的原理應該也是一致,大家仔細琢磨。

    寫文章也比較倉促,也沒校稿習慣,若有勘誤,還請諒解并提醒更正。感謝!

    源碼下載地址:github.com/purpleroc/h…
    另推薦學習源碼:testdisk、ReadPartTable
    ——Tracy_梓朋
    2016年1月6日21:31:52

    總結

    以上是生活随笔為你收集整理的磁盘分区表恢复原理的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    国产精品国产三级国产aⅴ入口 | 成人黄色av网站 | 香蕉91视频 | 免费看的黄色 | 日韩av综合网站 | 欧美日本一区 | 日韩精品在线视频 | 二区三区在线视频 | av再线观看 | 国产97视频在线 | 91热在线| 日精品在线观看 | 国产精品涩涩屋www在线观看 | 免费看国产一级片 | 久久免费播放 | 免费色视频网址 | 国内精品久久久久影院优 | 天天干天天干天天操 | 日本高清中文字幕有码在线 | 天天色天天射综合网 | 狠狠操狠狠干天天操 | 国产99久久久国产精品免费二区 | 国产小视频在线观看免费 | 国产小视频在线播放 | 国内久久久久 | 成年人在线免费看视频 | 国产在线最新 | 丁香 久久 综合 | 亚洲精品观看 | 久久精品一二三区 | 日本精品久久久久久 | 国产99在线| 日韩欧美精品在线观看 | 天天操天天爱天天爽 | 天天操天操 | 亚洲精品va | 久久夜色精品国产亚洲aⅴ 91chinesexxx | 国产在线观看你懂得 | 美女视频a美女大全免费下载蜜臀 | 中文字幕一区二区三区四区 | av手机在线播放 | 超级碰视频 | 国产在线资源 | 亚洲免费av一区二区 | 欧美成人性战久久 | 国产精品九九久久久久久久 | 亚洲一区久久久 | 国产精品视频不卡 | 夜夜夜夜爽 | 中文字幕在线观看av | 国产婷婷vvvv激情久 | 亚洲午夜久久久久久久久电影网 | 久久久精品电影 | jizz欧美性9| 欧美二区视频 | 黄网站色视频 | 国产精品99久久久久久久久久久久 | 一区二区伦理 | 国产精品久久久久国产精品日日 | 国产精品一区二区三区四 | 天天操天天爽天天干 | 日日干夜夜操视频 | 五月天丁香亚洲 | 五月天丁香综合 | 免费在线成人av电影 | 91在线免费公开视频 | 天堂网一区二区 | 成人超碰97| 91九色在线 | 超碰97人人射妻 | 99视频在线精品 | 久久系列 | 激情综合五月天 | 成人 亚洲 欧美 | 国产成人区 | 草久在线视频 | 狠狠激情中文字幕 | 免费看十八岁美女 | 91免费看片黄 | 337p日本欧洲亚洲大胆裸体艺术 | 色999五月色 | 婷婷99| 青青河边草观看完整版高清 | 色吊丝av中文字幕 | 欧美国产精品一区二区 | 免费成人黄色 | 91经典在线 | www.色婷婷.com| 国产精品一区二区久久久 | 欧美性色综合 | 久久久精品亚洲 | 亚洲综合网 | 欧美一级艳片视频免费观看 | 久久成人精品电影 | 最近最新中文字幕视频 | 亚洲视频精选 | 99色在线播放 | 亚洲天天干 | www.888av | 亚洲国产精品久久久久久 | 亚洲欧洲精品在线 | 久艹在线观看视频 | 91中文在线视频 | 久久久精品久久 | 国产精品久久久久久久久搜平片 | 亚洲最大成人免费网站 | 日韩视频图片 | a黄色一级片| 中国一区二区视频 | 欧美极品一区二区三区 | 欧美一级欧美一级 | 色婷婷五 | 国内精品二区 | 一区二区在线不卡 | 99资源网 | 在线电影 你懂得 | 欧美黑人xxxx猛性大交 | 亚洲高清视频在线观看免费 | 中文字幕一区二区三区四区视频 | 人人超碰免费 | 久热精品国产 | 亚洲精品国偷拍自产在线观看蜜桃 | 国产成人高清 | 国产又粗又猛又爽 | 欧美a免费| 久草视频免费在线播放 | 久久久影院官网 | 久久综合久久综合久久综合 | 美女视频永久黄网站免费观看国产 | 在线免费看黄网站 | 久av在线 | 亚洲污视频 | 蜜桃视频成人在线观看 | 91精选在线 | 欧美精品做受xxx性少妇 | 亚洲高清视频一区二区三区 | 顶级bbw搡bbbb搡bbbb | 99r精品视频在线观看 | 日韩网站在线免费观看 | 青青河边草免费观看 | 在线亚洲午夜片av大片 | 亚洲日本一区二区在线 | 成人一级黄色片 | 午夜手机电影 | 米奇狠狠狠888 | 色天天综合久久久久综合片 | 99久久日韩精品免费热麻豆美女 | 久久精品国产亚洲精品 | 91亚洲欧美激情 | 国产69精品久久99的直播节目 | 在线观看视频黄色 | 天天综合网 天天 | 欧美视频18 | 日韩va亚洲va欧美va久久 | 日韩美av在线 | 欧洲一区二区在线观看 | 国产精品99久久久久久小说 | 在线观看成人国产 | 国产vs久久 | 成人av在线直播 | 国产精品一区二区果冻传媒 | 亚洲黄色在线播放 | 超碰在线最新网址 | 国产精品美女久久久久久 | 在线观看日本韩国电影 | av免费网页 | 天天搞天天干 | 国模吧一区 | 国产黄色播放 | 在线观看色网 | 蜜臀av在线一区二区三区 | 日韩精品中文字幕av | 国产高清在线观看 | 欧美精品一二三 | 免费看的av片 | 亚洲 欧美 综合 在线 精品 | 国产精品美女久久 | 99操视频 | 激情亚洲综合在线 | 国产 成人 久久 | 国产在线观看,日本 | 精品国产123 | 又色又爽又黄 | 国产一区免费看 | 欧美午夜剧场 | 国产激情小视频在线观看 | 亚洲日本欧美在线 | 91丨porny丨九色 | 91麻豆精品国产自产在线游戏 | 一级片免费视频 | 免费在线a | 手机av永久免费 | 亚洲精选在线观看 | 欧美日韩国产在线精品 | 在线综合 亚洲 欧美在线视频 | 亚洲视频456 | 久久新| 久草免费资源 | av电影久久 | 五月婷婷综合久久 | 日韩特黄av| 成年人视频免费在线播放 | 99久久99久久 | 精品一区电影 | av九九| 久久99久久99精品免观看软件 | 精品一区二区视频 | 国产福利午夜 | 99热 精品在线 | 色婷婷综合久久久 | 日日操日日 | 在线国产不卡 | 亚洲美女精品区人人人人 | 国产精品mv | 欧美精品久久久久久久久久丰满 | 日韩特级片 | 精品国产乱码一区二 | 激情视频免费在线 | av黄色影院 | 久久女同性恋中文字幕 | 日韩免费三区 | 亚洲免费小视频 | 香蕉视频久久久 | 日日夜夜精品 | 在线观看中文字幕av | 精品久久亚洲 | 国产精品一二三 | 在线播放91 | 国产中文字幕在线播放 | 月丁香婷婷 | 中文字幕av全部资源www中文字幕在线观看 | 国产黄色片免费观看 | 成人av电影在线播放 | 91成人精品在线 | 亚洲高清91 | 久久久夜色 | 日韩在线观看中文字幕 | 美女黄频 | 在线观看视频在线 | 日本99精品| 日日夜夜艹 | 欧美日韩视频网站 | 九九在线国产视频 | 久久黄色免费视频 | 国产精品电影在线 | 国产精品99久久久久久武松影视 | 亚洲精品免费在线观看视频 | 国产伦理久久精品久久久久_ | 日韩电影在线观看中文字幕 | 国产婷婷在线观看 | 中文字幕免费高清av | 777视频在线观看 | 97碰碰碰| 国产精品白丝jk白祙 | 少妇bbw搡bbbb搡bbb | av在线不卡观看 | 久久综合婷婷 | 日韩免费专区 | 免费国产在线精品 | 粉嫩av一区二区三区四区在线观看 | 久久精品站 | 中文国产在线观看 | 亚洲一区二区三区毛片 | 国产精品黑丝在线观看 | 国产在线播放一区二区 | 玖玖精品在线 | 97在线观看免费高清完整版在线观看 | 国产不卡av在线 | 丰满少妇在线观看 | 国产群p | 操操操av | 黄色在线观看www | 成人99免费视频 | 波多野结衣一区二区 | 久久免费视频7 | 二区三区在线视频 | 九九视频热 | 久久精品网站视频 | 99在线观看免费视频精品观看 | 欧美精品乱码久久久久久 | 少妇啪啪av入口 | 日本韩国在线不卡 | 欧美99精品 | 久草视频中文 | 国产中文| 中文字幕在线观看你懂的 | 久草在线视频网站 | 日韩欧美视频一区二区 | 亚洲欧美激情插 | 国产一区在线免费 | 国产手机在线播放 | 午夜av免费在线观看 | 99精品久久只有精品 | 久久不射电影院 | 婷婷久久五月天 | 在线99热 | 久久久毛片 | 国产精品一区二区精品视频免费看 | 97视频免费观看2区 亚洲视屏 | 中文字幕在线成人 | 国产成人久久精品 | 国产人成在线观看 | 欧美日韩国产一二三区 | 国产成人精品一区二区三区福利 | 欧美激情综合五月色丁香 | 久久午夜网 | 免费看亚洲毛片 | 国产偷国产偷亚洲清高 | 亚洲精品午夜视频 | 久久久久久久久久久综合 | 国产 一区二区三区 在线 | 人人干天天干 | 国产精品第一 | 欧美一二区视频 | 国产成人久久av免费高清密臂 | 日本三级在线观看中文字 | 在线观看的黄色 | 五月天婷亚洲天综合网鲁鲁鲁 | 麻豆播放 | 久久欧洲视频 | 黄色网中文字幕 | 日韩高清三区 | 日韩精品在线视频 | 手机成人av | 99精品久久久久 | 97国产精品视频 | 天天插夜夜操 | 色婷婷99| 国产视频亚洲视频 | 这里只有精彩视频 | 欧美日韩亚洲在线观看 | 成人羞羞视频在线观看免费 | 日韩精品短视频 | 伊人国产在线播放 | 午夜三级大片 | 国产 视频 久久 | 国产精品综合在线观看 | 精品在线观看一区二区三区 | 欧美性另类| 91成人在线观看高潮 | 日韩免费在线视频观看 | av在线不卡观看 | 欧美精品久久 | 色www精品视频在线观看 | 最近日本mv字幕免费观看 | 丁香六月天婷婷 | 国产精品久久久久久久久免费 | 国产视频一二三 | 视频一区二区免费 | 久久久精品欧美 | 一区二区三区在线观看免费视频 | 国产在线毛片 | 亚洲免费一级 | 综合精品在线 | 麻豆手机在线 | 在线免费观看麻豆视频 | 美国av大片| 亚洲性xxxx| 99综合电影在线视频 | 2018亚洲男人天堂 | 91在线中文 | www.天天色| 天天操夜夜干 | 天天综合网在线观看 | 69视频永久免费观看 | 国产精品 亚洲精品 | 欧美在线视频第一页 | 狠狠狠狠狠狠狠干 | 视频在线观看国产 | 亚洲欧洲一级 | www.啪啪.com| 成年人免费看的视频 | 五月婷在线播放 | 最近2019年日本中文免费字幕 | 久草av在线播放 | 亚洲欧洲国产视频 | 又黄又爽免费视频 | 九九热国产| 精品久久久久久久久亚洲 | 一区中文字幕 | av色影院| 日韩资源在线播放 | 欧美国产高清 | 久久av伊人 | 麻豆久久精品 | 久久视频一区二区 | 欧美性黄网官网 | 四虎永久网站 | 亚洲理论在线观看电影 | 精品在线观看视频 | 免费av观看网站 | 伊人国产女 | 在线国产一区 | 99这里只有精品视频 | 在线观看v片 | 欧美a级免费视频 | 99久久99久久精品国产片果冰 | 91香蕉亚洲精品 | 日本精品va在线观看 | 婷婷色网站 | 成人免费视频免费观看 | 91网在线看 | 日韩久久久久久久 | 最新一区二区三区 | www在线观看国产 | 国产99久久久国产精品免费二区 | 成人在线黄色电影 | 国产热re99久久6国产精品 | 免费在线激情电影 | 日韩av影视在线观看 | 国产黄色免费电影 | 国产区 在线 | 在线小视频你懂得 | 国产精品免费一区二区 | 中文av在线免费观看 | 草久久久久 | 在线免费观看视频一区二区三区 | 国产成人在线观看 | 免费在线观看av | 超碰97在线资源站 | 五月天丁香综合 | 888av| 久久字幕精品一区 | 在线观看911视频 | 中文字幕在线观看网址 | 激情五月婷婷激情 | 久久观看最新视频 | 婷婷伊人网 | 国产精品美女免费视频 | 狠狠色香婷婷久久亚洲精品 | 91试看 | 国产日韩视频在线观看 | 久久精品福利视频 | 天天综合日日夜夜 | 国产视频一区二区在线 | 成人午夜剧场在线观看 | 国产小视频91 | 91天堂影院 | 成人av在线直播 | 久久久免费观看 | 亚洲精品乱码久久久久久 | 高清av网 | 国产午夜精品一区二区三区嫩草 | 国产.精品.日韩.另类.中文.在线.播放 | 黄色精品久久 | 日韩视频在线不卡 | 婷婷电影在线观看 | 国产高h视频 | 成人av网页 | 深夜免费小视频 | 亚洲国产日本 | 成人免费在线播放视频 | 日本资源中文字幕在线 | 五月天视频网站 | 午夜久久成人 | 日日干天天爽 | 欧美激情va永久在线播放 | 99久久久国产免费 | 精品国产一区在线观看 | 日韩高清国产精品 | 激情综合啪啪 | 久久r精品| 国精产品满18岁在线 | 麻豆久久 | 久久99精品热在线观看 | 91av资源网| 欧美精品在线视频观看 | 97电影在线看视频 | 久久久www免费电影网 | 人人澡超碰碰97碰碰碰软件 | 久草在线免费资源 | 97国产精品一区二区 | 国产裸体视频网站 | 毛片网站在线看 | 日韩欧美精品一区 | 中文字幕一区av | 麻豆影视在线播放 | 在线观看视频99 | 国产婷婷视频在线 | 久久免费视频在线 | 久草视频在线新免费 | 碰天天操天天 | 亚洲综合狠狠干 | 久久久综合电影 | 深爱五月激情五月 | 亚洲国产中文字幕在线观看 | 最近中文字幕高清字幕在线视频 | av一区二区三区在线 | 丁香影院在线 | 久久天天操 | 99爱视频在线观看 | 操综合| 亚洲一区二区视频在线 | 日本精a在线观看 | 日日添夜夜添 | 国产精品激情在线观看 | 亚洲在线免费视频 | 人人爽人人插 | 一本之道乱码区 | 五月天中文在线 | a天堂最新版中文在线地址 久久99久久精品国产 | 天天操欧美 | 91色视频| 成人理论在线观看 | 久久亚洲精品国产亚洲老地址 | 美女免费电影 | 亚洲人成免费 | 亚洲日本一区二区在线 | 成人免费在线电影 | 免费看黄色小说的网站 | 狠狠干免费 | 正在播放五月婷婷狠狠干 | av中文字幕网址 | 国产精品毛片一区二区 | www.狠狠干 | 97超碰人人模人人人爽人人爱 | 成人国产综合 | 97视频在线观看网址 | 一区二区中文字幕在线观看 | 在线小视频你懂得 | 成年人在线观看视频免费 | 人人爽人人爽人人爽人人爽 | 日韩在线观看三区 | 免费看片在线观看 | 亚洲一区黄色 | 黄色网大全 | 久草精品国产 | 亚州精品一二三区 | 欧美日韩亚洲一 | 探花视频网站 | 久久超碰网| 国产一区二区午夜 | 爱情影院aqdy鲁丝片二区 | 免费看国产a | 天天鲁一鲁摸一摸爽一爽 | 91精品对白一区国产伦 | 五月天久久综合网 | 黄色国产在线 | 91视频久久久久久 | 亚洲激情影院 | 日韩成人黄色av | 久久久久婷 | 久久永久免费 | 天天色综合久久 | 亚洲欧美久久 | 四虎影视精品 | av久久在线 | 久久精品视频中文字幕 | 91精品国产一区二区三区 | 精品国产精品国产偷麻豆 | 超碰999| 99久久精品久久久久久清纯 | 色婷婷六月| 国产亚洲欧美在线视频 | 亚洲日本精品 | 91porny九色91啦中文 | 狠狠狠狠狠色综合 | 日韩黄在线观看 | 日韩成人中文字幕 | 天天干天天操天天射 | 夜夜婷婷| 国产女教师精品久久av | 五月花婷婷 | 午夜精品久久久久久久99无限制 | 国产精品美女久久久久久久 | 在线观看黄网 | 国产女教师精品久久av | 99色在线| 欧美精品久久久久久久久久 | 天堂入口网站 | 国产视频不卡 | av解说在线观看 | 在线视频精品播放 | 日韩一区二区三区免费视频 | 黄色特级一级片 | 日韩中文在线播放 | 色婷婷综合久久久久 | 婷婷99| 欧美日韩在线观看一区 | 日韩在线看片 | 国内精品免费 | 嫩草av在线 | 免费在线观看国产精品 | 亚洲美女免费视频 | 国产黄色免费观看 | 91九色porny在线 | av福利在线 | 婷婷福利影院 | 日本久久久久久久久久久 | 久久九九久久九九 | 国产一级淫片在线观看 | 久草国产在线观看 | 永久免费毛片 | 97在线看 | 在线免费试看 | 国产中文字幕久久 | 亚洲狠狠丁香婷婷综合久久久 | 久久久久黄色 | 免费视频成人 | 久久亚洲免费视频 | 99久久99久久精品国产片 | 亚洲三级av | 免费日韩视| 九九日韩| 亚洲三级精品 | 777久久久| 国产在线观看污片 | 欧美日韩亚洲第一页 | 91视频免费看片 | 欧美在线视频一区二区三区 | 麻豆传媒视频在线播放 | 日av免费 | 91av原创| 午夜精品久久久久久久99 | 久久婷婷一区二区三区 | 91在线在线观看 | www.狠狠插.com | 香蕉视频啪啪 | 免费在线观看日韩视频 | 正在播放国产一区 | www.久久久| 超碰在线成人 | 国产伦理久久精品久久久久_ | 日日摸日日添日日躁av | 97视频在线免费播放 | 亚洲三级网| 精品久久久99 | 国产精品不卡视频 | av在线一| 国产中文字幕一区二区 | 国产亚洲欧美一区 | 91爱爱网址 | 九九免费精品视频在线观看 | 久久久国产一区二区三区 | 国产 日韩 在线 亚洲 字幕 中文 | 黄色软件网站在线观看 | 国产一级黄色片免费看 | 一区二区精品视频 | 丁香激情五月 | 蜜桃视频日本 | 国产精品久久久久久影院 | 毛片99 | 国产精品毛片久久久久久久 | 国产黄色精品 | 婷婷久久综合九色综合 | 久久成人亚洲欧美电影 | www.亚洲视频| 久久人人爽爽 | 深爱激情站 | 99综合电影在线视频 | av亚洲产国偷v产偷v自拍小说 | 精品国产乱码久久久久久三级人 | 男女啪啪视屏 | 久草视频在线免费 | 999国产在线| 天天曰天天射 | 久久久久免费看 | 欧美久久久 | 日韩在线无| 激情综合五月 | 波多野结衣亚洲一区二区 | a v在线视频 | 亚洲成年人av | 亚洲欧美va | 久久久久久久久艹 | 国际精品久久久久 | 最近2019年日本中文免费字幕 | 成人毛片在线视频 | 成全免费观看视频 | 日韩三级视频在线看 | 国产视频亚洲视频 | 日韩中字在线 | 狠狠干婷婷色 | 99视频网站| 成人欧美一区二区三区在线观看 | 欧美在线观看视频免费 | 国产精品久久久久久模特 | 久久免费精品一区二区三区 | 黄色在线观看网站 | 狠狠ri| 极品国产91在线网站 | 久久久久国产精品免费免费搜索 | av在线免费在线 | 精品国产乱码 | 亚洲精品乱码久久久久久蜜桃动漫 | 久久久久一区二区三区 | 在线亚洲人成电影网站色www | 中文在线字幕免费观 | 欧美xxxxx在线视频 | 精品人人人人 | 国产精品久久久久久久久久东京 | 国产精品视频你懂的 | 国产一二区视频 | 成人黄色中文字幕 | 91黄视频在线观看 | 欧美日韩一二三四区 | 久久国产亚洲精品 | 久久国产露脸精品国产 | 亚洲,国产成人av | 欧美久久综合 | 色噜噜狠狠色综合中国 | 欧美日韩精品在线观看视频 | 久久a久久 | 精品一区二区在线播放 | 四虎在线永久免费观看 | 狠狠干夜夜 | 日日爽天天操 | 婷婷亚洲激情 | 欧美久久久久久久久久久久 | 丝袜美女视频网站 | 99久久精品国产一区 | 免费看特级毛片 | 女人18毛片a级毛片一区二区 | 免费色视频网址 | 亚洲不卡av一区二区三区 | 2023国产精品自产拍在线观看 | 91在线91 | 精品久久久久亚洲 | 99久久久久久久久 | 日韩精品国产一区 | 91九色精品国产 | 婷婷四房综合激情五月 | 日韩精品久久中文字幕 | 色悠悠久久综合 | 久久精品视频免费 | av免费在线观看网站 | 黄视频色网站 | 麻豆影视网站 | 91福利在线导航 | 在线观看岛国 | 92中文资源在线 | 成人羞羞视频在线观看免费 | 麻豆免费视频网站 | 国产不卡免费av | 日韩中文字幕第一页 | 欧美国产日韩一区二区 | 久久国产精品系列 | 欧美成人在线免费观看 | 中文字幕一区二区三区视频 | 成人av免费在线观看 | 亚洲成人资源网 | 久久国产精品二国产精品中国洋人 | 久久97久久 | 91在线一区| 91日韩精品| 亚洲国产一区二区精品专区 | 天天在线视频色 | 亚洲国产69 | 色视频在线免费观看 | 国产福利91精品一区 | 成人中文字幕+乱码+中文字幕 | 国产精品普通话 | 国产正在播放 | 亚洲精品福利在线观看 | 亚洲精品国产品国语在线 | 国产精品 视频 | 不卡的av中文字幕 | 中文字幕在线免费看 | av在线影视 | www.99热精品| 国产精品亚洲片夜色在线 | 九九精品久久久 | 国产五月婷 | 国产精品一区二区三区在线免费观看 | 欧美一级裸体视频 | 四虎永久国产精品 | 日韩欧美一区二区三区视频 | 天天拍天天草 | 天天躁日日躁狠狠躁 | 国产九九在线 | 欧美综合在线观看 | 久久午夜精品 | 人人玩人人添人人 | 亚州精品成人 | 久久久久久国产精品亚洲78 | 成人网页在线免费观看 | 黄色免费在线视频 | 免费在线观看日韩 | av女优中文字幕在线观看 | 丁香婷婷激情网 | 中文字幕91视频 | 一级久久精品 | 久久久久久蜜av免费网站 | 91片在线观看| 久久久久久久99 | 欧美一区二区三区在线观看 | 午夜精品久久一牛影视 | 日韩欧美大片免费观看 | 99riav1国产精品视频 | 久久爱综合| 久久国产精品99国产精 | 91视频黄色 | 色综合久久久久综合体 | 国产精品99久久久久久久久 | av三区在线| 在线观看免费高清视频大全追剧 | 国产一二区免费视频 | 久久新视频 | 在线免费色 | 欧美不卡视频在线 | 99热都是精品 | 国产精品系列在线观看 | 激情图片区 | 天天操天天操天天操天天操天天操 | 一区二区三区高清在线观看 | 欧美日韩高清在线观看 | 成人在线免费视频 | 有码中文字幕 | 国产精品久久久久亚洲影视 | 久亚洲 | 国产一级性生活视频 | 青青河边草观看完整版高清 | 国产欧美久久久精品影院 | 日韩在线视频在线观看 | 久久99深爱久久99精品 | 欧美日韩国产网站 | 91av蜜桃| 国产群p | 91香蕉视频黄 | 视频99爱| 国产va在线观看免费 | 午夜精品久久久久久久久久 | 国产美女主播精品一区二区三区 | 久久69精品久久久久久久电影好 | 在线视频在线观看 | 国产精品一区二区三区视频免费 | 久草在线中文视频 | 亚洲伊人婷婷 | 超黄视频网站 | 亚洲国产精品久久久久久 | 日韩黄色免费 | 亚洲精品一区二区三区新线路 | 日韩午夜电影院 | 在线国产日韩 | 日韩高清二区 | 欧美日韩在线观看一区二区三区 | 国产黄免费 | 91精品久 | 亚洲一级国产 | 国产精品国产三级在线专区 | 黄色录像av| 在线观看网站你懂的 | 久久亚洲欧美日韩精品专区 | 久久麻豆精品 | 在线观看国产高清视频 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 91在线视频播放 | 精品你懂的 | 三级av在线免费观看 | 狠狠色2019综合网 | 国产视频手机在线 | 一区二区三区日韩精品 | 天堂av色婷婷一区二区三区 | 成年人电影免费看 | 欧美日韩视频观看 | 最近免费观看的电影完整版 | 久久99精品国产麻豆婷婷 | 欧美一级免费片 | 亚洲精品午夜久久久久久久久久久 | 最近高清中文字幕在线国语5 | 免费在线观看午夜视频 | 亚洲一区美女视频在线观看免费 | 国产精品99久久久久久人免费 | 91网免费观看 | 亚洲国产成人在线 | 丁香六月在线 | 国产成人av在线 | 国产91成人在在线播放 | 波多野结衣理论片 | 免费看十八岁美女 | 夜夜骑日日操 | 免费观看性生交大片3 | 久久精品视频在线观看免费 | av三级在线免费观看 | 中文字幕在线第一页 | 欧美性生活小视频 | 久草在线视频中文 | 日韩av一区二区在线影视 | 在线小视频 | 丝袜网站在线观看 | 日本精品一二区 | 香蕉在线影院 | 一级欧美黄 | 91亚洲精品久久久蜜桃网站 | 日韩爱爱网站 | 亚洲精色 | 亚洲视频中文 | 黄色片网站大全 | 国产美女主播精品一区二区三区 | 亚洲a在线观看 | 日韩精品视频在线免费观看 | 久久看片网站 | 黄色免费观看网址 | 免费成人看片 | 国产剧情一区二区 | 久久再线视频 | 国产 色 | 午夜三级在线 | 日韩在线无 | 激情www| 成人久久久久 | 五月婷婷激情五月 | 97视频亚洲 | 美女免费视频一区二区 | 午夜天天操| 国产黄色av | 中文字幕 在线看 | 国产亚洲精品久久久久久久久久 | 日韩 在线观看 | 在线小视频你懂的 | 久久综合九色99 | 中文字幕一区二区三区四区视频 | 999国产在线 | 午夜视频在线观看一区二区 | 国产成人精品亚洲日本在线观看 | 国产精品av电影 | 在线观看国产一区 | 成年人app网址 | 国产精品久久久久久久久久白浆 | 婷婷亚洲五月色综合 | 日韩在线观看视频网站 | 国产精品乱码久久久久 | 久久久久亚洲天堂 | 国产成人av综合色 | 91手机视频在线 | 国产主播大尺度精品福利免费 | 在线观看av小说 | 一个色综合网站 | 在线视频亚洲 | 韩日精品中文字幕 | 久久高清免费观看 | 中文字幕一区二区三区四区久久 | 超碰人人干人人 | 欧美超碰在线 | 午夜10000 | 日韩亚洲在线 | 亚洲天堂色婷婷 | 精品专区一区二区 | 久久久国产网站 | 欧美专区亚洲专区 | 国产日韩精品一区二区三区在线 | 久久久精品网站 | 国产特级毛片aaaaaa高清 | 91精品视频免费看 | 久久人人爽爽人人爽人人片av | 国产99久久久精品 | 久久免费视频在线观看 | 国产韩国日本高清视频 | 中文在线亚洲 | 久久国产露脸精品国产 | 久久久久激情视频 | 久久久国产一区二区三区四区小说 | 中文字幕乱码电影 | 91大神精品视频在线观看 | 伊人六月 | 久久99精品国产99久久 | 国产做爰视频 | 超级av在线 | 国产精品免费久久久久 | 日韩啪视频 | 久久这里只有精品视频99 | av九九| 国产在线国偷精品产拍免费yy | 久久国产亚洲视频 | 最近日韩中文字幕中文 | 久久伦理视频 | 97高清免费视频 | 手机在线小视频 | 免费一级片视频 | 波多野结衣在线视频免费观看 | 色一级片 | 在线免费黄色av | 人人爽夜夜爽 | 少妇bbbb揉bbbb日本 | 久久久电影网站 | 手机看片1042 | 超碰免费97 | 91刺激视频 | a在线观看国产 | 91成人精品一区在线播放69 | www.色五月 | 九草在线视频 | 国产999视频 | 在线观看一区二区视频 | 国产精品9区 | 成人av在线电影 | 亚州黄色一级 | 亚洲综合在线一区二区三区 | 中文字幕av最新 | 一区二区 不卡 | 狠狠躁夜夜a产精品视频 | 999抗病毒口服液 | 久久综合狠狠综合久久狠狠色综合 | 最新日韩精品 | 91大神在线观看视频 | 亚洲国产小视频在线观看 | 久久精品久久99 | 国产99视频在线观看 |