亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C++ > 正文

用C語言來實(shí)現(xiàn)一個(gè)簡(jiǎn)單的虛擬機(jī)

2020-05-23 14:18:42
字體:
供稿:網(wǎng)友

這篇文章主要介紹了用C語言來實(shí)現(xiàn)一個(gè)簡(jiǎn)單的虛擬機(jī),其中棧數(shù)組的部分非常值得學(xué)習(xí),需要的朋友可以參考下

必要的準(zhǔn)備工作及注意事項(xiàng):

在開始之前需要做以下工作:

一個(gè)C編譯器——我使用了 clang 3.4,也可以用其它支持 c99/c11 的編譯器;

文本編輯器——我建議使用基于IDE的文本編輯器,我使用 Emacs;

基礎(chǔ)編程知識(shí)——最基本的變量,流程控制,函數(shù),數(shù)據(jù)結(jié)構(gòu)等;

Make 腳本——能使程序更快一點(diǎn)。

為什么要寫個(gè)虛擬機(jī)?

有以下原因:

想深入了解計(jì)算機(jī)工作原理。本文將幫助你了解計(jì)算機(jī)底層如何工作,虛擬機(jī)提供簡(jiǎn)潔的抽象層,這不就是一個(gè)最好的學(xué)習(xí)它們?cè)淼姆椒▎?

更深入了解一些編程語言是如何工作。例如,當(dāng)下多種經(jīng)常使用那些語言的虛擬機(jī)。包括JVM,Lua VM,F(xiàn)aceBook 的 Hip—Hop VM(PHP/Hack) 等。

只是因?yàn)橛信d趣學(xué)習(xí)虛擬機(jī)。

指令集

我們將要實(shí)現(xiàn)一種非常簡(jiǎn)單的自定義的指令集。我不會(huì)講一些高級(jí)的如位移寄存器等,希望在讀過這篇文章后掌握這些。

我們的虛擬機(jī)具有一組寄存器,A,B,C,D,E, 和F。這些是通用寄存器,也就是說,它們可以用于存儲(chǔ)任何東西。一個(gè)程序?qū)?huì)是一個(gè)只讀指令序列。這個(gè)虛擬機(jī)是一個(gè)基于堆棧的虛擬機(jī),也就是說它有一個(gè)可以讓我們壓入和彈出值的堆棧,同時(shí)還有少量可用的寄存器。這要比實(shí)現(xiàn)一個(gè)基于寄存器的虛擬機(jī)簡(jiǎn)單的多。

言歸正傳,下面是我們將要實(shí)現(xiàn)的指令集:

 

 
  1. PSH 5 ; pushes 5 to the stack 
  2. PSH 10 ; pushes 10 to the stack 
  3. ADD ; pops two values on top of the stack, adds them pushes to stack 
  4. POP ; pops the value on the stack, will also print it for debugging 
  5. SET A 0 ; sets register A to 0 
  6. HLT ; stop the program 

這就是我們的指令集,注意,POP 指令將會(huì)打印我們彈出的指令,這樣我們就能夠看到 ADD 指令工作了。我還加入了一個(gè) SET 指令,主要是讓你理解寄存器是可以訪問和寫入的。你也可以自己實(shí)現(xiàn)像MOV A B(將A的值移動(dòng)到B)這樣的指令。HTL 指令是為了告訴我們程序已經(jīng)運(yùn)行結(jié)束。

虛擬機(jī)是如何工作的呢?

現(xiàn)在我們已經(jīng)到了本文最關(guān)鍵的部分,虛擬機(jī)比你想象的簡(jiǎn)單,它們遵循一個(gè)簡(jiǎn)單的模式:讀取;解碼;執(zhí)行。首先,我們從指令集合或代碼中讀取下一條指令,然后將指令解碼并執(zhí)行解碼后的指令。為簡(jiǎn)單起見,我們忽略了虛擬機(jī)的編碼部分,典型的虛擬機(jī)將會(huì)把一個(gè)指令(操作碼和它的操作數(shù))打包成一個(gè)數(shù)字,然后再解碼這個(gè)指令。

項(xiàng)目結(jié)構(gòu)

開始編程之前,我們需要設(shè)置好我們的項(xiàng)目。第一,你需要一個(gè)C編譯器(我使用 clang 3.4)。還需要一個(gè)文件夾來放置我們的項(xiàng)目,我喜歡將我的項(xiàng)目放置于~/Dev:

 

 
  1. $cd ~/Dev/ 
  2. mkdir mac 
  3. cd mac 
  4. mkdir src 

如上,我們先 cd 進(jìn)入~/Dev 目錄,或者任何你想放置的位置,然后新建一個(gè)目錄(我稱這個(gè)虛擬機(jī)為"mac")。然后再 cd 進(jìn)這個(gè)目錄并新建我們 src 目錄,這個(gè)目錄用于放置代碼。

Makefile

makefile 相對(duì)直接,我們不需要將什么東西分成多個(gè)文件,也不用包含任何東西,所以我們只需要用一些標(biāo)志來編譯文件:

 

 
  1. SRC_FILES = main.c 
  2. CC_FLAGS = -Wall -Wextra -g -std=c11 
  3. CC = clang 
  4.  
  5. all: 
  6. ${CC} ${SRC_FILES} ${CC_FLAGS} -o mac 

這對(duì)目前來說已經(jīng)足夠了,你以后還可以改進(jìn)它,但是只要它能完成這個(gè)工作,我們應(yīng)該滿足了。

指令編程(代碼)

現(xiàn)在開始寫虛擬機(jī)的代碼了。第一,我們需要定義程序的指令。為此,我們可以使用一個(gè)枚舉類型enum,因?yàn)槲覀兊闹噶罨旧鲜菑?到X的數(shù)字。事實(shí)上,可以說你是在組裝一個(gè)匯編文件,它會(huì)使用像 mov 這樣的詞,然后翻譯成聲明的指令。

我們可以只寫一個(gè)指令文件,例如 PSH, 5 是0, 5,但是這樣并不易讀,所以我們使用枚舉器!

 

 
  1. typedef enum { 
  2. PSH, 
  3. ADD, 
  4. POP, 
  5. SET, 
  6. HLT 
  7. } InstructionSet; 

現(xiàn)在我們可以將一個(gè)測(cè)試程序存儲(chǔ)為一個(gè)數(shù)組。我們寫一個(gè)簡(jiǎn)單的程序用于測(cè)試:將5和6相加,然后將他們打印出來(用POP指令)。如果你愿意,你可以定義一個(gè)指令將棧頂?shù)闹荡蛴〕鰜怼?/p>

指令應(yīng)該存儲(chǔ)成一個(gè)數(shù)組,我將在文檔的頂部定義它;但你或許會(huì)將它放在一個(gè)頭文件中,下面是我們的測(cè)試程序:

 

  1. const int program[] = { 
  2. PSH, 5, 
  3. PSH, 6, 
  4. ADD, 
  5. POP, 
  6. HLT 
  7. }; 

上面的程序?qū)?huì)把5和6壓入棧,調(diào)用 ADD 指令,這將會(huì)把棧頂?shù)膬蓚€(gè)值彈出,相加后將結(jié)果壓回棧中,接下來我們彈出結(jié)果,因?yàn)?POP 指令將會(huì)打印這個(gè)值,但是你不必自己再做了,我已經(jīng)做好并測(cè)試過了。最后,HLT 指令結(jié)束程序。

很好,這樣我們有了自己的程序?,F(xiàn)在我們實(shí)現(xiàn)了虛擬機(jī)的讀取,解碼,求值的模式。但是要記住,我們沒有解碼任何東西,因?yàn)槲覀兘o出的是原始指令。也就是說我們只需要關(guān)注讀取和求值!我們可以將它們簡(jiǎn)化成兩個(gè)函數(shù) fetch 和 evaluate。

取得當(dāng)前指令

因?yàn)槲覀円呀?jīng)將我們的程序存成了一個(gè)數(shù)組,所以很簡(jiǎn)單的就可以取得當(dāng)前指令。一個(gè)虛擬機(jī)有一個(gè)計(jì)數(shù)器,一般來說叫做程序計(jì)數(shù)器,指令指針等等,這些名字是一個(gè)意思取決于你的個(gè)人喜好。在虛擬機(jī)的代碼庫里,IP 或 PC 這樣的簡(jiǎn)寫形式也隨處可見。

如果你之前有記得,我說過我們要把程序計(jì)數(shù)器以寄存器的形式存儲(chǔ)...我們將那么做——在以后。現(xiàn)在,我們只是在我們代碼的最頂端創(chuàng)建一個(gè)叫 ip 的變量,并且設(shè)置為 0。

 

 
  1. int ip = 0; 

ip 變量代表指令指針。因?yàn)槲覀円呀?jīng)將程序存成了一個(gè)數(shù)組,所以使用 ip 變量去指明程序數(shù)組中當(dāng)前索引。例如,如果創(chuàng)建了一個(gè)被賦值了程序 ip 索引的變量 x,它將存儲(chǔ)我們程序的第一條指令。

[假設(shè)ip為0]

 

 
  1. int ip = 0; 
  2.  
  3. int main() { 
  4. int instr = program[ip]; 
  5. return 0; 

如果我們打印變量 instr,本來應(yīng)是 PSH 的它將顯示為0,因?yàn)樵谒俏覀兠杜e里的第一個(gè)值。我們也可以寫一個(gè)取回函數(shù)像這樣:

 

 
  1. int fetch() { 
  2. return program[ip]; 

這個(gè)函數(shù)將會(huì)返回當(dāng)前被調(diào)用指令。太棒了,那么如果我們想要下一條指令呢?很容易,我們只要增加指令指針就好了:

 

 
  1. int main() { 
  2. int x = fetch(); // PSH 
  3. ip++; // increment instruction pointer 
  4. int y = fetch(); // 5 

那么怎樣讓它自己動(dòng)起來呢?我們知道一個(gè)程序直到它執(zhí)行 HLT 指令才會(huì)停止。因此我們使用一個(gè)無限的循環(huán)持續(xù)直到當(dāng)前指令為HLT。

 

 
  1. // INCLUDE <stdbool.h>! 
  2. bool running = true
  3.  
  4. int main() { 
  5. while (running) { 
  6. int x = fetch(); 
  7. if (x == HLT) running = false
  8. ip++; 

這工作的很好,但是有點(diǎn)凌亂。我們正在循環(huán)每一條指令,檢查是否 HLT,如果是就停止循環(huán),否則“吃掉”指令接著循環(huán)。

判斷一條指令

因此這就是我們虛擬機(jī)的主體,然而我們想要確實(shí)的評(píng)判每一條指令,并且使它更簡(jiǎn)潔一些。好的,這個(gè)簡(jiǎn)單的虛擬機(jī),你可以寫一個(gè)“巨大”的 switch 聲明。讓 switch 中的每一個(gè) case 對(duì)應(yīng)一條我們定義在枚舉中的指令。這個(gè) eval 函數(shù)將使用一個(gè)簡(jiǎn)單的指令的參數(shù)來判斷。我們?cè)诤瘮?shù)中不會(huì)使用任何指令指針遞增除非我們想操作數(shù)浪費(fèi)操作數(shù)。

 

 
  1. void eval(int instr) { 
  2. switch (instr) { 
  3. case HLT: 
  4. running = false
  5. break

因此如果我們?cè)诨氐街骱瘮?shù),就可以像這樣使用我們的 eval 函數(shù)工作:

 

 
  1. bool running = true
  2. int ip = 0; 
  3.  
  4. // instruction enum here 
  5.  
  6. // eval function here 
  7.  
  8. // fetch function here 
  9.  
  10. int main() { 
  11. while (running) { 
  12. eval(fetch()); 
  13. ip++; // increment the ip every iteration 

棧!

很好,那會(huì)很完美的完成這個(gè)工作。現(xiàn)在,在我們加入其他指令之前,我們需要一個(gè)棧。幸運(yùn)的是,棧是很容易實(shí)現(xiàn)的,我們僅僅需要使用一個(gè)數(shù)組而已。數(shù)組會(huì)被設(shè)置為合適的大小,這樣它就能包含256個(gè)值了。我們也需要一個(gè)棧指針(常被縮寫為sp)。這個(gè)指針會(huì)指向棧數(shù)組。

為了讓我們對(duì)它有一個(gè)更加形象化的印象,讓我們來看看這個(gè)用數(shù)組實(shí)現(xiàn)的棧吧:

 

 
  1. [] // empty 
  2.  
  3. PSH 5 // put 5 on **top** of the stack 
  4. [5] 
  5.  
  6. PSH 6 
  7. [5, 6] 
  8.  
  9. POP 
  10. [5] 
  11.  
  12. POP 
  13. [] // empty 
  14.  
  15. PSH 6 
  16. [6] 
  17.  
  18. PSH 5 
  19. [6, 5] 

那么,在我們的程序里發(fā)生了什么呢?

 

 
  1. PSH, 5, 
  2. PSH, 6, 
  3. ADD, 
  4. POP, 
  5. HLT 

我們首先把5壓入了棧

 

 
  1. [5] 

然后壓入6:

 

 
  1. [5, 6] 

接著添加指令,取出這些值,把它們加在一起并把結(jié)果壓入棧中:

 

 
  1. [5, 6] 
  2.  
  3. // pop the top value, store it in a variable called a 
  4. a = pop; // a contains 6 
  5. [5] // stack contents 
  6.  
  7. // pop the top value, store it in a variable called b 
  8. b = pop; // b contains 5 
  9. [] // stack contents 
  10.  
  11. // now we add b and a. Note we do it backwards, in addition 
  12. // this doesn't matter, but in other potential instructions 
  13. // for instance divide 5 / 6 is not the same as 6 / 5 
  14. result = b + a; 
  15. push result // push the result to the stack 
  16. [11] // stack contents 

那么我們的棧指針在哪起作用呢?棧指針(或者說sp)一般是被設(shè)置為-1,這意味著這個(gè)指針是空的。請(qǐng)記住一個(gè)數(shù)組是從0開始的,如果沒有初始化sp的值,那么他會(huì)被設(shè)置為C編譯器放在那的一個(gè)隨機(jī)值。

如果我們將3個(gè)值壓棧,那么sp將變成2。所以這個(gè)數(shù)組保存了三個(gè)值:

sp指向這里(sp = 2)

|

V

[1, 5, 9]

0 1 2 <- 數(shù)組下標(biāo)

現(xiàn)在我們從棧上出棧一次,我們僅需要減小棧頂指針。比如我們接下來把9出棧,那么棧頂將變?yōu)?:

sp指向這里(sp = 1)

|

V

[1, 5]

0 1 <- 數(shù)組下標(biāo)

所以,當(dāng)我們想知道棧頂內(nèi)容的時(shí)候,只需要查看sp的當(dāng)前值。OK,你可能想知道棧是如何工作的,現(xiàn)在我們用C語言實(shí)現(xiàn)它。很簡(jiǎn)單,和ip一樣,我們也應(yīng)該定義一個(gè)sp變量,記得把它賦為-1!再定義一個(gè)名為stack的數(shù)組,代碼如下:

 

 
  1. int ip = 0; 
  2. int sp = -1; 
  3. int stack[256]; // 用數(shù)組或適合此處的其它結(jié)構(gòu) 
  4.  
  5. // 其它C代碼 

現(xiàn)在如果我們想入棧一個(gè)值,我們先增加棧頂指針,接著設(shè)置當(dāng)前sp處的值(我們剛剛增加的)。注意:這兩步的順序很重要!

 

 
  1. // 壓棧5 
  2.  
  3. // sp = -1 
  4. sp++; // sp = 0 
  5. stack[sp] = 5; // 棧頂現(xiàn)在變?yōu)? 

所以,在我們的執(zhí)行函數(shù)eval()里,可以像這樣實(shí)現(xiàn)push出棧指令:

 

 
  1. void eval(int instr) { 
  2. switch (instr) { 
  3. case HLT: { 
  4. running = false
  5. break
  6. case PSH: { 
  7. sp++; 
  8. stack[sp] = program[++ip]; 
  9. break

現(xiàn)在你看到,它和我們之前實(shí)現(xiàn)的eval()函數(shù)有一些不同。首先,我們把每個(gè)case語句塊放到大括號(hào)里。你可能不太了解這種用法,它可以讓你在每條case的作用域里定義變量。雖然現(xiàn)在不需要定義變量,但將來會(huì)用到。并且它可以很容易得讓所有的case語句塊保持一致的風(fēng)格。

其次是神奇的表達(dá)式program[++ip]。它做了什么?呃,我們的程序存儲(chǔ)在一個(gè)數(shù)組里,PSH指令需要獲得一個(gè)操作數(shù)。操作數(shù)本質(zhì)是一個(gè)參數(shù),就像當(dāng)你調(diào)用一個(gè)函數(shù)時(shí),你可以給它傳遞一個(gè)參數(shù)。這種情況我們稱作壓棧數(shù)值5。我們可以通過增加指令指針(譯者注:一般也叫做程序計(jì)數(shù)器)ip來獲取操作數(shù)。當(dāng)ip為0時(shí),這意味著執(zhí)行到了PSH指令,接下來我們希望取得下一條指令——即壓棧的數(shù)值。這可以通過ip自增的方法實(shí)現(xiàn)(注意:增加ip的位置十分重要,我們希望在取得指令前自增,否則我們只是拿到了PSH指令),接下來需要跳到下一條指令否則會(huì)引發(fā)奇怪的錯(cuò)誤。當(dāng)然我們也可以把sp++簡(jiǎn)化到stack[++sp]里。

對(duì)于POP指令,實(shí)現(xiàn)非常簡(jiǎn)單。只需要減小棧頂指針,但是我一般希望能夠在出棧的時(shí)候打印出棧值。

我省略了實(shí)現(xiàn)其它指令的代碼和swtich語句,僅列出POP指令的實(shí)現(xiàn):

 

 
  1. // 記得#include <stdio.h>! 
  2.  
  3. case POP: { 
  4. int val_popped = stack[sp--]; 
  5. printf("popped %d/n", val_popped); 
  6. break

現(xiàn)在,POP指令能夠工作了!我們剛剛做的只是把棧頂放到變量val_popped里,接著棧頂指針減一。如果我們首先棧頂減一,那么將得到一些無效值,因?yàn)閟p可能取值為0,那么我們可能把stack[-1]賦給val_popped,通常這不是一個(gè)好主意。

最后是ADD指令。這條指令可能要花費(fèi)你一些腦細(xì)胞,同時(shí)這也是我們需要用大括號(hào){}實(shí)現(xiàn)case語句內(nèi)作用域的原因。

 

 
  1. case ADD: { 
  2. // 首先我們出棧,把數(shù)值存入變量a 
  3. int a = stack[sp--]; 
  4.  
  5. // 接著我們出棧,把數(shù)值存入變量b 
  6.  
  7. // 接著兩個(gè)變量相加,再把結(jié)果入棧 
  8. int result = a + b; 
  9. sp++; // 棧頂加1 **放在賦值之前** 
  10. stack[sp] = result; // 設(shè)置棧頂值 
  11.  
  12. // 完成! 
  13. break

寄存器

寄存器是虛擬機(jī)中的選配件,很容易實(shí)現(xiàn)。之前提到過我們可能需要六個(gè)寄存器:A,B,C,D,E和F。和實(shí)現(xiàn)指令集一樣,我們也用一個(gè)枚舉來實(shí)現(xiàn)它們。

 

 
  1. typedef enum { 
  2. A, B, C, D, E, F, 
  3. NUM_OF_REGISTERS 
  4. } Registers; 

小技巧:枚舉的最后放置了一個(gè)數(shù) NUM_OF_REGISTERS。通過這個(gè)數(shù)可以獲取寄存器的個(gè)數(shù),即便你又添加了其它的寄存器?,F(xiàn)在我們需要一個(gè)數(shù)組為寄存器存放數(shù)值:

 

 
  1. int registers[NUM_OF_REGISTERS]; 

接下來你可以讀取寄存器內(nèi)的值:

 

 
  1. printf("%d/n", registers[A]); // 打印寄存器A的值 

修訂

我沒有在寄存器花太多心思,但你應(yīng)該能夠?qū)懗鲆恍┎僮骷拇嫫鞯闹噶?。比如,如果你想?shí)現(xiàn)任何分支跳轉(zhuǎn),可以通過把指令指針(譯者注:或叫程序計(jì)數(shù)器)和/或棧頂指針存到寄存器里,或者通過實(shí)現(xiàn)分支指令。

前者實(shí)現(xiàn)起來相對(duì)快捷、簡(jiǎn)單。我們可以這樣做,增加代表IP和SP的寄存器:

 

 
  1. typedef enum { 
  2. A, B, C, D, E, F, PC, SP, 
  3. NUM_OF_REGISTERS 
  4. } Registers; 

現(xiàn)在我們需要實(shí)現(xiàn)代碼來使用指令指針和棧頂指針。一個(gè)簡(jiǎn)單的辦法——刪掉上面定義的sp和ip變量,用宏定義實(shí)現(xiàn)它們:

 

 
  1. #define sp (registers[SP]) 
  2. #define ip (registers[IP]) 

譯者注:此處應(yīng)同Registers枚舉中保持一致,IP應(yīng)改為PC

這個(gè)修改恰到好處,你不需要重寫很多代碼,同時(shí)它工作的很好。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
免费看一级一片| 黄色片网站免费| 亚洲国产99| 屁屁影院ccyy国产第一页| 亚洲男女网站| 99久久免费精品国产72精品九九| 国产精品欧美日韩一区| 亚洲精品色午夜无码专区日韩| 久久91亚洲精品中文字幕奶水| 日韩欧美电影| 欧美日韩精品久久久免费观看| 亚洲欧洲久久| 欧美日韩国产不卡| 中文字幕一区二区在线观看| 精品久久久久久无码人妻| 51精品视频一区二区三区| 国产精品免费一区二区三区在线观看| 69国产精品成人在线播放| 国产高清精品在线| 亚洲日本久久| 日韩一级大片在线| 天堂网在线观看在线观看精品| 亚洲图片在线综合| 日韩欧美一区二区三区久久婷婷| 精品国产青草久久久久96| 国产wwwwxxxx| 宅男在线一区| 欧美二区在线| 黄色一区二区视频| 国产丝袜控视频在线观看| 成人午夜av| 91精品国产综合久久精品app| 波多野结衣在线电影| 在线精品视频在线观看高清| segui88久久综合9999| 亚洲一级二级在线| 亚洲一二三四区| 女女色综合影院| 国产精品三级a三级三级午夜| 99久久免费国产| 91国产成人在线| 国产国产精品人在线视| 在线观看的av网站| 中文字幕人妻一区二区三区视频| 日韩高清在线免费观看| a级片国产精品自在拍在线播放| 日本欧美在线看| 精品一区二区三孕妇视频| 黄色精品免费| 中文字幕视频在线观看| 自拍偷拍欧美| 欧美精品一区二区久久| 亚洲精品国产精品国自产网站| 美日韩在线观看| 91精品国产免费久久久久久| 95av在线视频| 洋洋av久久久久久久一区| 欧美国产偷国产精品三区| 国产黄色片网站| 欧美性受xxxx免费视频| 免费看特级毛片| 成人美女视频在线观看| 最新日本中文字幕| 欧美影院精品一区| 久久久午夜精品理论片中文字幕| 欧洲性视频在线播放| 国产精品无码永久免费888| av男人的天堂av| 欧美日韩精品在线一区二区| 精品xxxxxbbbb欧美中文| 国产情侣第一页| 日本免费看黄色| 黄色欧美在线| 农村妇女精品一区二区| 欧美精品久久久久久久自慰| 乱熟女高潮一区二区在线| 亚洲电影第三页| 快播电影网址老女人久久| 亚洲va码欧洲m码| 91小视频免费观看| 欧美孕妇孕交xxⅹ孕妇交| 美女诱惑一区二区| 中文字幕成人av| 国产精品理论片| 日本h片在线观看| 一级一级黄色片| 日韩欧美国产片| 午夜福利一区二区三区| 欧美日韩亚洲一区| 日本女人一区二区三区| 免费精品一区二区三区在线观看| 国产欧美亚洲精品| 四虎永久在线精品无码视频| 精品国产免费视频| 清纯唯美亚洲综合| 亚洲熟妇av乱码在线观看| 亚洲欧美自拍一区| 四虎精品在永久在线观看| 亚洲精品国产品国语在线| 二级片在线观看| 欧美日韩激情网| 亚洲成人免费视| 国产成人精品久久二区二区| 亚洲福利一区二区| 多男操一女视频| 欧美a免费在线| 国内精品在线一区| 亚洲天堂视频网站| 国产精品12p| 亚洲成av人影片在线观看| 亚洲三级黄色片| 亚洲欧洲日韩女同| 国产女人18水真多毛片18精品| 国产成人一区二区在线| 五月婷婷激情综合| 国产电影一区二区三区爱妃记| 国产成人一区二区三区小说| 成人精品鲁一区一区二区| 91在线无精精品一区二区| 亚洲国产日韩精品| 中文字幕日韩欧美| 久久99精品久久久久婷婷| 日本五十路女优| 久久久久久久久久久久久久免费看| 欧美精品一区二区久久久| 影音先锋在线国产| 中文字幕日产av一二三区| 国产香蕉久久| 久久激情五月婷婷| 中文字幕激情小说| 中文字幕日韩久久| 女同互添互慰av毛片观看| 玛雅亚洲电影| 国产精品久久久久久久久久东京| 青青草精品视频在线观看| 精品粉嫩aⅴ一区二区三区四区| 欧美日韩一视频区二区| 26uuu久久天堂性欧美| 精品视频亚洲| 男女视频网站免费观看| 男人和女人做事情在线视频网站免费观看| 欧美综合国产精品久久丁香| 在线午夜影院| 超碰免费97在线观看| 国产伦精品一区二区三区照片| 日韩一区二区电影| 免费高清视频在线一区| 国产私拍一区| 欧美大片免费观看网址| 在线观看一区视频| 午夜精彩视频| 新欧美整片sss第一页| 在线观看亚洲成人| 欧美成人欧美edvon| 午夜精品三级视频福利| 激情高潮到大叫狂喷水| 99久久久无码国产精品| xxxxx99| 成人av网页| 日韩精品一区二区三区国语自制| 一区二区三区亚洲变态调教大结局| 成年人在线视频免费观看| 欧美精三区欧美精三区| 性做久久久久久久免费看| 国产精品福利视频| 国产乱人伦精品一区二区| 亚洲人成在线观| 免费三级毛片| 国产又粗又长免费视频| 欧美mv日韩mv国产网站app| 黄色大片a级| 亚洲熟妇无码乱子av电影| 久久精品女人毛片国产| 色婷婷久久一区二区三区麻豆| 亚洲伊人第一页| 99精品视频精品精品视频| av免费中文字幕| 欧美三级蜜桃2在线观看| 青青草社区视频| 中文字幕人成人乱码| 日韩欧美激情在线| 日本不卡高清| 天堂成人娱乐在线视频免费播放网站| 伊人久久大香线蕉综合75| 欧美老女人第四色| 久久综合久久久| 久久中文视频| 91在线一区二区| 伊人久久大香线蕉av超碰演员| 国产精品99精品久久免费| 99九九电视剧免费观看| 欧美性视频精品| 欧美中文字幕一二三区视频| 米奇777超碰欧美日韩亚洲| 亚洲综合av一区二区三区| 国产一级粉嫩xxxx| 欧美国产日产韩国视频| 日韩av影片在线观看| 亚洲日本在线播放| 亚洲91精品在线| 亚洲综合在线电影| 182在线视频| 国产日韩精品电影| 亚洲视频欧美视频| 亚洲人成无码网站久久99热国产| 国产精品一区二区在线看| 久久久久久久久久久久久av| 亚洲永久精品一区| 国产伦精品一区二区三区视频小说| 8x国产一区二区三区精品推荐| 中文字幕佐山爱一区二区免费| 黄色一级片av| 亚洲国产成人av在线| 欧产日产国产v| 看av免费毛片手机播放| 久久国产精品99国产精| 亚洲视频狠狠| 日韩av在线一区二区三区| 亚洲欧美卡通另类91av| 91视频最新入口| 日韩欧美一中文字暮专区| 国产精品一品| 奇米777欧美一区二区| 亚洲欧美激情一区二区三区| 91视频观看| 成人毛片在线精品国产| 成年人三级网站| 日本免费一区视频| 日本我和搜子同居的日子高清在线| 欧美交换国产一区内射| 免费在线不卡视频| av一区二区三区免费观看| 性疯狂做受xxxx高清视频| 99久久精品免费视频| 在线亚洲激情| 中文字幕激情小说| 久99久在线视频| 成人做爰视频网站| 亚洲精品国产一区二区| 欧美日韩五码| 福利在线小视频| 国产一区二区视频在线免费观看| 亚洲色成人www永久网站| 成人同人动漫免费观看| 成人一道本在线| 136国产福利精品导航网址| 神马影院午夜我不卡影院| 日韩欧美在线视频一区二区| av成人资源网| 男女爱爱网站| 国产成人精品一区二区三区免费| 久久免费在线观看| 成人毛片视频免费看| 99re久久| 亚洲热av色在线播放| 黄色成人在线免费观看| 欧美老女人另类| 日韩小视频在线观看| 国产精品入口麻豆完整版| 欧美日韩亚洲高清| 国产特级aaaaaa大片| 在线天堂av| 蜜桃在线视频| 国产玖玖精品视频| 香蕉久久夜色精品| www.久久久久久久久久久| 国产麻豆成人传媒免费观看| free极度另类性欧美| 97超碰在线免费观看| 在线免费看黄| 美女亚洲精品| 国产成年人免费视频| 91影院成人| 国产一区二区三区高清视频| 放荡的美妇在线播放| 视频一区在线观看| 韩国三级在线看| 久久99视频免费| 欧美少妇性生活视频| 麻豆一区二区三区| 99精品热视频只有精品10| 日韩女优在线视频| 国模一区二区三区| eeuss草民免费| 影音先锋亚洲视频| 欧美激情 亚洲a∨综合| 尤物九九久久国产精品的特点| 国产精品人人做人人爽| 精品裸体bbb| 亚洲超碰精品一区二区| 国产精品白丝av| 欧美最猛性xxxxx喷水| 国产av 一区二区三区| 亚洲伊人一本大道中文字幕| 日韩三级免费观看| 亚洲国产欧美国产综合一区| 亚洲第一色网站| 日韩高清dvd| 欧美在线观看视频一区二区| 国产在线一区二区三区欧美| 欧洲精品乱码久久久久蜜桃| 成人久久久精品国产乱码一区二区| av电影网站在线观看| 国产美女91呻吟求| 久久免费观看视频| 亚洲欧洲国产精品| 99久久人妻无码中文字幕系列| 免费亚洲精品视频| 欧美熟妇交换久久久久久分类| 欧美激情啊啊啊| 不卡一区二区三区视频| 无套内谢大学处破女www小说| xfplay资源站夜色先锋| 先锋影音欧美官网| 污污在线观看| 中文在线资源| 日本福利片高清在线观看| 在线中文字幕网站| 久久99久久99精品蜜柚传媒| 国产精彩视频在线观看免费蜜芽| 黄色一区三区| 一起操在线视频| 九色蝌蚪在线| 麻豆国产精品视频| 黄色网页免费在线观看| 日本三级中国三级99人妇网站| 一区二区三区欧美在线| 在线观看日韩精品视频|