• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            隨筆 - 40, 文章 - 0, 評論 - 9, 引用 - 0
            數據加載中……

            Linux下的多進程編程初步

            http://www.bccn.net/Article/czxt/linux/200511/1037.html

            文章摘要:
               多線程程序設計的概念早在六十年代就被提出,但直到八十年代中期,Unix系統中才引入多線程機制,如今,由于自身的許多優點,多線程編程已經得到了廣泛的應用。本文我們將介紹在Linux下編寫多進程和多線程程序的一些初步知識。


            --------------------------------------------------------------------------------

            正文:
            Linux下的多進程編程初步

            1 引言
            對于沒有接觸過Unix/Linux操作系統的人來說,fork是最難理解的概念之一:它執行一次卻返回兩個值。fork函數是Unix系統最杰出的成就 之一,它是七十年代UNIX早期的開發者經過長期在理論和實踐上的艱苦探索后取得的成果,一方面,它使操作系統在進程管理上付出了最小的代價,另一方面, 又為程序員提供了一個簡潔明了的多進程方法。與DOS和早期的Windows不同,Unix/Linux系統是真正實現多任務操作的系統,可以說,不使用 多進程編程,就不能算是真正的Linux環境下編程。
               多線程程序設計的概念早在六十年代就被提出,但直到八十年代中期,Unix系統中才引入多線程機制,如今,由于自身的許多優點,多線程編程已經得到了廣泛的應用。
               下面,我們將介紹在Linux下編寫多進程和多線程程序的一些初步知識。

            2 多進程編程
            什么是一個進程?進程這個概念是針對系統而不是針對用戶的,對用戶來說,他面對的概念是程序。當用戶敲入命令執行一個程序的時候,對系統而言,它將啟動一 個進程。但和程序不同的是,在這個進程中,系統可能需要再啟動一個或多個進程來完成獨立的多個任務。多進程編程的主要內容包括進程控制和進程間通信,在了 解這些之前,我們先要簡單知道進程的結構。

              2.1 Linux下進程的結構
               Linux下一個進程在內存里有三部分的數據,就是"代碼段"、"堆棧段"和"數據段"。其實學過匯編語言的人一定知道,一般的CPU都有上述三種段寄存器,以方便操作系統的運行。這三個部分也是構成一個完整的執行序列的必要的部分。
            "代碼段",顧名思義,就是存放了程序代碼的數據,假如機器中有數個進程運行相同的一個程序,那么它們就可以使用相同的代碼段。"堆棧段"存放的就是子程 序的返回地址、子程序的參數以及程序的局部變量。而數據段則存放程序的全局變量,常數以及動態數據分配的數據空間(比如用malloc之類的函數取得的空 間)。這其中有許多細節問題,這里限于篇幅就不多介紹了。系統如果同時運行數個相同的程序,它們之間就不能使用同一個堆棧段和數據段。

              2.2 Linux下的進程控制
            在傳統的Unix環境下,有兩個基本的操作用于創建和修改進程:函數fork( )用來創建一個新的進程,該進程幾乎是當前進程的一個完全拷貝;函數族exec( )用來啟動另外的進程以取代當前運行的進程。Linux的進程控制和傳統的Unix進程控制基本一致,只在一些細節的地方有些區別,例如在Linux系統 中調用vfork和fork完全相同,而在有些版本的Unix系統中,vfork調用有不同的功能。由于這些差別幾乎不影響我們大多數的編程,在這里我們 不予考慮。
               2.2.1 fork( )
               fork在英文中是"分叉"的意思。為什么取這個名字呢?因為一個進程在運行中,如果使用了fork,就產生了另一個進程,于是進程就"分叉"了,所以這個名字取得很形象。下面就看看如何具體使用fork,這段程序演示了使用fork的基本框架:

            void main(){
            int i;
            if ( fork() == 0 ) {
            /* 子進程程序 */
            for ( i = 1; i <1000; i ++ ) printf("This is child process\n");
            }
            else {
            /* 父進程程序*/
            for ( i = 1; i <1000; i ++ ) printf("This is process process\n");
            }
            }
               程序運行后,你就能看到屏幕上交替出現子進程與父進程各打印出的一千條信息了。如果程序還在運行中,你用ps命令就能看到系統中有兩個它在運行了。
            那么調用這個fork函數時發生了什么呢?fork函數啟動一個新的進程,前面我們說過,這個進程幾乎是當前進程的一個拷貝:子進程和父進程使用相同的代 碼段;子進程復制父進程的堆棧段和數據段。這樣,父進程的所有數據都可以留給子進程,但是,子進程一旦開始運行,雖然它繼承了父進程的一切數據,但實際上 數據卻已經分開,相互之間不再有影響了,也就是說,它們之間不再共享任何數據了。它們再要交互信息時,只有通過進程間通信來實現,這將是我們下面的內容。 既然它們如此相象,系統如何來區分它們呢?這是由函數的返回值來決定的。對于父進程,fork函數返回了子程序的進程號,而對于子程序,fork函數則返 回零。在操作系統中,我們用ps函數就可以看到不同的進程號,對父進程而言,它的進程號是由比它更低層的系統調用賦予的,而對于子進程而言,它的進程號即 是fork函數對父進程的返回值。在程序設計中,父進程和子進程都要調用函數fork()下面的代碼,而我們就是利用fork()函數對父子進程的不同返 回值用if...else...語句來實現讓父子進程完成不同的功能,正如我們上面舉的例子一樣。我們看到,上面例子執行時兩條信息是交互無規則的打印出 來的,這是父子進程獨立執行的結果,雖然我們的代碼似乎和串行的代碼沒有什么區別。
            讀者也許會問,如果一個大程序在運行中,它的數據段和堆棧都很大,一次fork就要復制一次,那么fork的系統開銷不是很大嗎?其實UNIX自有其解決 的辦法,大家知道,一般CPU都是以"頁"為單位來分配內存空間的,每一個頁都是實際物理內存的一個映像,象INTEL的CPU,其一頁在通常情況下是 4086字節大小,而無論是數據段還是堆棧段都是由許多"頁"構成的,fork函數復制這兩個段,只是"邏輯"上的,并非"物理"上的,也就是說,實際執 行fork時,物理空間上兩個進程的數據段和堆棧段都還是共享著的,當有一個進程寫了某個數據時,這時兩個進程之間的數據才有了區別,系統就將有區別的" 頁"從物理上也分開。系統在空間上的開銷就可以達到最小。
               下面演示一個足以"搞死"Linux的小程序,其源代碼非常簡單:
               void main()
               {
                 for( ; ; ) fork();
               }
            這個程序什么也不做,就是死循環地fork,其結果是程序不斷產生進程,而這些進程又不斷產生新的進程,很快,系統的進程就滿了,系統就被這么多不斷產生 的進程"撐死了"。當然只要系統管理員預先給每個用戶設置可運行的最大進程數,這個惡意的程序就完成不了企圖了。
               2.2.2 exec( )函數族
            下面我們來看看一個進程如何來啟動另一個程序的執行。在Linux中要使用exec函數族。系統調用execve()對當前進程進行替換,替換者為一個指 定的程序,其參數包括文件名(filename)、參數列表(argv)以及環境變量(envp)。exec函數族當然不止一個,但它們大致相同,在 Linux中,它們分別是:execl,execlp,execle,execv,execve和execvp,下面我只以execlp為例,其它函數究 竟與execlp有何區別,請通過manexec命令來了解它們的具體情況。
            一個進程一旦調用exec類函數,它本身就"死亡"了,系統把代碼段替換成新的程序的代碼,廢棄原有的數據段和堆棧段,并為新程序分配新的數據段與堆棧 段,唯一留下的,就是進程號,也就是說,對系統而言,還是同一個進程,不過已經是另一個程序了。(不過exec類函數中有的還允許繼承環境變量之類的信 息。)
               那么如果我的程序想啟動另一程序的執行但自己仍想繼續運行的話,怎么辦呢?那就是結合fork與exec的使用。下面一段代碼顯示如何啟動運行其它程序:

            char command[256];
            void main()
            {
            int rtn; /*子進程的返回數值*/
            while(1) {
            /* 從終端讀取要執行的命令 */
            printf( ">" );
            fgets( command, 256, stdin );
            command[strlen(command)-1] = 0;
            if ( fork() == 0 ) {
            /* 子進程執行此命令 */
            execlp( command, command );
            /* 如果exec函數返回,表明沒有正常執行命令,打印錯誤信息*/
            perror( command );
            exit( errorno );
            }
            else {
            /* 父進程, 等待子進程結束,并打印子進程的返回值 */
            wait ( &rtn );
            printf( " child process return %d\n",. rtn );
            }
            }
            }

            此程序從終端讀入命令并執行之,執行完成后,父進程繼續等待從終端讀入命令。熟悉DOS和WINDOWS系統調用的朋友一定知道DOS/WINDOWS也 有exec類函數,其使用方法是類似的,但DOS/WINDOWS還有spawn類函數,因為DOS是單任務的系統,它只能將"父進程"駐留在機器內再執 行"子進程",這就是spawn類的函數。WIN32已經是多任務的系統了,但還保留了spawn類函數,WIN32中實現spawn函數的方法同前述 UNIX中的方法差不多,開設子進程后父進程等待子進程結束后才繼續運行。UNIX在其一開始就是多任務的系統,所以從核心角度上講不需要spawn類函 數。
            在這一節里,我們還要講講system()和popen()函數。system()函數先調用fork(),然后再調用exec()來執行用戶的登錄 shell,通過它來查找可執行文件的命令并分析參數,最后它么使用wait()函數族之一來等待子進程的結束。函數popen()和函數 system()相似,不同的是它調用pipe()函數創建一個管道,通過它來完成程序的標準輸入和標準輸出。這兩個函數是為那些不太勤快的程序員設計 的,在效率和安全方面都有相當的缺陷,在可能的情況下,應該盡量避免。

              2.3 Linux下的進程間通信
            詳細的講述進程間通信在這里絕對是不可能的事情,而且筆者很難有信心說自己對這一部分內容的認識達到了什么樣的地步,所以在這一節的開頭首先向大家推薦著 名作者Richard Stevens的著名作品:《Advanced Programming in the UNIX Environment》,它的中文譯本《UNIX環境高級編程》已有機械工業出版社出版,原文精彩,譯文同樣地道,如果你的確對在Linux下編程有濃 厚的興趣,那么趕緊將這本書擺到你的書桌上或計算機旁邊來。說這么多實在是難抑心中的景仰之情,言歸正傳,在這一節里,我們將介紹進程間通信最最初步和最 最簡單的一些知識和概念。
            首先,進程間通信至少可以通過傳送打開文件來實現,不同的進程通過一個或多個文件來傳遞信息,事實上,在很多應用系統里,都使用了這種方法。但一般說來, 進程間通信(IPC:InterProcess Communication)不包括這種似乎比較低級的通信方法。Unix系統中實現進程間通信的方法很多,而且不幸的是,極少方法能在所有的Unix系 統中進行移植(唯一一種是半雙工的管道,這也是最原始的一種通信方式)。而Linux作為一種新興的操作系統,幾乎支持所有的Unix下常用的進程間通信 方法:管道、消息隊列、共享內存、信號量、套接口等等。下面我們將逐一介紹。

               2.3.1 管道
               管道是進程間通信中最古老的方式,它包括無名管道和有名管道兩種,前者用于父進程和子進程間的通信,后者用于運行于同一臺機器上的任意兩個進程間的通信。
               無名管道由pipe()函數創建:
               #include <unistd.h>
               int pipe(int filedis[2]);
               參數filedis返回兩個文件描述符:filedes[0]為讀而打開,filedes[1]為寫而打開。filedes[1]的輸出是filedes[0]的輸入。下面的例子示范了如何在父進程和子進程間實現通信。

            #define INPUT 0
            #define OUTPUT 1

            void main() {
            int file_descriptors[2];
            /*定義子進程號 */
            pid_t pid;
            char buf[256];
            int returned_count;
            /*創建無名管道*/
            pipe(file_descriptors);
            /*創建子進程*/
            if((pid = fork()) == -1) {
            printf("Error in fork\n");
            exit(1);
            }
            /*執行子進程*/
            if(pid == 0) {
            printf("in the spawned (child) process...\n");
            /*子進程向父進程寫數據,關閉管道的讀端*/
            close(file_descriptors[INPUT]);
            write(file_descriptors[OUTPUT], "test data", strlen("test data"));
            exit(0);
            } else {
            /*執行父進程*/
            printf("in the spawning (parent) process...\n");
            /*父進程從管道讀取子進程寫的數據,關閉管道的寫端*/
            close(file_descriptors[OUTPUT]);
            returned_count = read(file_descriptors[INPUT], buf, sizeof(buf));
            printf("%d bytes of data received from spawned process: %s\n",
            returned_count, buf);
            }
            }
            在Linux系統下,有名管道可由兩種方式創建:命令行方式mknod系統調用和函數mkfifo。下面的兩種途徑都在當前目錄下生成了一個名為myfifo的有名管道:
                 方式一:mkfifo("myfifo","rw");
                 方式二:mknod myfifo p
               生成了有名管道后,就可以使用一般的文件I/O函數如open、close、read、write等來對它進行操作。下面即是一個簡單的例子,假設我們已經創建了一個名為myfifo的有名管道。
              /* 進程一:讀有名管道*/
            #include <stdio.h>
            #include <unistd.h>
            void main() {
            FILE * in_file;
            int count = 1;
            char buf[80];
            in_file = fopen("mypipe", "r");
            if (in_file == NULL) {
            printf("Error in fdopen.\n");
            exit(1);
            }
            while ((count = fread(buf, 1, 80, in_file)) > 0)
            printf("received from pipe: %s\n", buf);
            fclose(in_file);
            }
              /* 進程二:寫有名管道*/
            #include <stdio.h>
            #include <unistd.h>
            void main() {
            FILE * out_file;
            int count = 1;
            char buf[80];
            out_file = fopen("mypipe", "w");
            if (out_file == NULL) {
            printf("Error opening pipe.");
            exit(1);
            }
            sprintf(buf,"this is test data for the named pipe example\n");
            fwrite(buf, 1, 80, out_file);
            fclose(out_file);
            }

               2.3.2 消息隊列
               消息隊列用于運行于同一臺機器上的進程間通信,它和管道很相似,事實上,它是一種正逐漸被淘汰的通信方式,我們可以用流管道或者套接口的方式來取代它,所以,我們對此方式也不再解釋,也建議讀者忽略這種方式。

               2.3.3 共享內存
            共享內存是運行在同一臺機器上的進程間通信最快的方式,因為數據不需要在不同的進程間復制。通常由一個進程創建一塊共享內存區,其余進程對這塊內存區進行 讀寫。得到共享內存有兩種方式:映射/dev/mem設備和內存映像文件。前一種方式不給系統帶來額外的開銷,但在現實中并不常用,因為它控制存取的將是 實際的物理內存,在Linux系統下,這只有通過限制Linux系統存取的內存才可以做到,這當然不太實際。常用的方式是通過shmXXX函數族來實現利 用共享內存進行存儲的。
               首先要用的函數是shmget,它獲得一個共享存儲標識符。
                 #include <sys/types.h>
                 #include <sys/ipc.h>
                 #include <sys/shm.h>
                  int shmget(key_t key, int size, int flag);
            這個函數有點類似大家熟悉的malloc函數,系統按照請求分配size大小的內存用作共享內存。Linux系統內核中每個IPC結構都有的一個非負整數 的標識符,這樣對一個消息隊列發送消息時只要引用標識符就可以了。這個標識符是內核由IPC結構的關鍵字得到的,這個關鍵字,就是上面第一個函數的 key。數據類型key_t是在頭文件sys/types.h中定義的,它是一個長整形的數據。在我們后面的章節中,還會碰到這個關鍵字。
               當共享內存創建后,其余進程可以調用shmat()將其連接到自身的地址空間中。
               void *shmat(int shmid, void *addr, int flag);
               shmid為shmget函數返回的共享存儲標識符,addr和flag參數決定了以什么方式來確定連接的地址,函數的返回值即是該進程數據段所連接的實際地址,進程可以對此進程進行讀寫操作。
            使用共享存儲來實現進程間通信的注意點是對數據存取的同步,必須確保當一個進程去讀取數據時,它所想要的數據已經寫好了。通常,信號量被要來實現對共享存 儲數據存取的同步,另外,可以通過使用shmctl函數設置共享存儲內存的某些標志位如SHM_LOCK、SHM_UNLOCK等來實現。

               2.3.4 信號量
               信號量又稱為信號燈,它是用來協調不同進程間的數據對象的,而最主要的應用是前一節的共享內存方式的進程間通信。本質上,信號量是一個計數器,它用來記錄對某個資源(如共享內存)的存取狀況。一般說來,為了獲得共享資源,進程需要執行下列操作:
               (1) 測試控制該資源的信號量。
               (2) 若此信號量的值為正,則允許進行使用該資源。進程將進號量減1。
               (3) 若此信號量為0,則該資源目前不可用,進程進入睡眠狀態,直至信號量值大于0,進程被喚醒,轉入步驟(1)。
               (4) 當進程不再使用一個信號量控制的資源時,信號量值加1。如果此時有進程正在睡眠等待此信號量,則喚醒此進程。
            維護信號量狀態的是Linux內核操作系統而不是用戶進程。我們可以從頭文件/usr/src/linux/include /linux /sem.h 中看到內核用來維護信號量狀態的各個結構的定義。信號量是一個數據集合,用戶可以單獨使用這一集合的每個元素。要調用的第一個函數是semget,用以獲 得一個信號量ID。
               #include <sys/types.h>
               #include <sys/ipc.h>
               #include <sys/sem.h>
               int semget(key_t key, int nsems, int flag);
            key是前面講過的IPC結構的關鍵字,它將來決定是創建新的信號量集合,還是引用一個現有的信號量集合。nsems是該集合中的信號量數。如果是創建新 集合(一般在服務器中),則必須指定nsems;如果是引用一個現有的信號量集合(一般在客戶機中)則將nsems指定為0。
               semctl函數用來對信號量進行操作。
               int semctl(int semid, int semnum, int cmd, union semun arg);
               不同的操作是通過cmd參數來實現的,在頭文件sem.h中定義了7種不同的操作,實際編程時可以參照使用。
               semop函數自動執行信號量集合上的操作數組。
               int semop(int semid, struct sembuf semoparray[], size_t nops);
               semoparray是一個指針,它指向一個信號量操作數組。nops規定該數組中操作的數量。
               下面,我們看一個具體的例子,它創建一個特定的IPC結構的關鍵字和一個信號量,建立此信號量的索引,修改索引指向的信號量的值,最后我們清除信號量。在下面的代碼中,函數ftok生成我們上文所說的唯一的IPC關鍵字。

            #include <stdio.h>
            #include <sys/types.h>
            #include <sys/sem.h>
            #include <sys/ipc.h>
            void main() {
            key_t unique_key; /* 定義一個IPC關鍵字*/
            int id;
            struct sembuf lock_it;
            union semun options;
            int i;

            unique_key = ftok(".", 'a'); /* 生成關鍵字,字符'a'是一個隨機種子*/
            /* 創建一個新的信號量集合*/
            id = semget(unique_key, 1, IPC_CREAT | IPC_EXCL | 0666);
            printf("semaphore id=%d\n", id);
            options.val = 1; /*設置變量值*/
            semctl(id, 0, SETVAL, options); /*設置索引0的信號量*/

            /*打印出信號量的值*/
            i = semctl(id, 0, GETVAL, 0);
            printf("value of semaphore at index 0 is %d\n", i);

            /*下面重新設置信號量*/
            lock_it.sem_num = 0; /*設置哪個信號量*/
            lock_it.sem_op = -1; /*定義操作*/
            lock_it.sem_flg = IPC_NOWAIT; /*操作方式*/
            if (semop(id, &lock_it, 1) == -1) {
            printf("can not lock semaphore.\n");
            exit(1);
            }

            i = semctl(id, 0, GETVAL, 0);
            printf("value of semaphore at index 0 is %d\n", i);

            /*清除信號量*/
            semctl(id, 0, IPC_RMID, 0);
            }

               2.3.5 套接口
            套接口(socket)編程是實現Linux系統和其他大多數操作系統中進程間通信的主要方式之一。我們熟知的WWW服務、FTP服務、TELNET服務 等都是基于套接口編程來實現的。除了在異地的計算機進程間以外,套接口同樣適用于本地同一臺計算機內部的進程間通信。關于套接口的經典教材同樣是 Richard Stevens編著的《Unix網絡編程:聯網的API和套接字》,清華大學出版社出版了該書的影印版。它同樣是Linux程序員的必備書籍之一。
            關于這一部分的內容,可以參照本文作者的另一篇文章《設計自己的網絡螞蟻》,那里由常用的幾個套接口函數的介紹和示例程序。這一部分或許是Linux進程 間通信編程中最須關注和最吸引人的一部分,畢竟,Internet 正在我們身邊以不可思議的速度發展著,如果一個程序員在設計編寫他下一個程序的時候,根本沒有考慮到網絡,考慮到Internet,那么,可以說,他的設 計很難成功。

            3 Linux的進程和Win32的進程/線程比較
               熟悉WIN32編程的人一定知道,WIN32的進程管理方式與Linux上有著很大區別,在UNIX里,只有進程的概念,但在WIN32里卻還有一個"線程"的概念,那么Linux和WIN32在這里究竟有著什么區別呢?
            WIN32里的進程/線程是繼承自OS/2的。在WIN32里,"進程"是指一個程序,而"線程"是一個"進程"里的一個執行"線索"。從核心上 講,WIN32的多進程與Linux并無多大的區別,在WIN32里的線程才相當于Linux的進程,是一個實際正在執行的代碼。但是,WIN32里同一 個進程里各個線程之間是共享數據段的。這才是與Linux的進程最大的不同。
               下面這段程序顯示了WIN32下一個進程如何啟動一個線程。

            int g;
            DWORD WINAPI ChildProcess( LPVOID lpParameter ){
            int i;
            for ( i = 1; i <1000; i ++) {
            g ++;
            printf( "This is Child Thread: %d\n", g );
            }
            ExitThread( 0 );
            };

            void main()
            {
            int threadID;
            int i;
            g = 0;
            CreateThread( NULL, 0, ChildProcess, NULL, 0, &threadID );
            for ( i = 1; i <1000; i ++) {
            g ++;
            printf( "This is Parent Thread: %d\n", g );
            }
            }

            在WIN32下,使用CreateThread函數創建線程,與Linux下創建進程不同,WIN32線程不是從創建處開始運行的,而是由 CreateThread指定一個函數,線程就從那個函數處開始運行。此程序同前面的UNIX程序一樣,由兩個線程各打印1000條信息。 threadID是子線程的線程號,另外,全局變量g是子線程與父線程共享的,這就是與Linux最大的不同之處。大家可以看出,WIN32的進程/線程 要比Linux復雜,在Linux要實現類似WIN32的線程并不難,只要fork以后,讓子進程調用ThreadProc函數,并且為全局變量開設共享 數據區就行了,但在WIN32下就無法實現類似fork的功能了。所以現在WIN32下的C語言編譯器所提供的庫函數雖然已經能兼容大多數 Linux/UNIX的庫函數,但卻仍無法實現fork。
            對于多任務系統,共享數據區是必要的,但也是一個容易引起混亂的問題,在WIN32下,一個程序員很容易忘記線程之間的數據是共享的這一情況,一個線程修 改過一個變量后,另一個線程卻又修改了它,結果引起程序出問題。但在Linux下,由于變量本來并不共享,而由程序員來顯式地指定要共享的數據,使程序變 得更清晰與安全。
            至于WIN32的"進程"概念,其含義則是"應用程序",也就是相當于UNIX下的exec了。

            posted on 2008-10-08 15:47 閱讀(1551) 評論(0)  編輯 收藏 引用 所屬分類: liunx編程技術

            色欲久久久天天天综合网| 91久久精品无码一区二区毛片| 久久成人国产精品一区二区| 国产精品一区二区久久精品无码 | 精品久久久无码中文字幕天天| 蜜桃麻豆www久久国产精品| 精品国产99久久久久久麻豆| 久久久久国产精品熟女影院| 99久久精品无码一区二区毛片 | 精品久久香蕉国产线看观看亚洲| 久久国产成人亚洲精品影院| 久久久SS麻豆欧美国产日韩| 7国产欧美日韩综合天堂中文久久久久 | 久久国产精品二国产精品| 久久久久亚洲AV片无码下载蜜桃| 久久久久一级精品亚洲国产成人综合AV区| 久久精品成人欧美大片| 久久午夜综合久久| 91久久精品91久久性色| 人妻无码αv中文字幕久久琪琪布| 久久婷婷国产麻豆91天堂| 国产精品99久久久精品无码| 久久国产精品免费一区二区三区| 久久国产精品成人影院| 久久久久亚洲AV片无码下载蜜桃| 久久人人爽人人爽人人片AV东京热 | 久久这里只有精品首页| 日韩人妻无码精品久久免费一| 无码人妻久久一区二区三区蜜桃| 青青国产成人久久91网 | 国产69精品久久久久99| 天天躁日日躁狠狠久久| 久久久国产打桩机| 精品国产乱码久久久久软件| 无码人妻久久一区二区三区蜜桃| 伊人久久五月天| 天天做夜夜做久久做狠狠| 久久亚洲视频| 人人狠狠综合久久亚洲高清| 久久强奷乱码老熟女网站| 日韩欧美亚洲国产精品字幕久久久|