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

歡迎訪問 生活随笔!

生活随笔

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

windows

自己动手写文件系统

發(fā)布時間:2025/6/15 windows 34 豆豆
生活随笔 收集整理的這篇文章主要介紹了 自己动手写文件系统 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

這一課中,將創(chuàng)建一個磁盤分區(qū),并在她上面建立文件系統(tǒng)。文章看起來比較長,但是過程比較簡單。




大家都知道,硬盤最小的管理單位是扇區(qū),每個扇區(qū)512字節(jié)。有兩種方式定位訪問磁盤。一種是CHS模式(Cylinder, Head,?Sector),她反映了磁盤的物理結構。扇區(qū)(Sector)是磁盤可以直接訪問的最小數據單位(即一個塊)。柱面(Cylinder)也成為磁軌,通常一個1.44M軟盤每柱面有18扇區(qū),而硬盤每柱面含有64扇區(qū)。磁頭(Head)表示有多少個盤面(通常每個盤面使用兩個磁頭,正反盤面各一個)。關于CHS相關的資料網絡上可以搜索出很多,這里不贅述。
例如我建立了一個100M字節(jié)的磁盤映象文件,在bocsh可以這樣定義它的結構:

ata0-master: type=disk, path="minix.img", mode=flat, cylinders=200, heads=16, spt=63

計算:16 * 64 * 200 * 512 =?100 M(512是每扇區(qū)字節(jié)數)

很明顯,在CHS模式下,磁盤容量最大可表示為??柱面數?*?每柱面扇區(qū)數?*?磁頭數?* 512.?注意:柱面和磁頭從0開始計數,而扇區(qū)數則從1開始。不要問我為什么。


另一種硬盤訪問模式叫LBA(Linear Block Addressing:線性塊尋址模式),它是物理硬盤的邏輯尋址方式。和線性內存尋址一樣,LBA用線性地址來定位扇區(qū)(這樣,柱面和磁頭就不會用到了)。這種方式非常簡單,但是驅動使用的是CHS模式,所以需要進行轉換(LBA也是從0開始計數的)。

LBA =(C-Cs)*PH*PS+(H-Hs)*PS+(S-Ss)
磁盤大小LBA值?= C *?H *?S - 1

方向計算(由LBA獲取CHS:下面的公式好像有問題,讀者最好在網上找到正確的資料):
C = LBA / (heads per cylinder * sectors per track)
temp = LBA % (heads per cylinder * sectors per track)
H = temp / sectors per track
S = temp % sectors per track + 1

CHS跟LBA之間,其實相當于seg:offset與線形地址的關系。不同點在于:

  • CHS是三維的,seg:offset是二維的
  • LBA的空間更大2^32,CHS空間很小2^24(不考慮sector不能取0的問題)
  • 如果你使用vmware建立一個虛擬磁盤,可以在啟動時進入bios看到該虛擬磁盤的?CHS?和?LBA?信息。

    下面三個demo對應的源程序分別為dpt(分區(qū)表disk partition table),fs和root目錄。


    我們可以定義一個數據結構:


    07/dpt/hd.c

    struct HD_PARAM {
    ??? unsigned int cyl;
    ??? unsigned int head;
    ??? unsigned int sect;
    } HD0 = {208, 16, 63};????????//?定義一個104M磁盤

    ?

    由于skelix使用LBA尋址,所以需要把LBA地址轉換成CHS地址:
    ??? unsigned int cyl = lba / (HD0.head * HD0.sect);
    ??? unsigned int head = (lba % (HD0.head * HD0.sect)) / HD0.sect;
    ??? unsigned int sect = (lba % (HD0.head * HD0.sect)) % HD0.sect + 1;


    現在已經得到chs地址了,我們將通過?0x1F0-0x1F7端口來訪問硬盤。

    (這些繁文縟節(jié)一點也不好玩,但是我又不得不講)


    07/include/hd.h

    #define HD_PORT_DATA??? ???? 0x1f0
    #define HD_PORT_ERROR??? ??? 0x1f1
    #define HD_PORT_SECT_COUNT?? 0x1f2
    #define HD_PORT_SECT_NUM???? 0x1f3
    #define HD_PORT_CYL_LOW??? ? 0x1f4
    #define HD_PORT_CYL_HIGH???? 0x1f5
    #define HD_PORT_DRV_HEAD???? 0x1f6
    #define HD_PORT_STATUS??? ?? 0x1f7
    #define HD_PORT_COMMAND??? ? 0x1f7

    ?

    另人恐慌,不過可以很清楚的來個分組。我們從0x1f0端口讀取數據,

    如果發(fā)生了錯誤就從0x1f1讀取錯誤碼,

    0x1f2和0x1f3端口設置扇區(qū)號,

    0x1f4和0x1f5端口設置柱面號,

    0x1f6端口設置磁頭號。

    端口0x1f7用于讀硬盤狀態(tài)或者寫操作命令。


    寫0x1f7端口常用的命令字如下:
    #define HD_READ??? ??? ??? ? 0x20????????//?讀硬盤
    #define HD_WRITE??? ??? ???? 0x30????????//?寫硬盤

    通過上面定義,我們可以粗略的認為通過以下幾個步驟訪問硬盤,

    (為了簡化步驟,暫時不做一些錯誤檢測)

    下面這個函數hd_rw就是用來操作硬盤的接口。


    07/hd.c

    ??? while ((inb(HD_PORT_STATUS)&0xc0)!=0x40)
    ??? //?等待硬盤狀態(tài),直到可以寫或讀為止,狀態(tài)字節(jié)說明如下:

    Bit 7

    控制器忙

    Bit 6

    驅動器準備就緒

    Bit 5

    寫出錯

    Bit 4

    尋道結束

    Bit 3

    數據請求復位

    Bit 2

    ECC效驗錯誤

    Bit 1

    硬盤已收到索引

    Bit 0

    命令執(zhí)行錯誤

    ??? outb(sects_to_access, HD_PORT_SECT_COUNT); //?準備讀或寫多少個扇區(qū)

    ?

    ??? outb(sect, HD_PORT_SECT_NUM);????????????? //?發(fā)送chs地址
    ??? outb(cyl, HD_PORT_CYL_LOW);
    ??? outb(cyl>>8, HD_PORT_CYL_HIGH);
    ??? outb(0xa0|head, HD_PORT_DRV_HEAD);???????? // a0是第一塊硬盤

    Bits 7-5

    必須是?101b

    Bit 4

    HD0(=0第一塊硬盤), HD1(=1第二塊硬盤)

    Bits 3-0

    磁頭號

    ?

    ??? outb(com, HD_PORT_COMMAND);????????????????//?硬盤操作命令

    HD_READ=0x20

    如果不成功會反復讀

    HD_WRITE=0x30

    如果不成功會反復寫

    ??? if (com == HD_READ)
    ??? ??? insl(HD_PORT_DATA, buf, sects_to_access<<7);
    ??? else if (com == HD_WRITE)
    ??? ??? outsl(buf, sects_to_access<<7, HD_PORT_DATA);

    ????//?說明:insl和outsl是從io端口讀寫一串數據的宏匯編指令,

    ????//?這里使用的是pio方式,mdma和udma不作討論


    ??? while ((inb(HD_PORT_STATUS)&0x80));????????//?等待完成

    事實上,這只是最簡單的處理流程,連錯誤檢測都沒有。雖然是pio方式,

    仍然可以使用中斷,以避免上面的while循環(huán)等待,而減少內核浪費的時間。

    不過skelix不準備做這么復雜。

    ?

    磁盤分區(qū)表(disk partitiontable,以下簡稱dpt)

    現在我們有能力訪問硬盤的扇區(qū)了,需要把這些扇區(qū)管理起來。硬盤的第一個扇區(qū)必須包含硬盤分區(qū)表。這個分區(qū)表從第一個扇區(qū)的0x1be偏移開始,長度是64字節(jié)。最多可以包含4個分區(qū)(本文不考慮邏輯分區(qū),都使用主分區(qū))。這4個分區(qū)的相對偏移分別為0x1be, 0x1ce, 0x1de, 0x1fe,第一個扇區(qū)的最后兩個字節(jié)必須是0xaa55。

    每個分區(qū)項的格式如下:

    Byte? 0

    引導標識

    Bytes 1-3

    CHS?信息

    Byte? 4

    分區(qū)類型

    Bytes 5-7

    CHS?信息

    Bytes 8-11

    該分區(qū)的起始扇區(qū)號

    Bytes 12-16

    扇區(qū)數量

    ?

    第0個字節(jié)是引導標識,如果值為0x80標識可引導。對于一塊硬盤來說,只有一個分區(qū)是可以引導的。第4個字節(jié)定義分區(qū)類型,例如FAT32, NTFS, ext2等。有一篇文章http://www.osdever.net/documents/partitiontypes.php?the_id=35,里面定義了常見的分區(qū)類型。

    從上面的表可以看到dpt項有兩種方式定位扇區(qū),一種是通過字節(jié)1~3和字節(jié)5~7中的CHS信息,另一種是字節(jié)8~16的LBA信息。隨便用哪一種都是一樣的,在本文中使用LBA方式,所以我不準備解釋字節(jié)1~3和字節(jié)5~7的具體格式了。

    現在我們來建立分區(qū)表:
    07/dpt/hd.c

    ?

    static void
    setup_DPT(void) {
    ??? unsigned char sect[512] = {0};
    ??? sect[0x1be] = 0x80;?????????????//?第一個分區(qū)可引導

    ??? sect[0x1be + 0x04] = FST_FS;????//?自定義的數據分區(qū)類型
    ??? *(unsigned long *)&sect[0x1be + 0x08] = 1;
    ??? *(unsigned long *)&sect[0x1be + 0x0c] = 85*1024*2; /* 85MB */

    ??? sect[0x1ce + 0x04] = FST_SW;????//?自定義的交換分區(qū)類型,后續(xù)課程使用
    ??? *(unsigned long *)&sect[0x1ce + 0x08] = 85*1024*2+1;
    ??? *(unsigned long *)&sect[0x1ce + 0x0c] = 16*1024*2; /* 16MB */

    ??? sect[0x1fe] = 0x55;
    ??? sect[0x1ff] = 0xaa;

    ??? hd_rw(0, HD_WRITE, 1, sect);????//?寫入磁盤

    ????//?寫到扇區(qū)0,扇區(qū)數為1,sect是寫入緩沖
    }

    現在,我們在啟動的過程中把分區(qū)表信息打印出來:
    07/dpt/hd.c

    void
    verify_DPT(void) {
    ??? unsigned char sect[512];
    ??? unsigned i = 0;
    ??? unsigned int *q = (unsigned int *)(HD0_ADDR);

    ????//?變量q存放讀出的分區(qū)表(起始扇區(qū)號和扇區(qū)數量)數組

    ??? hd_rw(0, HD_READ, 1, sect);
    ??? if ((sect[0x1fe]==0x55) && (sect[0x1ff]==0xaa)) {
    ??????? unsigned char *p = &sect[0x1be];
    ??????? char *s;
    ??? ??? kprintf(KPL_DUMP, "?? | Bootable | Type????? | Start Sector | Capacity \n");

    ??????? for (i=0; i<4; ++i) {
    ??? ??? ??? kprintf(KPL_DUMP, " %d ", i);
    ??????????? if (p[0x04] == 0x00) {
    ??????????????? kprintf(KPL_DUMP, "| Empty\n");
    ??? ??? ??? ??? p += 16;
    ??? ??? ??? ??? q += 2;
    ??? ??? ??? ??? continue;
    ??? ??? ??? }
    ??? ??? ??? if (p[0x00] == 0x80)
    ??? ??? ??? ??? s = "| Yes????? ";
    ??? ??? ??? else
    ??? ??? ??? ??? s = "| No?????? ";

    ??????????? kprintf(KPL_DUMP, s);
    ??? ??? ??? /* system indicator at offset 0x04 */
    ??? ??? ??? if (p[0x04] == FST_FS) {
    ??? ??? ??? ??? kprintf(KPL_DUMP, "| Skelix FS ");
    ??? ??? ??? } else if (p[0x04] == FST_SW) {
    ??? ??? ??? ??? kprintf(KPL_DUMP, "| Skelix SW ");
    ??? ??? ??? } else
    ??? ??? ??? ??? kprintf(KPL_DUMP, "| Unknown?? ", *p);

    ??????????? /* starting sector number */
    ??? ??? ??? *q++ = *(unsigned long *)&p[0x08];
    ??? ??? ??? kprintf(KPL_DUMP, "| 0x%x?? ", *(unsigned long*)&p[0x08]);
    ??? ??? ??? /* capacity */
    ??? ??? ??? *q++ = *(unsigned long*)&p[0x0c];
    ??? ??? ??? kprintf(KPL_DUMP, "| %dM\n", (*(unsigned long*)&p[0x0c]*512)>>20);

    ??????????? //?保存到內存中,32字節(jié)偏移,32字節(jié)長度

    ??????????? p += 16;
    ??? ??? }
    ??? } else {
    ??? ??? kprintf(KPL_DUMP, "No bootable DPT found on HD0\n");
    ??? ??? kprintf(KPL_DUMP, "Creating DPT on HD0 automaticly\n");
    ??? ??? kprintf(KPL_DUMP, "Creating file system whatever you want it or not!!\n");
    ??? ??? setup_DPT();
    ??? ??? verify_DPT();
    ??? }
    }

    在我們編譯觀察結果之前,還需要修改任務函數task1_run?和?task2_run,因為它們會滾動屏幕把我們想要的結果覆蓋掉。
    07/init.c

    void
    do_task1(void) {
    ??? __asm__ ("incb 0xb8000+160*24+2");
    }

    void
    do_task2(void) {
    ??? __asm__ ("incb 0xb8000+160*24+4");
    }

    按例,還得改改Makefile,加入?hd.o?到?KERNEL_OBJS,?并在sti()?之前就調用?verify_DPT()函數:
    07/dpt/Makefile

    KERNEL_OBJS= load.o init.o isr.o timer.o libcc.o scr.o kb.o task.o kprintf.o hd.o exceptions.o

    ?

    07/dpt/init.c

    ??? __asm__ ("ltrw??? %%ax\n\t"::"a"(TSS_SEL));
    ??? __asm__ ("lldt??? %%ax\n\t"::"a"(LDT_SEL));

    ??? kprintf(KPL_DUMP, "Verifing disk partition table....\n");
    ??? verify_DPT();?? ?? /* <<<<< Here it is */

    ??? sti();???????????? //?任務調度可以進行了

    ?

    編譯運行一把,OK!(最好使用一個未分區(qū)的磁盤映象來測試)


    文件系統(tǒng)

    分區(qū)已經建立,下一步就是組織各個分區(qū)上的文件系統(tǒng)。雖然我們可以做到訪問扇區(qū)了,但是對于訪問文件卻是不方便的。需要做一些結構化的工作,為此定義了一個表示文件的數據結構:
    07/fs/include/fs.h

    ?

    #define FT_NML??? 1???????????? /* normal file */
    #define FT_DIR??? 2

    struct INODE {????????????????? /*?存放在硬盤里面,在inode區(qū)?*/
    ??? unsigned int i_mode;??? ??? /* file mode */
    ??? unsigned int i_size;??? ??? /* size in bytes */
    ??? unsigned int i_block[8];
    };
    *nix?用戶可能對inode比較敏感。現在我來一一解釋這個數據結構中的域,i_mode定義文件類型。FT_NML?表示這是一個普通文件,FT_DIR?表示是一個目錄。i_size?是文件大小,對于文件夾則是另外意思,后面將會講到。i_block的前6個整形表示文件的前6個扇區(qū)號,第七個表示二級指針扇區(qū)(即它指向一個扇區(qū),這個扇區(qū)里面存放文件后續(xù)部分使用扇區(qū)號),由?512 / 4 = 128?扇區(qū),表示文件接下來使用的128個扇區(qū)。128 * 512 = 64K。i_block數組的最后一個表示三級指針,最大可以表示?(512 / 4) * (512 / 4) * 512 = 8MB字節(jié)。所以這個i_block數組最大可以表示?3k + 64K +?8M文件的大小,雖然比較小,但是對于我們的85M?分區(qū)來說已經足夠了。最重要的是,它比較簡單。舉例來說把,這個文件節(jié)點使用了如下扇區(qū)序列:?13, 7, 9, 16, 257, 57, 3, ....., 35, 33, ....., 55, ......., 128, ....., 81.


    對于目錄(也是一種文件)來說,它以類似數組的形式組織:?{{file_name, file_inode}, {file_name, file_inode}, {file_name, file_inode}, },定義如下:


    07/fs/include/fs.h

    #define MAX_NAME_LEN 11

    struct DIR_ENTRY {????????????/?存放在硬盤里面,在data區(qū)?*/

    ??? char de_name[MAX_NAME_LEN];
    ??? int de_inode;
    };

    操作系統(tǒng)中的所有文件都有一個獨一無二的節(jié)點編號,如果有了這個節(jié)點號,就可以找到對應的文件。最開始的兩個文件永遠是"."?和?"..",表示當前目錄和上級目錄,如果我們切換到下級目錄,可以通過".."來回到上一級。"/"表示最上級目錄,它沒有父節(jié)點。
    舉例來說,我們需要定位到?/usr/doc/fvwm/TODO?文件,首先我們找到"/"文件,然后搜索這個文件項下面的doc文件,因為"/"是個目錄,所以先得到"/"目錄的節(jié)點編號,然后搜索指向的節(jié)點表。然后再搜索到fvwm目錄,并且在這個目錄的節(jié)點表中搜索到"TODO"這個文件,并通過"TODO"的節(jié)點編號來定位節(jié)點這個文件的節(jié)點數據結構。最后就可以訪問i_block數組了,也就是可以訪問這個文件了。怎么自己看的都昏菜了,s**t!

    還有兩個問題,一個是需要指定從哪里搜索節(jié)點號,我們在磁盤中組織所有節(jié)點為數組,并由節(jié)點號來索引節(jié)點數組。另一個問題是,"/"沒有父節(jié)點,需要知道"/"存放在什么地方,這個也好辦,就放在節(jié)點數組的第一項好了。

    文件名聲明成12字節(jié),這樣每個節(jié)點將占用16字節(jié)(另4字節(jié)是節(jié)點編號),這樣方便磁盤IO之后的一些操作。當磁盤使用一段時間后,有的節(jié)點使用了,有的節(jié)點沒有使用,那怎么知道呢?一種方法是建立一個位圖表,每個位表示inode數組中的一項。

    07/fs/include/fs.h

    struct SUPER_BLOCK {
    ??? unsigned char sb_magic;????/*?分區(qū)類型?FST_FS?或?FST_SW *'
    ????
    ??? unsigned int sb_start;???? /* DPT 0x08:?起始扇區(qū)?*/
    ??? unsigned int sb_blocks;??? /* DPT 0x0c:?扇區(qū)數量?*/

    ??? unsigned int sb_dmap_blks;
    ??? unsigned int sb_imap_blks;
    ??? unsigned int sb_inode_blks;
    };

    ?

    這個超級塊的數據結構用來管理各個分區(qū)。例如,下面是一個磁盤分區(qū):
    ?________________________________________________________

    |??|?\\\\?| ?|?\\\\?|??|????? ?data????????? |?
    ?--------------------------------------------------------

    每個分區(qū)的第一個扇區(qū)(藍色)是boot secotr,我不打算使用它,一個扇區(qū)大小。

    第二個扇區(qū)(綠色)是超級塊(super block,以下簡稱sb),一個扇區(qū)大小。

    黑色是dmap,336個扇區(qū)大小。

    紅色是imap,一個扇區(qū)大小。

    灰色是inodes,將占有342個block,即342 * 8?個扇區(qū)大小。

    為了管理這個85M分區(qū),我們額外花了?1.5M?的空間。

    ?

    在verify_fs()函數中定義了超級塊(局部變量)sb,為了方便訪問定義了一些宏,獲取相對整個硬盤的絕對地址(LBA):

    07/fs/incude/fs.h

    #define ABS_BOOT_BLK(sb)??? ??? ((sb).sb_start)
    #define ABS_SUPER_BLK(sb)??? ??? ((ABS_BOOT_BLK(sb))+1)
    #define ABS_DMAP_BLK(sb)??? ??? ((ABS_SUPER_BLK(sb))+1)
    #define ABS_IMAP_BLK(sb)??? ??? ((ABS_DMAP_BLK(sb))+(sb).sb_dmap_blks)
    #define ABS_INODE_BLK(sb)??? ??? ((ABS_IMAP_BLK(sb))+(sb).sb_imap_blks)
    #define ABS_DATA_BLK(sb)??? ??? ((ABS_INODE_BLK(sb))+INODE_BLKS)

    ?

    說明:dmap(data map)存放的是扇區(qū)使用位圖

    ????? imap(inode map)存放inode使用位圖。

    ????? inodes存放節(jié)點表。

    為了方便,一些位的操作函數如下:

    07/fs/fs.c

    void
    setb(void *s, unsigned int i) {
    ??? unsigned char *v = s;
    ??? v += i>>3;????????????// i的單位由位轉換成字節(jié)
    ??? *v |= 1<<(7-(i%8));
    }

    void
    clrb(void *s, unsigned int i) {
    ??? unsigned char *v = s;
    ??? v += i>>3;
    ??? *v &= ~(1<<(7-(i%8)));
    }

    int
    testb(void *s, unsigned int i) {
    ??? unsigned char *v = s;
    ??? v += i>>3;
    ??? return (*v&(1<<(7-(i%8)))) !=0;
    }

    例如,設置緩沖區(qū)sect的1796位,可以使用setb(sect, 1796)

    init.c在調用verify_DPT();創(chuàng)建分區(qū)表后,緊接著調用verify_fs();創(chuàng)建文件系統(tǒng):
    07/fs/fs.c

    void
    verify_fs(void) {
    ??? unsigned int *q = (unsigned int *)(HD0_ADDR);
    ??? unsigned char sect[512] = {0};
    ??? struct SUPER_BLOCK sb;
    ??? unsigned int i = 0, j = 0, m = 0, n = 0;

    ??? /*?讀取超級塊?*/
    ??? sb.sb_start = q[0];
    ??? hd_rw(ABS_SUPER_BLK(sb), HD_READ, 1, &sb);

    ????//?很難想象這段代碼不越界,正好越界到sect上了?昏菜!


    ??? /*?判斷超級塊是否正確,不是就創(chuàng)建文件系統(tǒng)?*/
    ??? if (sb.sb_magic != FST_FS) {
    ??????? kprintf(KPL_DUMP, "Partition 1 does not have a valid file system\n");
    ??? ??? kprintf(KPL_DUMP, "Creating file system\t\t\t\t\t\t\t? ");
    ??? ??? sb.sb_magic = FST_FS;
    ??? ??? sb.sb_start = q[0];
    ??? ??? sb.sb_blocks = q[1];
    ??? ??? sb.sb_dmap_blks = (sb.sb_blocks+0xfff)>>12;
    ??? ??? sb.sb_imap_blks = INODE_BIT_BLKS;
    ??? ??? sb.sb_inode_blks = INODE_BLKS;
    ??? ??? hd_rw(ABS_SUPER_BLK(sb), HD_WRITE, 1, &sb);

    ?

    ????????// dmap位圖中,每個位表示1個扇區(qū),也就是說dmap中每個扇區(qū)可以標識512 * 8扇區(qū)。

    ????????//?另外,我們把inode位圖大小固定,即使用1個扇區(qū)。
    ??????? /*?初始化dmap位圖?*/
    ??? ??? n = ABS_DMAP_BLK(sb);
    ??? ??? j = sb.sb_dmap_blks+sb.sb_imap_blks+sb.sb_inode_blks+2;
    ??? ??? memset(sect, 0xff, sizeof sect/sizeof sect[0]);
    ??? ??? for (i=j/(512*8); i>0; --i) {
    ??? ??? ??? hd_rw(n++, HD_WRITE, 1, sect);
    ??? ??? ??? m += 4096;
    ??? ??? }
    ??? ??? m += 4096;
    ??? ??? for (i=j%(512*8); i<512*8; ++i) {
    ??? ??? ??? clrb(sect, i);
    ??? ??? ??? --m;
    ??? ??? }
    ??? ??? hd_rw(n++, HD_WRITE, 1, sect);

    ??? ??? memset(sect, 0, sizeof sect/sizeof sect[0]);
    ??? ??? for (i=sb.sb_imap_blks-(n-ABS_DMAP_BLK(sb)); i>0; --i)
    ??? ??? ??? hd_rw(n++, HD_WRITE, 1, sect);

    ??? ??? /*?初始化inode?位圖?*/
    ??? ??? for (i=sb.sb_imap_blks; i>0; --i)
    ??? ??? ??? hd_rw(ABS_IMAP_BLK(sb)+i-1, HD_WRITE, 1, sect);

    ??? ??? /*?初始化inode?數組?*/
    ??? ??? for (i=sb.sb_inode_blks; i>0; --i)
    ??? ??? ??? hd_rw(ABS_INODE_BLK(sb)+i-1, HD_WRITE, 1, sect);
    ??? ??? kprintf(KPL_DUMP, "[DONE]");
    ??? }
    ??? q += 2;

    ??? kprintf(KPL_DUMP, "0: Type: FST_FS ");
    ??? kprintf(KPL_DUMP, "start at: %d ", sb.sb_start);
    ??? kprintf(KPL_DUMP, "blocks: %d ", sb.sb_blocks);
    ??? kprintf(KPL_DUMP, "dmap: %d ", sb.sb_dmap_blks);
    ??? kprintf(KPL_DUMP, "imap: %d ", sb.sb_imap_blks);
    ??? kprintf(KPL_DUMP, "inodes: %d\n", sb.sb_inode_blks);

    ??? /*?初始化交互分區(qū)?*/
    ??? sb.sb_start = q[0];
    ??? hd_rw(ABS_SUPER_BLK(sb), HD_READ, 1, &sb);
    ??? if (sb.sb_magic != FST_SW) {

    ????????//?注意,和數據分區(qū)不同(每個塊占有1個扇區(qū)),

    ????????//?交互分區(qū)每個塊占有8個扇區(qū),即4096字節(jié),和內存頁對齊

    ??????? kprintf(KPL_DUMP, "\nPartition 2 does not have a valid file system\n");
    ??? ??? kprintf(KPL_DUMP, "Creating file system\t\t\t\t\t\t\t? ");
    ??? ??? sb.sb_magic = FST_SW;
    ??? ??? sb.sb_start = q[0];
    ??? ??? sb.sb_blocks = q[1];
    ??? ??? sb.sb_dmap_blks = (sb.sb_blocks)>>15;??? /* 1 bits == 4K page */
    ??? ??? hd_rw(ABS_SUPER_BLK(sb), HD_WRITE, 1, &sb);
    ??? ??? kprintf(KPL_DUMP, "[DONE]");????
    ??? }
    ??? /*?初始化數據位圖?*/
    ??? n = ABS_DMAP_BLK(sb);
    ??? j = sb.sb_dmap_blks+2;
    ??? memset(sect, 0xff, sizeof sect/sizeof sect[0]);
    ??? for (i=j/(512*8); i>0; --i) {
    ??? ??? hd_rw(n++, HD_WRITE, 1, sect);
    ??? ??? m += 4096;
    ??? }
    ??? m += 4096;
    ??? for (i=j%(512*8); i<512*8; ++i) {
    ??? ??? clrb(sect, i);
    ??? ??? --m;
    ??? }
    ??? hd_rw(n++, HD_WRITE, 1, sect);

    ??? kprintf(KPL_DUMP, "1: Type: FST_SW ");
    ??? kprintf(KPL_DUMP, "start at: %d ", sb.sb_start);
    ??? kprintf(KPL_DUMP, "blocks: %d ", sb.sb_blocks);
    ??? kprintf(KPL_DUMP, "dmap: %d, presents %d 4k-page\n",
    ??? ??? ??? sb.sb_dmap_blks, sb.sb_blocks>>3);
    }
    最后修改Makefile,然后make clean && make dep && make
    07/fs/Makefile

    KERNEL_OBJS= load.o init.o isr.o timer.o libcc.o scr.o kb.o task.o kprintf.o hd.o exceptions.o fs.o
    編譯,運行。

    ?

    root?根目錄

    最后一件事情建立根目錄。"/"是所有文件的根目錄,所以我們一開始就必須設置好它。"/"文件永遠使用inode 0,這樣skelix內核才知道怎樣找到它。然后再讀取"/"文件的內容,也就是DIR_ENTRY結構數組。為了更方便的操作,我們先完成一些基礎函數,用來操作blocks和inodes。

    07/root/fs.c

    static struct INODE iroot = {FT_DIR, 2*sizeof(struct DIR_ENTRY), {0,}};

    unsigned int
    alloc_blk(struct SUPER_BLOCK *sb) {

    ????//?根據dmap位圖查找空閑的扇區(qū),返回LBA地址(從1開始)
    ??? unsigned int i = 0, j = 0, n = 0, m = 0;
    ??? unsigned char sect[512] = {0};

    ??? n = ABS_DMAP_BLK(*sb);
    ??? for (; i<sb->sb_dmap_blks; ++i) {
    ??? ??? hd_rw(n, HD_READ, 1, sect);
    ??? ??? for (j=0; j<512*8; ++j) {
    ??? ??? ??? if (testb(sect, j)) {
    ??? ??? ??? ??? ++m;
    ??? ??? ??? } else {??? ??? ??? /* gotcha */
    ??? ??? ??? ??? setb(sect, j);
    ??? ??? ??? ??? if (m >= sb->sb_blocks)
    ??? ??? ??? ??? ??? return 0;
    ??? ??? ??? ??? else {
    ??? ??? ??? ??? ??? hd_rw(n, HD_WRITE, 1, sect);
    ??? ??? ??? ??? ??? return ABS_BOOT_BLK(*sb) + m;
    ??? ??? ??? ??? }
    ??? ??? ??? }
    ??? ??? }
    ??? ??? ++n;
    ??? }
    ??? return 0;
    }

    void
    free_blk(struct SUPER_BLOCK *sb, unsigned int n) {

    ????//?釋放一個扇區(qū):設置dmap位圖中對應的位即可
    ??? unsigned char sect[512] = {0};
    ??? unsigned int t = (n-ABS_BOOT_BLK(*sb))/(512*8)+ABS_DMAP_BLK(*sb);
    ??? hd_rw(t, HD_READ, 1, sect);
    ??? clrb(sect, (n-ABS_BOOT_BLK(*sb))%(512*8));
    ??? hd_rw(t, HD_WRITE, 1, sect);
    }

    static int
    alloc_inode(struct SUPER_BLOCK *sb) {

    ????//?在imap表中中找一個空閑的項
    ??? unsigned char sect[512] = {0};
    ??? int i = 0;
    ??? hd_rw(ABS_IMAP_BLK(*sb), HD_READ, 1, sect);
    ??? for (; i<512; ++i) {
    ??? ??? if (! testb(sect, i)) {
    ??? ??? ??? setb(sect, i);
    ??? ??? ??? hd_rw(ABS_IMAP_BLK(*sb), HD_WRITE, 1, sect);
    ??? ??? ??? break;
    ??? ??? }
    ??? }
    ??? return (i==512)?-1:i;
    }

    static void
    free_inode(struct SUPER_BLOCK *sb, int n) {

    ????//?釋放inode項
    ??? unsigned char sect[512] = {0};
    ??? hd_rw(ABS_IMAP_BLK(*sb), HD_READ, 1, sect);
    ??? clrb(sect, n);
    ??? hd_rw(ABS_IMAP_BLK(*sb), HD_WRITE, 1, sect);
    }

    //?上面4個函數就是針對dmap和imap的操作(申請,釋放)

    static struct INODE *
    iget(struct SUPER_BLOCK *sb, struct INODE *inode, int n) {
    ??? unsigned char sect[512] = {0};
    ??? int i = n / INODES_PER_BLK;
    ??? int j = n % INODES_PER_BLK;

    ??? hd_rw(ABS_INODE_BLK(*sb)+i, HD_READ, 1, sect);
    ??? memcpy(inode, sect+j*sizeof(struct INODE), sizeof(struct INODE));
    ??? return inode;
    }

    static void
    iput(struct SUPER_BLOCK *sb, struct INODE *inode, int n) {
    ??? unsigned char sect[512] = {0};
    ??? int i = n/INODES_PER_BLK;
    ??? int j = n%INODES_PER_BLK;
    ??? hd_rw(ABS_INODE_BLK(*sb)+i, HD_READ, 1, sect);
    ??? memcpy(sect+j*sizeof(struct INODE), inode, sizeof(struct INODE));
    ??? hd_rw(ABS_INODE_BLK(*sb)+i, HD_WRITE, 1, sect);
    }

    //?上面兩個函數分別完成讀/寫磁盤指定下標號對應的inode節(jié)點到內存中。

    //?需要注意的是,這些函數對競態(tài)條件做處理,因為skelix僅內核讀寫硬盤。

    //?本文中暫時沒有用戶態(tài)的多任務。

    主流程如下:
    07/root/fs.c

    static void
    check_root(void) {
    ??? struct SUPER_BLOCK sb;
    ??? unsigned char sect[512] = {0};
    ??? struct DIR_ENTRY *de = NULL;

    ??? sb.sb_start = *(unsigned int *)(HD0_ADDR);
    ??? hd_rw(ABS_SUPER_BLK(sb), HD_READ, 1, sect);
    ??? memcpy(&sb, sect, sizeof(struct SUPER_BLOCK));
    ??? hd_rw(ABS_IMAP_BLK(sb), HD_READ, 1, sect);

    ????//?加載imap扇區(qū),判斷"/"目錄有沒有創(chuàng)建
    ??? if (! testb(sect, 0)) {????????????// "/"目錄必須使用inode 0,否則halt
    ??????? kprintf(KPL_DUMP, "/ has not been created, creating....\t\t\t\t\t? ");
    ??? ??? if (alloc_inode(&sb) != 0) {?? //?分配節(jié)點號:即imap位圖中的一位
    ??????????? kprintf(KPL_PANIC, "\n/ must be inode 0!!!\n");
    ??? ??? ??? halt();
    ??? ??? }

    ??????? iroot.i_block[0] = alloc_blk(&sb);????//?節(jié)點分配一個塊(一個扇區(qū))
    ??????? iput(&sb, &iroot, 0);???????????????? //?寫入節(jié)點

    ?

    ??????? de = (struct DIR_ENTRY *)sect;
    ??? ??? strcpy(de->de_name, ".");
    ??? ??? de->de_inode = 0;???????????????????? //?節(jié)點號為0
    ??????? ++de;
    ??? ??? strcpy(de->de_name, "..");
    ??? ??? de->de_inode = -1;????????????????????//?節(jié)點號為-1,這樣我們就知道是最上層目錄了
    ??????? hd_rw(iroot.i_block[0], HD_WRITE, 1, sect);????//?寫入"."?和?".."文件夾

    ??????? kprintf(KPL_DUMP, "[DONE]");
    ??? }
    ??? iget(&sb, &iroot, 0);
    ??? hd_rw(iroot.i_block[0], HD_READ, 1, sect);
    ??? de = (struct DIR_ENTRY *)sect;

    ??? if ((strcmp(de[0].de_name, ".")) || (de[0].de_inode) ||
    ??? ??? (strcmp(de[1].de_name, "..")) || (de[1].de_inode) != -1) {
    ??? ??? kprintf(KPL_PANIC, "File system is corrupted!!!\n");
    ??? ??? halt();
    ??? }
    }

    ?

    //?再來一個函數打印文件的相關信息
    static void
    stat(struct INODE *inode) {
    ??? unsigned int i = 0;
    ??? char sect[512] = {0};
    ??? struct DIR_ENTRY *de;

    ??? kprintf(KPL_DUMP, "======== stat / ========\n");
    ??? switch (inode->i_mode) {
    ??? case FT_NML:
    ??? ??? kprintf(KPL_DUMP, "File, ");
    ??? ??? break;
    ??? case FT_DIR:
    ??? ??? kprintf(KPL_DUMP, "Dir,? ");
    ??? ??? break;
    ??? default:
    ??? ??? kprintf(KPL_PANIC, "UNKNOWN FILE TYPE!!");
    ??? ??? halt();
    ??? }

    ??? kprintf(KPL_DUMP, "Size: %d, ", inode->i_size);
    ??? kprintf(KPL_DUMP, "Blocks: ");
    ??? for (; i<8; ++i)????????//?打印inode標識使用的扇區(qū)
    ??????? kprintf(KPL_DUMP, "%d, ", inode->i_block[i]);

    ??? hd_rw(inode->i_block[0], HD_READ, 1, sect);
    ??? switch (inode->i_mode) {
    ??? case FT_DIR:
    ??? ??? kprintf(KPL_DUMP, "\nName\tINode\n");
    ??? ??? de = (struct DIR_ENTRY *)sect;????//?打印子目錄(只一個扇區(qū))
    ??????? for (i=0; i<inode->i_size/sizeof(struct DIR_ENTRY); ++i) {
    ??? ??? ??? kprintf(KPL_DUMP, "%s\t%x\n", de[i].de_name, de[i].de_inode);
    ??? ??? }
    ??????? break;
    ??? default:
    ??? ??? break;
    ??? }
    }

    現在,我們把上面的函數整理到程序中
    void
    verify_dir(void) {
    ??? unsigned char sect[512] = {0};
    ??? unsigned int *q = (unsigned int *)(HD0_ADDR);
    ??? struct INODE inode;
    ??? struct SUPER_BLOCK sb;

    ??? sb.sb_start = q[0];
    ??? hd_rw(ABS_SUPER_BLK(sb), HD_READ, 1, sect);
    ??? check_root();
    ??? memcpy(&sb, sect, sizeof(struct SUPER_BLOCK));
    ??? stat(iget(&sb, &inode, 0));
    }

    07/root/init.c

    void?
    init(void) {

    ????……

    ??? kprintf(KPL_DUMP, "Verifing disk partition table....\n");
    ??? verify_DPT();
    ??? kprintf(KPL_DUMP, "Verifing file systes....\n");
    ??? verify_fs();
    ??? kprintf(KPL_DUMP, "Checking / directory....\n");
    ????verify_dir();

    ????……

    }

    不需要再編輯Makefile了,直接make && run好了。

    總結

    以上是生活随笔為你收集整理的自己动手写文件系统的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    www色片 | 日本夜夜草视频网站 | 成人av网站在线观看 | 国产中文字幕视频在线观看 | 色狠狠久久av五月综合 | 欧美天天干 | 婷婷丁香六月天 | 日韩高清二区 | 国产成人1区 | 国产精品美女久久久免费 | 亚洲午夜久久久久久久久久久 | 99视频99| 国产一级高清视频 | 日本精品久久久久 | 99久久精品午夜一区二区小说 | 久草在线视频免费资源观看 | 欧美先锋影音 | 婷婷伊人综合亚洲综合网 | 97av在线视频| 九九热有精品 | 天天色成人网 | 最近免费观看的电影完整版 | 麻豆影视在线免费观看 | 精品96久久久久久中文字幕无 | 丝袜少妇在线 | 91少妇精拍在线播放 | 天天摸天天舔天天操 | 日韩精品视频在线免费观看 | 精品久久久久久久久中文字幕 | www天天操| 少妇自拍av| 99精品视频在线播放观看 | 久草在线手机视频 | 97人人添人澡人人爽超碰动图 | 国产九九热 | 午夜在线资源 | 亚洲欧美日韩精品一区二区 | 国产一卡二卡四卡国 | 欧美激情视频一二区 | 日日摸日日添夜夜爽97 | 成人免费观看视频网站 | 99免在线观看免费视频高清 | 色资源网免费观看视频 | 人人射人人澡 | 午夜视频在线观看欧美 | 国产精品一区二区av影院萌芽 | 97视频免费看| 婷婷久久综合网 | 99热在线国产 | 亚洲国产欧洲综合997久久, | 久久九九国产精品 | 香蕉视频久久久 | 亚洲综合少妇 | 久久只精品99品免费久23小说 | 久久久网 | 午夜久久久久久久 | 国产免费中文字幕 | 一区电影 | 探花视频在线观看 | 夜色资源网 | 国产中年夫妇高潮精品视频 | 九九在线免费视频 | 五月婷婷一区 | 欧美日韩国产三级 | 精品在线视频观看 | 综合色在线观看 | 91黄色在线观看 | 成人免费在线视频观看 | 欧美性生活免费看 | 五月综合 | 成人夜晚看av | 久久久蜜桃一区二区 | 国产精品系列在线 | 日韩 国产 | 国产精品久久久久久久久费观看 | 99精品视频在线观看 | 亚洲资源在线网 | 国产精品18久久久久久久网站 | 国产打女人屁股调教97 | 欧美日韩一区二区三区不卡 | 黄网站免费看 | 亚洲成人av一区二区 | 国产精品一区二区av影院萌芽 | 国内精品久久久久影院一蜜桃 | 免费h在线观看 | 99精品视频播放 | 高清精品在线 | 欧洲性视频 | 免费三级大片 | 激情视频免费在线观看 | 天天天天天天天天操 | 国产精品高 | 久久综合综合久久综合 | 亚洲精品美女久久久久 | 久久免费的精品国产v∧ | 久久精品99国产精品酒店日本 | 韩国视频一区二区三区 | 久久只有精品 | 91高清免费 | 国产精品va最新国产精品视频 | www黄| 香蕉视频在线网站 | 激情婷婷av | 69精品| 91麻豆国产福利在线观看 | 一区精品在线 | 天天操天天操天天 | 日韩成人在线免费观看 | 亚洲aⅴ免费在线观看 | 欧美另类tv | 久久综合欧美精品亚洲一区 | 欧美一级黄色网 | 天天爽夜夜爽人人爽曰av | 国产中文在线视频 | av在线电影播放 | 国产欧美精品一区二区三区 | 久久九九影视 | 色婷婷电影 | 成人av电影在线播放 | 六月丁香社区 | 日韩欧美视频免费在线观看 | 精品久久一区二区三区 | 香蕉影院在线 | 精品毛片一区二区免费看 | www.久艹| 久久精品99精品国产香蕉 | 国产午夜在线观看 | 最新日韩精品 | 国产精品 中文在线 | 国产高清av在线播放 | 日韩高清激情 | 97超碰在线免费观看 | 欧美性猛片, | 日韩久久精品一区二区 | 亚洲精品一区二区在线观看 | 亚洲黄色片在线 | 天天干天天射天天爽 | 91九色蝌蚪视频在线 | 天天操网址| 激情深爱五月 | 丁香六月婷婷开心婷婷网 | 欧美亚洲一区二区在线 | 三日本三级少妇三级99 | 人人干人人爽 | 国产精品久久久久一区 | 婷婷亚洲最大 | 成人久久久久久久久久 | 97超碰在线视 | 国产第页 | 成人中文字幕在线 | 成人91在线 | 欧美另类高潮 | 日韩一级电影网站 | 国产96av | 狠狠色狠狠色终合网 | 婷婷在线观看视频 | 在线а√天堂中文官网 | 一级α片 | 日本黄色免费在线 | 中文字幕在线观看免费观看 | 人操人| 在线免费成人 | 久久久久久久久久网 | 亚洲一级黄色av | 18久久久 | 精品在线二区 | 日韩欧美视频免费看 | 久久av在线播放 | 91视频免费看网站 | 国产尤物在线 | 国产自产高清不卡 | 久久久www成人免费毛片麻豆 | 手机在线看永久av片免费 | 国产精品久久久毛片 | 亚洲网站在线 | 国产免费人人看 | 91麻豆传媒| 96av在线| 日韩精品免费专区 | 久久国产精品久久久久 | av黄色影院 | 国产精品久久久久久久免费大片 | 亚州成人av在线 | 日韩欧美一区二区三区在线 | av在线播放快速免费阴 | 欧美a√大片 | 色综合亚洲精品激情狠狠 | 狠狠干激情 | 精品电影一区 | 最近久乱中文字幕 | 免费福利片2019潦草影视午夜 | 色视频在线免费观看 | 亚洲激情综合 | 婷婷在线免费 | 99久久精品国产观看 | 天天综合网久久综合网 | 亚洲精品在线观看不卡 | 婷婷久久一区二区三区 | 国产精品乱码在线 | av看片在线观看 | 久草在线资源视频 | 国产亚洲一区二区在线观看 | 日本三级香港三级人妇99 | 久久av免费观看 | 国产精华国产精品 | 欧美成人一二区 | 久久久久国产精品www | 中文字幕乱视频 | 激情网站 | 国产99久久久欧美黑人 | 激情五月五月婷婷 | 久草免费在线观看视频 | 久久精品美女视频网站 | 久久久久久久影院 | 在线观看视频你懂的 | 亚洲少妇影院 | 久久久久久美女 | 99久久日韩精品视频免费在线观看 | 欧洲视频一区 | 欧美在线1| 欧美色图狠狠干 | 99精品视频免费全部在线 | 91精品视屏 | 久久激情视频免费观看 | 国产一区二区免费看 | 久久综合久久综合这里只有精品 | 亚洲丝袜一区二区 | 日日夜夜中文字幕 | 男女啪啪视屏 | 亚洲国产精品成人精品 | 日本久久视频 | 久久官网 | 中文在线8新资源库 | 天天干.com| 中文字幕一区二 | 亚州国产视频 | 五月天婷婷在线视频 | 亚洲理论视频 | 制服丝袜一区二区 | 人人澡人人爽欧一区 | 久久久久久久综合色一本 | 91视频链接| 日韩电影中文字幕在线观看 | 免费av在线播放 | 一区二区三区电影 | 日韩精品免费一线在线观看 | 久草在线91 | av成人免费| 91视频下载 | 中文字幕免费观看全部电影 | 在线观看国产www | 国产精品成人av在线 | 免费一级片在线 | 日韩精品一区二区在线观看 | 99操视频 | 国产玖玖视频 | 日韩av午夜在线观看 | 特级大胆西西4444www | 人人爱夜夜操 | 日韩精品一区二区三区不卡 | 最近高清中文字幕在线国语5 | av三级在线免费观看 | 中文字幕精品三区 | 中文字幕在线看视频国产中文版 | 一级黄色片在线免费观看 | 国产精品你懂的在线观看 | 日韩高清三区 | 韩国av电影在线观看 | 国产精品综合在线 | av韩国在线 | 天天干天天色2020 | 日韩一级黄色大片 | 国产免费又爽又刺激在线观看 | 久草视频在线看 | 亚洲精品国产高清 | 国产精品久久久久久久av大片 | 精品国产一二区 | 国产精品美女久久久久久网站 | h网站免费在线观看 | 成人一级片在线观看 | 天天色天天操天天爽 | 久久综合狠狠 | 97成人资源 | 天天干天天拍天天操天天拍 | 久久av电影 | 在线成人av | 中文字幕在线播放日韩 | 99久久这里有精品 | 2023亚洲精品国偷拍自产在线 | 欧美国产三区 | 国产精品久久久久久妇 | 国产第页 | 欧美综合国产 | 国产亚洲精品中文字幕 | 激情伊人五月天 | 91秒拍国产福利一区 | 午夜精品久久久久久 | 欧美一区二区三区四区夜夜大片 | 亚州成人av在线 | 亚洲在线精品视频 | 91污污| 国产一卡在线 | 91日韩精品 | 亚洲精品一区二区精华 | 久久精品资源 | 人人舔人人爱 | 亚洲专区中文字幕 | 在线天堂中文www视软件 | 午夜在线免费观看 | 国产精品美女久久久 | 久久99久久99精品 | 日日干夜夜干 | 国产永久免费高清在线观看视频 | 日韩国产精品毛片 | www.午夜视频 | 久久九九影视网 | 99999精品视频 | 91.麻豆视频| 色五月成人 | 欧美极品xxx | 成人影音在线 | 91成人短视频在线观看 | 欧美国产日韩激情 | 国产黄色精品在线观看 | 日韩欧美成人网 | 日韩美视频 | 亚洲成av片人久久久 | 99久久99热这里只有精品 | 久久私人影院 | 五月婷婷香蕉 | 午夜免费在线观看 | 久久99国产视频 | 国产一区二区高清不卡 | 精品一区二区电影 | 国产最新在线视频 | 毛片www | www.亚洲在线| 成全在线视频免费观看 | 国产精品18p | 最近中文字幕在线 | 爱爱av网站| 一级黄毛片 | 午夜在线日韩 | 婷婷精品在线视频 | 国产不卡免费视频 | 国产中文字幕一区二区三区 | 久久字幕精品一区 | 国产成人三级三级三级97 | 日韩精品久久久久久久电影99爱 | 婷婷综合伊人 | 午夜电影中文字幕 | 五月天色中色 | 亚洲天堂香蕉 | 亚洲激情在线观看 | 欧美性粗大hdvideo | 日韩中文字幕视频在线 | 日韩伦理片一区二区三区 | 国产在线更新 | 久久久免费毛片 | 中文在线a∨在线 | 精品一二三四在线 | 日韩激情第一页 | 免费看国产a| 操一草| 色在线国产 | 丁香综合五月 | 精品久久久久久久久亚洲 | 婷婷视频| 日韩精品在线免费观看 | 91超在线 | 99热手机在线观看 | 日韩在线一区二区免费 | 99久久久久| 国产精品久久99综合免费观看尤物 | 草草草影院| 成人av片在线观看 | 最新国产精品久久精品 | 国产小视频在线免费观看视频 | 国产综合精品一区二区三区 | 色综合咪咪久久网 | 亚洲永久精品国产 | 91成人免费视频 | 久久综合狠狠综合 | 黄av资源 | 久久人人爽人人片av | 色综合网在线 | 91伊人影院 | 视频91在线 | 国产xvideos免费视频播放 | av高清免费在线 | 91九色自拍 | 国产精品一区二区三区视频免费 | 国产一区二区日本 | 午夜精品福利影院 | 97品白浆高清久久久久久 | 91视频久久久 | 在线直播av | 成人a v视频 | 天天爱天天舔 | 色综合五月天 | 欧美久草视频 | 久草电影在线 | 久久久免费毛片 | av短片在线观看 | 日韩电影中文字幕在线 | 成人理论电影 | 久久蜜桃av| 777视频在线观看 | 久久视频网址 | 欧美aaa大片 | 日韩欧美在线第一页 | 国产免费视频一区二区裸体 | 在线观看岛国片 | 国产欧美日韩一区 | 91亚洲视频在线观看 | 欧美日韩亚洲在线观看 | 丁香综合激情 | 久要激情网 | 中文字幕丰满人伦在线 | 免费看三级网站 | 欧美激情视频一二三区 | 久草久草在线 | 亚洲精品mv在线观看 | 精品国产一区二区三区日日嗨 | 黄视频色网站 | 精品欧美小视频在线观看 | 日韩欧美精品在线观看视频 | 亚洲韩国一区二区三区 | 九九免费观看视频 | 久久久久亚洲a | 综合网天天色 | 国产一区不卡在线 | 久久在线视频在线 | 国产成人精品一区二区三区在线观看 | 日本中文字幕网站 | 国产成人久久精品亚洲 | 国产在线精品区 | 国产精品精品久久久久久 | 99日精品 | 亚洲精品午夜久久久 | 久久不射电影院 | 国产精品第一 | 天天综合网天天 | 亚洲精品www久久久 www国产精品com | 欧美日韩免费观看一区二区三区 | 玖玖爱在线观看 | 国产精品九九九九九九 | 中文字幕一区二区三区乱码在线 | 成 人 黄 色 视频免费播放 | 免费a现在观看 | 国内精品久久久久久久久久久久 | 中文字幕一区二区三区在线观看 | 免费一级片在线观看 | 99精品视频免费观看 | 亚洲一区欧美激情 | 国产成在线观看免费视频 | 亚洲一二视频 | 国产成人久久精品亚洲 | 久草久草在线 | 激情影院在线观看 | 亚洲闷骚少妇在线观看网站 | 国产亚洲人 | 国产亚洲va综合人人澡精品 | 国产成人精品亚洲精品 | 成人亚洲网 | 久久av中文字幕片 | 亚洲午夜精品一区 | 99热只有精品在线观看 | 美女视频一区二区 | 国产精品高潮呻吟久久av无 | 日韩免费福利 | 在线亚洲小视频 | 欧美福利网站 | 欧美精品在线观看免费 | 免费黄色在线播放 | 日韩欧美有码在线 | 精品一区免费 | 国产精品欧美久久久久天天影视 | 在线免费视频一区 | 韩国视频一区二区三区 | 久久久亚洲国产精品麻豆综合天堂 | 亚洲综合视频网 | 日韩中文字幕电影 | 久久综合九色九九 | 亚洲精品综合欧美二区变态 | 日韩欧美国产精品 | 中文字幕免费不卡视频 | 青草视频在线播放 | 香蕉成人在线视频 | 亚洲精品乱码白浆高清久久久久久 | 337p日本欧洲亚洲大胆裸体艺术 | 天天摸天天舔天天操 | 国产一区在线精品 | 久久久久久久久久久电影 | 不卡视频一区二区三区 | 美女网站在线 | 黄av免费 | 色综合天天天天做夜夜夜夜做 | 国产超碰在线 | 青青河边草免费直播 | 最近免费观看的电影完整版 | 九色在线| 国产日产欧美在线观看 | 国产一卡在线 | 国产精品ⅴa有声小说 | 国产日本在线观看 | 久久精品视频在线观看 | 日韩精品视频免费专区在线播放 | 午夜在线看 | 精选久久 | 亚洲精品乱码久久久久 | 色天天综合久久久久综合片 | 一区二区三区日韩视频在线观看 | 久久伊人精品一区二区三区 | 人人爽人人舔 | 天天天干天天天操 | 午夜久久福利影院 | 久久成人精品 | 在线激情影院一区 | 日韩激情在线 | 国产精彩视频一区二区 | 国产在线不卡 | 91视频久久久久 | 日韩免费小视频 | 久久经典国产 | 久草在线资源观看 | 日韩a在线看 | sm免费xx网站 | 激情视频区| 天天久久综合 | 国产小视频在线免费观看视频 | 欧美黄在线 | 免费久久久久久 | 国产精品毛片一区二区 | 狠狠亚洲 | 欧美极品在线播放 | 亚洲天堂网站视频 | 欧美精品小视频 | 日韩免费小视频 | 亚洲黄色免费电影 | 久久精品视频在线 | 亚洲精品91天天久久人人 | 91麻豆福利 | 超碰999| 中文字幕在线观看91 | 精品久久久久免费极品大片 | 成年人免费观看在线视频 | 欧美一区二区在线刺激视频 | 成年人在线免费看片 | 麻豆av一区二区三区在线观看 | 97精品国产97久久久久久粉红 | 亚洲三级影院 | 天天夜夜亚洲 | 国产精品麻 | 色资源二区在线视频 | 国产精品一区久久久久 | av在线一级 | 亚洲黄色免费观看 | 国产亚洲片 | 国产亚洲aⅴaaaaaa毛片 | 亚洲一区视频免费观看 | 五月天网站在线 | 精品二区久久 | 最新av在线网址 | 91干干干 | 91精品视频免费看 | 一本之道乱码区 | 婷久久| 999抗病毒口服液 | 五月在线视频 | 手机在线看a | 美女天天操 | 久久精品99久久久久久 | 91资源在线免费观看 | 久久艹在线 | 日韩精品一区二区三区免费观看视频 | 久久综合久久综合这里只有精品 | 中文不卡视频 | 精品在线你懂的 | 日日噜噜噜噜夜夜爽亚洲精品 | 亚洲91中文字幕无线码三区 | 亚洲国产美女精品久久久久∴ | 亚洲精品资源在线观看 | 久久综合久久综合久久 | www.五月婷| 日韩精品视频在线观看网址 | 17婷婷久久www | 久久久久久久国产精品视频 | 午夜影院一区 | 又黄又爽的视频在线观看网站 | 日韩中字在线 | 精品色综合 | 久久免费精品一区二区三区 | 欧美一级片免费播放 | 国产黄色片久久 | 亚洲国产精品成人av | 久久电影网站中文字幕 | 久久精品一区二区 | 久久成人福利 | 91九色porny在线 | 午夜精品中文字幕 | 夜夜看av | 久草热久草视频 | 亚洲永久精品在线观看 | 久久不卡av | 久久综合射| 国产一区二区在线免费播放 | 久久精品3 | 欧美日韩另类视频 | 久在线观看视频 | 91精品影视 | 国产伦精品一区二区三区四区视频 | 热久久国产 | 日日日操操| 丰满少妇一级 | 国产裸体视频bbbbb | 在线免费看黄网站 | 五月天激情综合 | 免费黄色网址网站 | 亚洲一区二区精品在线 | 国内99视频 | 日韩视频图片 | 91视频3p| 国产高清视频在线观看 | 亚洲精品三级 | 超碰av免费 | 五月婷婷中文网 | 亚洲国产精彩中文乱码av | 91在线精品秘密一区二区 | 久久久久久久久久久国产精品 | 亚洲片在线观看 | 免费一级片视频 | 国产视频在线一区二区 | 日韩3区 | 久99热| 国产 欧美 在线 | 精品国产免费av | 欧美色精品天天在线观看视频 | 久久久久久国产精品亚洲78 | 波多野结衣在线播放视频 | 国产精品国产三级国产不产一地 | 亚洲成人av电影 | 九九激情视频 | 97在线看片| 人人dvd| 亚洲激情| 91久久国产综合精品女同国语 | 日韩一级理论片 | 国产一区欧美二区 | 国产不卡精品视频 | 亚洲国产日韩av | 欧美va天堂在线电影 | 中文字幕在线免费97 | 色爱成人网 | 免费观看一级视频 | 国产精品久久久久久久久大全 | 免费午夜视频在线观看 | 国产裸体bbb视频 | 国产精选在线 | 欧美在线视频第一页 | 国产精品一区在线播放 | 91你懂的 | 国产视频美女 | 国产精品久久久久久久7电影 | mm1313亚洲精品国产 | 日韩福利在线观看 | 久久香蕉国产精品麻豆粉嫩av | 亚洲精品五月天 | 欧美一级xxxx | 久久综合五月婷婷 | 免费看的黄色网 | 成人黄色电影免费观看 | 麻豆小视频在线观看 | 国产 精品 资源 | 日韩高清dvd | 黄色毛片视频免费 | 色射爱 | 天堂中文在线播放 | 日韩av午夜在线观看 | 久久久久久免费毛片精品 | 久久久国产毛片 | 一区在线观看视频 | 欧美尹人 | 亚洲精品系列 | 99精品观看 | 九九精品久久久 | 蜜臀av夜夜澡人人爽人人桃色 | 人人澡超碰碰97碰碰碰软件 | 久久欧美精品 | 日韩视频一二三区 | 日韩精品在线看 | 99高清视频有精品视频 | 亚洲激情综合 | 国产玖玖在线 | 91在线观看黄 | 欧美日韩aaaa | 国产97色| 亚洲成人av在线播放 | 色婷婷在线播放 | 黄色录像av | 日韩av进入 | 欧美韩国日本在线观看 | 韩国av一区二区三区在线观看 | 婷婷久久一区 | 黄色软件视频网站 | 人人干人人干人人干 | 国产片网站 | 亚洲成a人片77777潘金莲 | www久久| 午夜精品一区二区三区四区 | 97成人精品视频在线观看 | 欧美一级艳片视频免费观看 | 黄色小说视频在线 | 精品国产一区二区在线 | 91成人免费看 | 手机av在线网站 | 在线观看网站av | 国产一区二区在线播放视频 | 日韩二区在线播放 | 精品视频9999 | 人人看人人 | 99精品视频免费 | 国产色婷婷精品综合在线手机播放 | av片子在线观看 | 国产一区高清在线观看 | 久草在线视频免费资源观看 | 日日草夜夜操 | 亚洲另类人人澡 | 天天干天天色2020 | 久久久久久蜜av免费网站 | 深爱激情站 | 在线观看黄a | 亚洲性少妇性猛交wwww乱大交 | 肉色欧美久久久久久久免费看 | 天天综合色天天综合 | 一区二区三区福利 | 超碰97国产 | www夜夜操com | 久久久综合香蕉尹人综合网 | 91成人精品一区在线播放69 | 在线日韩 | 日本中文字幕在线视频 | 五月天婷婷视频 | 欧洲高潮三级做爰 | 日韩成人看片 | 欧美久久电影 | 九色视频自拍 | 97精品国产一二三产区 | 天天天色 | 成人在线免费小视频 | 国产精品毛片久久久久久久久久99999999 | 日韩av电影免费观看 | 欧美极品一区二区三区 | 婷婷在线免费视频 | 色偷偷av男人天堂 | 天天干天天色2020 | 日韩在线观看a | 一级成人免费视频 | 81国产精品久久久久久久久久 | 丁香六月五月婷婷 | 久久在线视频在线 | 成人久久久久久久久 | 欧美国产在线看 | 韩国一区二区三区视频 | 天堂在线一区二区 | 国产99久久| 国产精品av免费 | 国产一二区视频 | 国产一级片一区二区三区 | 婷婷久久综合九色综合 | 欧美综合在线观看 | 中文字幕视频在线播放 | 日韩精品一区二区三区外面 | 中文字幕在线播放日韩 | 国产精品女同一区二区三区久久夜 | 国产成人精品国内自产拍免费看 | 玖草在线观看 | 日本h视频在线观看 | 日韩在线视频播放 | 国产一二三区在线观看 | 99视频在线免费 | 中文字幕在线观看日本 | 国产精品免费观看国产网曝瓜 | 欧美日韩免费在线视频 | 99999精品 | 超碰99人人 | 欧美精品乱码久久久久 | 久久综合九色综合欧美狠狠 | 激情黄色一级片 | 成人羞羞免费 | 久香蕉| av网站有哪些 | 麻豆国产在线播放 | 亚洲精品视频免费在线 | 韩日精品在线 | 午夜视频免费播放 | 国产一级免费电影 | 欧美福利久久 | 国产亚洲综合在线 | 波多野结衣在线播放视频 | av中文字幕免费在线观看 | 国产高清精品在线观看 | 免费日韩av片 | 狠狠躁夜夜躁人人爽超碰91 | 亚洲成人欧美 | 国产在线观看国语版免费 | 麻豆视频一区二区 | 午夜神马福利 | 日韩精品字幕 | 99精品成人 | 日韩三级免费观看 | 欧美日韩中文国产一区发布 | 久久久999免费视频 日韩网站在线 | 夜色资源站国产www在线视频 | 99久久er热在这里只有精品15 | 91在线小视频 | 最新免费av在线 | av免费成人 | 久久久久福利视频 | 91av视频在线观看免费 | 九九99靖品 | 色综合久久久久久中文网 | 精品国产一区二区三区久久久久久 | 九九精品视频在线观看 | 狠狠色丁香婷婷综合久小说久 | 黄色在线看网站 | 国产在线观看黄 | 麻豆一级视频 | 久久黄色免费视频 | 久久久久美女 | 日韩欧美视频免费看 | 丝袜美腿亚洲 | 在线不卡中文字幕播放 | 国产精品理论在线观看 | 在线一二区 | 欧美日韩中文国产一区发布 | a级片久久久| 又爽又黄又无遮挡网站动态图 | 国产精品久久久久久久久久久久午夜 | 日韩免费在线观看视频 | 天天射天天干天天插 | 香蕉97视频观看在线观看 | 国产一二三精品 | 久久色亚洲 | 婷婷色网视频在线播放 | av高清网站在线观看 | a级国产乱理论片在线观看 伊人宗合网 | 国产精品门事件 | 99精品国产一区二区三区不卡 | 美女精品在线观看 | 天天色天天干天天 | 98超碰人人 | 天海翼一区二区三区免费 | 亚洲精品国产精品久久99热 | 国产字幕在线观看 | 波多野结衣在线中文字幕 | 91视频a| 亚洲一级电影视频 | 午夜精品一区二区国产 | 久久艹人人 | 久久精品国产亚洲 | 国产探花 | 不卡视频一区二区三区 | 18久久久久久 | 中国一级特黄毛片大片久久 | 免费av小说| 久久av免费 | 黄色影院在线观看 | 91精品欧美 | 精品自拍av| 免费一级片在线 | 久久久久久久网站 | 国产xx视频 | 香蕉视频18| 日本大尺码专区mv | 91精品播放| 国产va饥渴难耐女保洁员在线观看 | 亚洲欧洲精品一区 | 91视频麻豆 | 国产女教师精品久久av | 在线亚洲高清视频 | 激情视频二区 | 97精品久久人人爽人人爽 | 国产高清免费 | 韩国精品在线观看 | 深爱婷婷激情 | 欧美日韩高清一区二区三区 | 在线播放一区二区三区 | 日韩精品一区电影 | 97在线精品 | 精品久久久久久国产91 | 18国产精品福利片久久婷 | 激情婷婷亚洲 | 久久精品99国产精品酒店日本 | 久久综合久久久 | 午夜av大片| 婷婷草| 成年人黄色大全 | 精品一区二区在线免费观看 | 四虎最新域名 | 日本乱视频| 国产一区视频导航 | 伊人狠狠干 | 日韩在线视频国产 | 二区三区毛片 | 视频在线观看亚洲 | 亚洲国产视频a | av天天色| 国产成人精品电影久久久 | 456免费视频 | 在线中文字幕网站 | 久草在线在线精品观看 | 久久99国产综合精品免费 | 亚洲视频观看 | 久久精品精品电影网 | 综合网伊人 | 成人网444ppp | 国产精品久久久久一区二区国产 | 狠狠网站 | 亚洲精品www. | 久久免费视频6 | 国产清纯在线 | 日韩av免费一区 | 五月婷婷在线观看视频 | 人操人| 国产免费专区 | 最新91在线视频 | 国产一级片网站 | 精品国产大片 | 精品一区二区在线免费观看 | 中文字幕av日韩 | 国产无区一区二区三麻豆 | 五月开心激情 | 成人福利在线 | 国产高清视频免费 | 午夜视频在线观看欧美 | 日韩精品一区二区三区高清免费 | 日韩av一区二区三区在线观看 | 色久av| 亚洲精品视频在线观看免费 | 久久久香蕉视频 | 欧美视频网址 | 日韩h在线观看 | 热久久视久久精品18亚洲精品 | 国产色秀视频 | 新版资源中文在线观看 | 亚洲精品国产自产拍在线观看 | 黄色网址中文字幕 | 欧美一区,二区 | 韩国av免费 | 成人在线一区二区三区 | 久久久影院一区二区三区 | 久久久99国产精品免费 | 久久精品一区八戒影视 | 人人干狠狠操 | 在线亚洲人成电影网站色www | 国产精品亚洲片在线播放 | 久久久91精品国产一区二区精品 | 91视频首页 | 在线观看视频h | 久色伊人 | 狠狠干中文字幕 | 色综合 久久精品 | 中文字幕在线观看av | 日韩中文在线视频 | 黄色日视频 | av视屏在线播放 | 一区二区三区动漫 | 国产无遮挡又黄又爽在线观看 | 国产高清在线一区 | 主播av在线| 91精品中文字幕 | 亚洲精品在线观看视频 | 欧美一级黄色片 | 亚洲精品一区二区网址 | 综合网色 | 久久久精品国产免费观看一区二区 | 一级做a视频 | 狠狠色2019综合网 | 久久综合九色九九 | 99久久精品国产系列 | 亚洲另类交 | 国产免费一区二区三区网站免费 | 91大神精品视频 | 日韩国产精品毛片 | 在线 国产一区 | 伊人网av | 91一区啪爱嗯打偷拍欧美 | 亚洲另类人人澡 | 一区 二区 精品 | 四虎国产精品永久在线国在线 | 成人国产网站 | 日本黄色大片免费 | 欧美一区二区在线刺激视频 | 一区二区伦理 | 永久免费在线 | 美女久久久久久 | 91视频免费看 |