青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

隨筆 - 40, 文章 - 0, 評論 - 9, 引用 - 0
數據加載中……

system V隊列

Linux環境進程間通信(三):消息隊列
作者:鄭彥興    閱讀人次:2704    文章來源:developerWorks    發布時間:2007-8-29    網友評論(0)條
 
本系列文章中的前兩部分,我們探討管道信號兩種通信機制,本文將深入第三部分,介紹系統 V 消息隊列及其相應 API。

消息隊列(也叫做報文隊列)能夠克服早期unix通信機制的一些缺點。作為早期unix通信機制之一的信號能夠傳送的信息量有限,后來雖然 POSIX 1003.1b在信號的實時性方面作了拓廣,使得信號在傳遞信息量方面有了相當程度的改進,但是信號這種通信方式更像"即時"的通信方式,它要求接受信號 的進程在某個時間范圍內對信號做出反應,因此該信號最多在接受信號進程的生命周期內才有意義,信號所傳遞的信息是接近于隨進程持續的概念 (process-persistent),見附錄 1;管道及有名管道及有名管道則是典型的隨進程持續IPC,并且,只能傳送無格式的字節流無疑會給應用程序開發帶來不便,另外,它的緩沖區大小也受到限制。

消息隊列就是一個消息的鏈表。可以把消息看作一個記錄,具有特定的格式以及特定的優先級。對消息隊列有寫權限的進程可以向中按照一定的規則添加新消息;對消息隊列有讀權限的進程則可以從消息隊列中讀走消息。消息隊列是隨內核持續的(參見附錄 1)。

目前主要有兩種類型的消息隊列:POSIX消息隊列以及系統V消息隊列,系統V消息隊列目前被大量使用。考慮到程序的可移植性,新開發的應用程序應盡量使用POSIX消息隊列。

在本系列專題的序(深刻理解Linux進程間通信(IPC))中,提到對于消息隊列、信號燈、以及共享內存區來說,有兩個實現版本:POSIX的以 及系統V的。Linux內核(內核2.4.18)支持POSIX信號燈、POSIX共享內存區以及POSIX消息隊列,但對于主流Linux發行版本之一 redhad8.0(內核2.4.18),還沒有提供對POSIX進程間通信API的支持,不過應該只是時間上的事。

因此,本文將主要介紹系統V消息隊列及其相應API。在沒有聲明的情況下,以下討論中指的都是系統V消息隊列。

一、消息隊列基本概念

  1. 系統V消息隊列是隨內核持續的,只有在內核重起或者顯示刪除一個消息隊列時,該消息隊列才會真正被刪除。因此系統中記錄消息隊列的數據結構(struct ipc_ids msg_ids)位于內核中,系統中的所有消息隊列都可以在結構msg_ids中找到訪問入口。
  2. 消息隊列就是一個消息的鏈表。每個消息隊列都有一個隊列頭,用結構struct msg_queue來描述(參見附錄 2)。隊列頭中包含了該消息隊列的大量信息,包括消息隊列鍵值、用戶ID、組ID、消息隊列中消息數目等等,甚至記錄了最近對消息隊列讀寫進程的ID。讀者可以訪問這些信息,也可以設置其中的某些信息。
  3. 下圖說明了內核與消息隊列是怎樣建立起聯系的:
    其中:struct ipc_ids msg_ids是內核中記錄消息隊列的全局數據結構;struct msg_queue是每個消息隊列的隊列頭。

從上圖可以看出,全局數據結構 struct ipc_ids msg_ids 可以訪問到每個消息隊列頭的第一個成員:struct kern_ipc_perm;而每個struct kern_ipc_perm能夠與具體的消息隊列對應起來是因為在該結構中,有一個key_t類型成員key,而key則唯一確定一個消息隊列。 kern_ipc_perm結構如下:

struct kern_ipc_perm{   //內核中記錄消息隊列的全局數據結構msg_ids能夠訪問到該結構;
key_t key; //該鍵值則唯一對應一個消息隊列
uid_t uid;
gid_t gid;
uid_t cuid;
gid_t cgid;
mode_t mode;
unsigned long seq;
}


二、操作消息隊列

對消息隊列的操作無非有下面三種類型:

1、 打開或創建消息隊列
消息隊列的內核持續性要求每個消息隊列都在系統范圍內對應唯一的鍵值,所以,要獲得一個消息隊列的描述字,只需提供該消息隊列的鍵值即可;

注:消息隊列描述字是由在系統范圍內唯一的鍵值生成的,而鍵值可以看作對應系統內的一條路經。

2、 讀寫操作

消息讀寫操作非常簡單,對開發人員來說,每個消息都類似如下的數據結構:

struct msgbuf{
long mtype;
char mtext[1];
};


mtype成員代表消息類型,從消息隊列中讀取消息的一個重要依據就是消息的類型;mtext是消息內容,當然長度不一定為1。因此,對于發送消息 來說,首先預置一個msgbuf緩沖區并寫入消息類型和內容,調用相應的發送函數即可;對讀取消息來說,首先分配這樣一個msgbuf緩沖區,然后把消息 讀入該緩沖區即可。

3、 獲得或設置消息隊列屬性:

消息隊列的信息基本上都保存在消息隊列頭中,因此,可以分配一個類似于消息隊列頭的結構(struct msqid_ds,見附錄 2),來返回消息隊列的屬性;同樣可以設置該數據結構。



消息隊列API

1、文件名到鍵值

#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok (char*pathname, char proj);


它返回與路徑pathname相對應的一個鍵值。該函數不直接對消息隊列操作,但在調用ipc(MSGGET,…)或msgget()來獲得消息隊列描述字前,往往要調用該函數。典型的調用代碼是:

   key=ftok(path_ptr, 'a');
ipc_id=ipc(MSGGET, (int)key, flags,0,NULL,0);



2、linux為操作系統V進程間通信的三種方式(消息隊列、信號燈、共享內存區)提供了一個統一的用戶界面:
int ipc(unsigned int call, int first, int second, int third, void *ptr, long fifth);

第一個參數指明對IPC對象的操作方式,對消息隊列而言共有四種操作:MSGSND、MSGRCV、MSGGET以及MSGCTL,分別代表向消息 隊列發送消息、從消息隊列讀取消息、打開或創建消息隊列、控制消息隊列;first參數代表唯一的IPC對象;下面將介紹四種操作。

  • int ipc(MSGGET, int first, int second, int third, void *ptr, long fifth);
    與該操作對應的系統V調用為:int msgget( (key_t)first,second)。
  • int ipc(MSGCTL, int first, int second, int third, void *ptr, long fifth)
    與該操作對應的系統V調用為:int msgctl( first,second, (struct msqid_ds*) ptr)。
  • int ipc(MSGSND, int first, int second, int third, void *ptr, long fifth);
    與該操作對應的系統V調用為:int msgsnd( first, (struct msgbuf*)ptr, second, third)。
  • int ipc(MSGRCV, int first, int second, int third, void *ptr, long fifth);
    與該操作對應的系統V調用為:int msgrcv( first,(struct msgbuf*)ptr, second, fifth,third),

注:本人不主張采用系統調用ipc(),而更傾向于采用系統V或者POSIX進程間通信API。原因如下:

  • 雖然該系統調用提供了統一的用戶界面,但正是由于這個特性,它的參數幾乎不能給出特定的實際意義(如以first、second來命名參數),在一定程度上造成開發不便。
  • 正如ipc手冊所說的:ipc()是linux所特有的,編寫程序時應注意程序的移植性問題;
  • 該系統調用的實現不過是把系統V IPC函數進行了封裝,沒有任何效率上的優勢;
  • 系統V在IPC方面的API數量不多,形式也較簡潔。

3.系統V消息隊列API
系統V消息隊列API共有四個,使用時需要包括幾個頭文件:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>


1)int msgget(key_t key, int msgflg)

參數key是一個鍵值,由ftok獲得;msgflg參數是一些標志位。該調用返回與健值key相對應的消息隊列描述字。

在以下兩種情況下,該調用將創建一個新的消息隊列:

  • 如果沒有消息隊列與健值key相對應,并且msgflg中包含了IPC_CREAT標志位;
  • key參數為IPC_PRIVATE;

參數msgflg可以為以下:IPC_CREAT、IPC_EXCL、IPC_NOWAIT或三者的或結果。

調用返回:成功返回消息隊列描述字,否則返回-1。

注:參數key設置成常數IPC_PRIVATE并不意味著其他進程不能訪問該消息隊列,只意味著即將創建新的消息隊列。

2)int msgrcv(int msqid, struct msgbuf *msgp, int msgsz, long msgtyp, int msgflg);
該系統調用從msgid代表的消息隊列中讀取一個消息,并把消息存儲在msgp指向的msgbuf結構中。

msqid為消息隊列描述字;消息返回后存儲在msgp指向的地址,msgsz指定msgbuf的mtext成員的長度(即消息內容的長度),msgtyp為請求讀取的消息類型;讀消息標志msgflg可以為以下幾個常值的或:

  • IPC_NOWAIT 如果沒有滿足條件的消息,調用立即返回,此時,errno=ENOMSG
  • IPC_EXCEPT 與msgtyp>0配合使用,返回隊列中第一個類型不為msgtyp的消息
  • IPC_NOERROR 如果隊列中滿足條件的消息內容大于所請求的msgsz字節,則把該消息截斷,截斷部分將丟失。

msgrcv手冊中詳細給出了消息類型取不同值時(>0; <0; =0),調用將返回消息隊列中的哪個消息。

msgrcv()解除阻塞的條件有三個:

  1. 消息隊列中有了滿足條件的消息;
  2. msqid代表的消息隊列被刪除;
  3. 調用msgrcv()的進程被信號中斷;

調用返回:成功返回讀出消息的實際字節數,否則返回-1。

3)int msgsnd(int msqid, struct msgbuf *msgp, int msgsz, int msgflg);
向msgid代表的消息隊列發送一個消息,即將發送的消息存儲在msgp指向的msgbuf結構中,消息的大小由msgze指定。

對發送消息來說,有意義的msgflg標志為IPC_NOWAIT,指明在消息隊列沒有足夠空間容納要發送的消息時,msgsnd是否等待。造成msgsnd()等待的條件有兩種:

  • 當前消息的大小與當前消息隊列中的字節數之和超過了消息隊列的總容量;
  • 當前消息隊列的消息數(單位"個")不小于消息隊列的總容量(單位"字節數"),此時,雖然消息隊列中的消息數目很多,但基本上都只有一個字節。

msgsnd()解除阻塞的條件有三個:
  1. 不滿足上述兩個條件,即消息隊列中有容納該消息的空間;
  2. msqid代表的消息隊列被刪除;
  3. 調用msgsnd()的進程被信號中斷;

調用返回:成功返回0,否則返回-1。

4)int msgctl(int msqid, int cmd, struct msqid_ds *buf);
該系統調用對由msqid標識的消息隊列執行cmd操作,共有三種cmd操作:IPC_STAT、IPC_SET 、IPC_RMID。

  1. IPC_STAT:該命令用來獲取消息隊列信息,返回的信息存貯在buf指向的msqid結構中;
  2. IPC_SET:該命令用來設置消息隊列的屬性,要設置的屬性存儲在buf指向的msqid結構中;可設置屬性包括:msg_perm.uid、msg_perm.gid、msg_perm.mode以及msg_qbytes,同時,也影響msg_ctime成員。
  3. IPC_RMID:刪除msqid標識的消息隊列;

調用返回:成功返回0,否則返回-1。

三、消息隊列的限制
每個消息隊列的容量(所能容納的字節數)都有限制,該值因系統不同而不同。在后面的應用實例中,輸出了redhat 8.0的限制,結果參見附錄 3

另一個限制是每個消息隊列所能容納的最大消息數:在redhad 8.0中,該限制是受消息隊列容量制約的:消息個數要小于消息隊列的容量(字節數)。

注:上述兩個限制是針對每個消息隊列而言的,系統對消息隊列的限制還有系統范圍內的最大消息隊列個數,以及整個系統范圍內的最大消息數。一般來說,實際開發過程中不會超過這個限制。

四、消息隊列應用實例
消息隊列應用相對較簡單,下面實例基本上覆蓋了對消息隊列的所有操作,同時,程序輸出結果有助于加深對前面所講的某些規則及消息隊列限制的理解。

#include <sys/types.h>
#include <sys/msg.h>
#include <unistd.h>
void msg_stat(int,struct msqid_ds );
main()
{
int gflags,sflags,rflags;
key_t key;
int msgid;
int reval;
struct msgsbuf{
int mtype;
char mtext[1];
}msg_sbuf;
struct msgmbuf
{
int mtype;
char mtext[10];
}msg_rbuf;
struct msqid_ds msg_ginfo,msg_sinfo;
char* msgpath="/unix/msgqueue";
key=ftok(msgpath,'a');
gflags=IPC_CREAT|IPC_EXCL;
msgid=msgget(key,gflags|00666);
if(msgid==-1)
{
printf("msg create error\n");
return;
}
//創建一個消息隊列后,輸出消息隊列缺省屬性
msg_stat(msgid,msg_ginfo);
sflags=IPC_NOWAIT;
msg_sbuf.mtype=10;
msg_sbuf.mtext[0]='a';
reval=msgsnd(msgid,&msg_sbuf,sizeof(msg_sbuf.mtext),sflags);
if(reval==-1)
{
printf("message send error\n");
}
//發送一個消息后,輸出消息隊列屬性
msg_stat(msgid,msg_ginfo);
rflags=IPC_NOWAIT|MSG_NOERROR;
reval=msgrcv(msgid,&msg_rbuf,4,10,rflags);
if(reval==-1)
printf("read msg error\n");
else
printf("read from msg queue %d bytes\n",reval);
//從消息隊列中讀出消息后,輸出消息隊列屬性
msg_stat(msgid,msg_ginfo);
msg_sinfo.msg_perm.uid=8;//just a try
msg_sinfo.msg_perm.gid=8;//
msg_sinfo.msg_qbytes=16388;
//此處驗證超級用戶可以更改消息隊列的缺省msg_qbytes
//注意這里設置的值大于缺省值
reval=msgctl(msgid,IPC_SET,&msg_sinfo);
if(reval==-1)
{
printf("msg set info error\n");
return;
}
msg_stat(msgid,msg_ginfo);
//驗證設置消息隊列屬性
reval=msgctl(msgid,IPC_RMID,NULL);//刪除消息隊列
if(reval==-1)
{
printf("unlink msg queue error\n");
return;
}
}
void msg_stat(int msgid,struct msqid_ds msg_info)
{
int reval;
sleep(1);//只是為了后面輸出時間的方便
reval=msgctl(msgid,IPC_STAT,&msg_info);
if(reval==-1)
{
printf("get msg info error\n");
return;
}
printf("\n");
printf("current number of bytes on queue is %d\n",msg_info.msg_cbytes);
printf("number of messages in queue is %d\n",msg_info.msg_qnum);
printf("max number of bytes on queue is %d\n",msg_info.msg_qbytes);
//每個消息隊列的容量(字節數)都有限制MSGMNB,值的大小因系統而異。在創建新的消息隊列時,//msg_qbytes的缺省值就是MSGMNB
printf("pid of last msgsnd is %d\n",msg_info.msg_lspid);
printf("pid of last msgrcv is %d\n",msg_info.msg_lrpid);
printf("last msgsnd time is %s", ctime(&(msg_info.msg_stime)));
printf("last msgrcv time is %s", ctime(&(msg_info.msg_rtime)));
printf("last change time is %s", ctime(&(msg_info.msg_ctime)));
printf("msg uid is %d\n",msg_info.msg_perm.uid);
printf("msg gid is %d\n",msg_info.msg_perm.gid);
}

程序輸出結果見附錄 3

小結:
消息隊列 與管道以及有名管道相比,具有更大的靈活性,首先,它提供有格式字節流,有利于減少開發人員的工作量;其次,消息具有類型,在實際應用中,可作為優先級使 用。這兩點是管道以及有名管道所不能比的。同樣,消息隊列可以在幾個進程間復用,而不管這幾個進程是否具有親緣關系,這一點與有名管道很相似;但消息隊列 是隨內核持續的,與有名管道(隨進程持續)相比,生命力更強,應用空間更大。

附錄 1在參考文獻[1]中,給出了IPC隨進程持續、隨內核持續以及隨文件系統持續的定義:

  1. 隨進程持續:IPC一直存在到打開IPC對象的最后一個進程關閉該對象為止。如管道和有名管道;
  2. 隨內核持續:IPC一直持續到內核重新自舉或者顯示刪除該對象為止。如消息隊列、信號燈以及共享內存等;
  3. 隨文件系統持續:IPC一直持續到顯示刪除該對象為止。

附錄 2
結構msg_queue用來描述消息隊列頭,存在于系統空間:

struct msg_queue {
struct kern_ipc_perm q_perm;
time_t q_stime; /* last msgsnd time */
time_t q_rtime; /* last msgrcv time */
time_t q_ctime; /* last change time */
unsigned long q_cbytes; /* current number of bytes on queue */
unsigned long q_qnum; /* number of messages in queue */
unsigned long q_qbytes; /* max number of bytes on queue */
pid_t q_lspid; /* pid of last msgsnd */
pid_t q_lrpid; /* last receive pid */
struct list_head q_messages;
struct list_head q_receivers;
struct list_head q_senders;
};


結構msqid_ds用來設置或返回消息隊列的信息,存在于用戶空間;

struct msqid_ds {
struct ipc_perm msg_perm;
struct msg *msg_first; /* first message on queue,unused */
struct msg *msg_last; /* last message in queue,unused */
__kernel_time_t msg_stime; /* last msgsnd time */
__kernel_time_t msg_rtime; /* last msgrcv time */
__kernel_time_t msg_ctime; /* last change time */
unsigned long msg_lcbytes; /* Reuse junk fields for 32 bit */
unsigned long msg_lqbytes; /* ditto */
unsigned short msg_cbytes; /* current number of bytes on queue */
unsigned short msg_qnum; /* number of messages in queue */
unsigned short msg_qbytes; /* max number of bytes on queue */
__kernel_ipc_pid_t msg_lspid; /* pid of last msgsnd */
__kernel_ipc_pid_t msg_lrpid; /* last receive pid */
};

//可以看出上述兩個結構很相似。

附錄 3消息隊列實例輸出結果:

current number of bytes on queue is 0
number of messages in queue is 0
max number of bytes on queue is 16384
pid of last msgsnd is 0
pid of last msgrcv is 0
last msgsnd time is Thu Jan 1 08:00:00 1970
last msgrcv time is Thu Jan 1 08:00:00 1970
last change time is Sun Dec 29 18:28:20 2002
msg uid is 0
msg gid is 0
//上面剛剛創建一個新消息隊列時的輸出
current number of bytes on queue is 1
number of messages in queue is 1
max number of bytes on queue is 16384
pid of last msgsnd is 2510
pid of last msgrcv is 0
last msgsnd time is Sun Dec 29 18:28:21 2002
last msgrcv time is Thu Jan 1 08:00:00 1970
last change time is Sun Dec 29 18:28:20 2002
msg uid is 0
msg gid is 0
read from msg queue 1 bytes
//實際讀出的字節數
current number of bytes on queue is 0
number of messages in queue is 0
max number of bytes on queue is 16384 //每個消息隊列最大容量(字節數)
pid of last msgsnd is 2510
pid of last msgrcv is 2510
last msgsnd time is Sun Dec 29 18:28:21 2002
last msgrcv time is Sun Dec 29 18:28:22 2002
last change time is Sun Dec 29 18:28:20 2002
msg uid is 0
msg gid is 0
current number of bytes on queue is 0
number of messages in queue is 0
max number of bytes on queue is 16388 //可看出超級用戶可修改消息隊列最大容量
pid of last msgsnd is 2510
pid of last msgrcv is 2510 //對操作消息隊列進程的跟蹤
last msgsnd time is Sun Dec 29 18:28:21 2002
last msgrcv time is Sun Dec 29 18:28:22 2002
last change time is Sun Dec 29 18:28:23 2002 //msgctl()調用對msg_ctime有影響
msg uid is 8
msg gid is 8


參考文獻:

  • UNIX網絡編程第二卷:進程間通信,作者:W.Richard Stevens,譯者:楊繼張,清華大學出版社。對POSIX以及系統V消息隊列都有闡述,對Linux環境下的程序開發有極大的啟發意義。
  • linux內核源代碼情景分析(上),毛德操、胡希明著,浙江大學出版社,給出了系統V消息隊列相關的源代碼分析。
  • http://www.fanqiang.com/a4/b2/20010508/113315.html,主要闡述linux下對文件的操作,詳細介紹了對文件的存取權限位,對IPC對象的存取權限同樣具有很好的借鑒意義。
  • msgget、msgsnd、msgrcv、msgctl手冊

posted on 2008-10-22 09:31 閱讀(668) 評論(0)  編輯 收藏 引用 所屬分類: liunx編程技術

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            亚洲女人天堂成人av在线| 国产美女精品视频| 91久久精品一区二区别| 蜜桃av噜噜一区| 久久亚洲图片| 亚洲精品免费看| 一区二区三区精密机械公司| 国产精品卡一卡二| 久久一区亚洲| 欧美日韩精品欧美日韩精品| 亚洲一卡久久| 久久精品国产第一区二区三区| 怡红院精品视频| 亚洲精品乱码久久久久久黑人 | 亚洲综合色激情五月| 国产三区二区一区久久| 欧美福利在线| 国产精品日韩精品| 欧美成人按摩| 欧美午夜精品理论片a级大开眼界| 香蕉亚洲视频| 免费成人性网站| 香蕉久久一区二区不卡无毒影院| 久久久久久噜噜噜久久久精品| 日韩视频在线观看国产| 午夜精品久久久久久久蜜桃app| 在线看不卡av| 亚洲一区二区三区欧美 | 欧美日韩日日骚| 久热精品视频在线免费观看| 欧美日韩一区二区在线视频 | 久久精品国产亚洲aⅴ| 欧美久久一级| 免费人成网站在线观看欧美高清| 欧美视频一区二区三区在线观看| 美女国内精品自产拍在线播放| 欧美四级伦理在线| 亚洲成色www久久网站| 国产日本欧美一区二区三区在线| 91久久精品国产91性色| 影音欧美亚洲| 欧美一区二区三区四区夜夜大片| 宅男噜噜噜66一区二区66| 久久久久久久波多野高潮日日 | 亚洲精品一区二区三区樱花| 久久大综合网| 欧美一区影院| 国产精品日韩精品欧美在线| av不卡免费看| 欧美一区二区三区免费大片| 麻豆成人在线播放| 久久精品盗摄| 国产精品国产a级| 亚洲激情网站| 亚洲精品资源美女情侣酒店| 久久色在线观看| 麻豆91精品91久久久的内涵| 国产一区二区三区自拍| 午夜在线电影亚洲一区| 久久av一区二区三区亚洲| 国产精品亚洲激情| 亚洲欧美国产不卡| 欧美在线视频一区二区| 国产欧美日韩精品丝袜高跟鞋| 在线视频日韩精品| 亚洲尤物精选| 国产精品入口福利| 亚洲一区欧美激情| 欧美在线观看视频在线| 国产亚洲欧美日韩在线一区| 欧美在线看片| 欧美va亚洲va日韩∨a综合色| 在线观看日韩专区| 欧美aa国产视频| 亚洲全部视频| 亚洲免费视频网站| 国产情人综合久久777777| 性欧美xxxx大乳国产app| 久久久综合免费视频| 亚洲激情网站免费观看| 欧美精品在线一区二区三区| 这里只有精品丝袜| 久久久久欧美| 亚洲精品乱码久久久久久蜜桃麻豆 | 国产精品日本精品| 久久爱www久久做| 欧美韩国一区| 亚洲伊人伊色伊影伊综合网| 国产欧亚日韩视频| 蜜臀久久99精品久久久画质超高清| 亚洲国产美女精品久久久久∴| 一区二区三区国产在线观看| 国产农村妇女毛片精品久久莱园子 | 麻豆精品在线视频| 一区二区成人精品| 国产视频精品免费播放| 裸体女人亚洲精品一区| 一本色道久久88综合日韩精品 | 亚洲视频 欧洲视频| 国产亚洲欧美aaaa| 欧美国产日韩精品| 午夜国产精品视频免费体验区| 欧美a级片网站| 午夜精品国产精品大乳美女| 亚洲国产激情| 国产日韩欧美成人| 欧美激情在线| 久久九九国产精品| 中国女人久久久| 欧美激情精品久久久久久蜜臀 | 免费成人av资源网| 亚洲午夜久久久| 在线欧美不卡| 国产精品私人影院| 欧美黑人一区二区三区| 羞羞答答国产精品www一本| 亚洲激情一区二区| 久久久亚洲综合| 午夜精品国产| 亚洲视频一区二区| 亚洲国产日韩在线一区模特| 国产日韩av一区二区| 欧美午夜国产| 欧美激情在线| 欧美二区在线| 免费观看不卡av| 久久国产精品久久久久久| 亚洲少妇在线| 一区二区精品| 亚洲毛片在线观看| 亚洲国产成人av| 麻豆久久久9性大片| 久久久噜噜噜久久人人看| 欧美一区午夜视频在线观看| 亚洲一区欧美二区| 亚洲一区二区在线播放| 一本一本a久久| 日韩一级在线| 一本色道久久综合精品竹菊| 亚洲精选久久| 亚洲毛片一区| 一区二区三区欧美日韩| 亚洲伦理在线| 一本一本久久a久久精品牛牛影视| 亚洲三级影院| 这里只有精品视频| 中日韩高清电影网| 亚洲欧美不卡| 欧美一级二级三级蜜桃| 久久精品成人一区二区三区| 久久久久久自在自线| 玖玖玖免费嫩草在线影院一区| 老**午夜毛片一区二区三区| 欧美高清不卡在线| 亚洲欧洲一区二区天堂久久| 99这里有精品| 香蕉国产精品偷在线观看不卡| 午夜视频久久久| 久久人人爽国产| 欧美区二区三区| 国产精品视频免费观看| 狠狠色综合日日| 亚洲精品中文字幕女同| 亚洲欧美99| 欧美刺激性大交免费视频| 亚洲人午夜精品| 亚洲永久免费精品| 久久亚洲风情| 欧美亚洲动漫精品| 国产深夜精品| 亚洲精品久久久久久下一站| 亚洲一区精彩视频| 老巨人导航500精品| 日韩一级免费| 久久成人在线| 欧美三级精品| 亚洲国产精品国自产拍av秋霞| 一本色道久久综合亚洲精品不卡| 欧美一区二区三区在线播放| 欧美激情第二页| 亚洲直播在线一区| 欧美1区3d| 国产欧美精品在线播放| 亚洲欧洲精品一区| 久久成人久久爱| 亚洲精品国产精品国产自| 欧美一级午夜免费电影| 欧美日韩高清不卡| 亚洲成人资源| 久久精品欧洲| 欧美www在线| 国产精品久久久久久久久搜平片 | 在线观看日韩av电影| 亚洲免费网站| 亚洲欧洲日韩在线| 久久嫩草精品久久久久| 国产精品日韩欧美一区二区三区| 亚洲久久在线| 欧美韩日高清| 久久夜色精品国产亚洲aⅴ|