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

posts - 2,  comments - 2,  trackbacks - 0
    如果你不投遞(POST)Overlapped I/O,那么I/O Completion Ports 只能為你提供一個Queue. 
    CreateIoCompletionPort的NumberOfConcurrentThreads:
1.只有當(dāng)?shù)诙€參數(shù)ExistingCompletionPort為NULL時它才有效,它是個max threads limits.
2.大家有誰把它設(shè)置為超出cpu個數(shù)的值,當(dāng)然不只是cpu個數(shù)的2倍,而是下面的MAX_THREADS 100甚至更大。
對于這個值的設(shè)定,msdn并沒有說非得設(shè)成cpu個數(shù)的2倍,而且也沒有把減少線程之間上下文交換這些影響扯到這里來。I/O Completion Ports MSDN:"If your transaction required a lengthy computation, a larger concurrency value will allow more threads to run. Each completion packet may take longer to finish, but more completion packets will be processed at the same time. "。
    對于struct OVERLAPPED,我們常會如下擴展,
typedef struct {
  WSAOVERLAPPED overlapped; //must be first member?   是的,必須是第一個。如果你不肯定,你可以試試。
  SOCKET client_s;
  SOCKADDR_IN client_addr;
  WORD optCode;//1--read,2--send.  有人常會定義這個數(shù)據(jù)成員,但也有人不用,爭議在send/WSASend,此時的同步和異步是否有必要? 至少我下面的server更本就沒用它。
  char buf[MAX_BUF_SIZE];
  WSABUF wsaBuf;//inited ?  這個不要忘了!
  DWORD numberOfBytesTransferred;
  DWORD flags;   

}QSSOverlapped;//for per connection
我下面的server框架的基本思想是:
One connection VS one thread in worker thread pool ,worker thread performs completionWorkerRoutine.
A Acceptor thread 專門用來accept socket,關(guān)聯(lián)至IOCP,并WSARecv:post Recv Completion Packet to IOCP.
在completionWorkerRoutine中有以下的職責(zé):
1.handle request,當(dāng)忙時增加completionWorkerThread數(shù)量但不超過maxThreads,post Recv Completion Packet to IOCP.
2.timeout時檢查是否空閑和當(dāng)前completionWorkerThread數(shù)量,當(dāng)空閑時保持或減少至minThreads數(shù)量.
3.對所有Accepted-socket管理生命周期,這里利用系統(tǒng)的keepalive probes,若想實現(xiàn)業(yè)務(wù)層"心跳探測"只需將QSS_SIO_KEEPALIVE_VALS_TIMEOUT 改回系統(tǒng)默認(rèn)的2小時.
下面結(jié)合源代碼,淺析一下IOCP:
socketserver.h
#ifndef __Q_SOCKET_SERVER__
#define __Q_SOCKET_SERVER__
#include <winsock2.h>
#include <mstcpip.h>
#define QSS_SIO_KEEPALIVE_VALS_TIMEOUT 30*60*1000
#define QSS_SIO_KEEPALIVE_VALS_INTERVAL 5*1000

#define MAX_THREADS 100
#define MAX_THREADS_MIN  10
#define MIN_WORKER_WAIT_TIMEOUT  20*1000
#define MAX_WORKER_WAIT_TIMEOUT  60*MIN_WORKER_WAIT_TIMEOUT

#define MAX_BUF_SIZE 1024

/*當(dāng)Accepted socket和socket關(guān)閉或發(fā)生異常時回調(diào)CSocketLifecycleCallback*/
typedef void (*CSocketLifecycleCallback)(SOCKET cs,int lifecycle);//lifecycle:0:OnAccepted,-1:OnClose//注意OnClose此時的socket未必可用,可能已經(jīng)被非正常關(guān)閉或其他異常.

/*協(xié)議處理回調(diào)*/
typedef int (*InternalProtocolHandler)(LPWSAOVERLAPPED overlapped);//return -1:SOCKET_ERROR

typedef struct Q_SOCKET_SERVER SocketServer;
DWORD initializeSocketServer(SocketServer ** ssp,WORD passive,WORD port,CSocketLifecycleCallback cslifecb,InternalProtocolHandler protoHandler,WORD minThreads,WORD maxThreads,long workerWaitTimeout);
DWORD startSocketServer(SocketServer *ss);
DWORD shutdownSocketServer(SocketServer *ss);

#endif
 qsocketserver.c      簡稱 qss,相應(yīng)的OVERLAPPED簡稱qssOl.
#include "socketserver.h"
#include "stdio.h"
typedef struct { 
  WORD passive;//daemon
  WORD port;
  WORD minThreads;
  WORD maxThreads;
  volatile long lifecycleStatus;//0-created,1-starting, 2-running,3-stopping,4-exitKeyPosted,5-stopped
  long  workerWaitTimeout;//wait timeout 
  CRITICAL_SECTION QSS_LOCK;
  volatile long workerCounter;
  volatile long currentBusyWorkers;
  volatile long CSocketsCounter;//Accepted-socket引用計數(shù)
  CSocketLifecycleCallback cslifecb;
  InternalProtocolHandler protoHandler;
  WORD wsaVersion;//=MAKEWORD(2,0);
  WSADATA wsData;
  SOCKET server_s;
  SOCKADDR_IN serv_addr;
  HANDLE iocpHandle;
}QSocketServer;

typedef struct {
  WSAOVERLAPPED overlapped; 
  SOCKET client_s;
  SOCKADDR_IN client_addr;
  WORD optCode;
  char buf[MAX_BUF_SIZE];
  WSABUF wsaBuf;
  DWORD numberOfBytesTransferred;
  DWORD flags;
}QSSOverlapped;

DWORD  acceptorRoutine(LPVOID);
DWORD  completionWorkerRoutine(LPVOID);

static void adjustQSSWorkerLimits(QSocketServer *qss){
  /*adjust size and timeout.*/
  /*if(qss->maxThreads <= 0) {
   qss->maxThreads = MAX_THREADS;
        } else if (qss->maxThreads < MAX_THREADS_MIN) {           
         qss->maxThreads = MAX_THREADS_MIN;
        }
        if(qss->minThreads >  qss->maxThreads) {
         qss->minThreads =  qss->maxThreads;
        }
        if(qss->minThreads <= 0) {
            if(1 == qss->maxThreads) {
             qss->minThreads = 1;
            } else {
             qss->minThreads = qss->maxThreads/2;
            }
        }
       
        if(qss->workerWaitTimeout<MIN_WORKER_WAIT_TIMEOUT)
         qss->workerWaitTimeout=MIN_WORKER_WAIT_TIMEOUT;
        if(qss->workerWaitTimeout>MAX_WORKER_WAIT_TIMEOUT)
         qss->workerWaitTimeout=MAX_WORKER_WAIT_TIMEOUT;        */
}

typedef struct{
 QSocketServer * qss;
 HANDLE th;
}QSSWORKER_PARAM;

static WORD addQSSWorker(QSocketServer *qss,WORD addCounter){
 WORD res=0;
 if(qss->workerCounter<qss->minThreads||(qss->currentBusyWorkers==qss->workerCounter&&qss->workerCounter<qss->maxThreads)){
  DWORD threadId;
  QSSWORKER_PARAM * pParam=NULL;
  int i=0;  
  EnterCriticalSection(&qss->QSS_LOCK);
  if(qss->workerCounter+addCounter<=qss->maxThreads)
   for(;i<addCounter;i++)
   {
    pParam=malloc(sizeof(QSSWORKER_PARAM));
    if(pParam){
     pParam->th=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)completionWorkerRoutine,pParam,CREATE_SUSPENDED,&threadId);
     pParam->qss=qss;
     ResumeThread(pParam->th);
     qss->workerCounter++,res++; 
    }    
   }  
  LeaveCriticalSection(&qss->QSS_LOCK);
 }  
 return res;
}

static void SOlogger(const char * msg,SOCKET s,int clearup){
 perror(msg);
 if(s>0)
 closesocket(s);
 if(clearup)
 WSACleanup();
}

static int _InternalEchoProtocolHandler(LPWSAOVERLAPPED overlapped){
 QSSOverlapped *qssOl=(QSSOverlapped *)overlapped;
 
 printf("numOfT:%d,WSARecvd:%s,\n",qssOl->numberOfBytesTransferred,qssOl->buf);
 //Sleep(500); 
 return send(qssOl->client_s,qssOl->buf,qssOl->numberOfBytesTransferred,0);
}

DWORD initializeSocketServer(SocketServer ** ssp,WORD passive,WORD port,CSocketLifecycleCallback cslifecb,InternalProtocolHandler protoHandler,WORD minThreads,WORD maxThreads,long workerWaitTimeout){
 QSocketServer * qss=malloc(sizeof(QSocketServer));
 qss->passive=passive>0?1:0;
 qss->port=port;
 qss->minThreads=minThreads;
 qss->maxThreads=maxThreads;
 qss->workerWaitTimeout=workerWaitTimeout;
 qss->wsaVersion=MAKEWORD(2,0); 
 qss->lifecycleStatus=0;
 InitializeCriticalSection(&qss->QSS_LOCK);
 qss->workerCounter=0;
 qss->currentBusyWorkers=0;
 qss->CSocketsCounter=0;
 qss->cslifecb=cslifecb,qss->protoHandler=protoHandler;
 if(!qss->protoHandler)
  qss->protoHandler=_InternalEchoProtocolHandler; 
 adjustQSSWorkerLimits(qss);
 *ssp=(SocketServer *)qss;
 return 1;
}

DWORD startSocketServer(SocketServer *ss){ 
 QSocketServer * qss=(QSocketServer *)ss;
 if(qss==NULL||InterlockedCompareExchange(&qss->lifecycleStatus,1,0))
  return 0; 
 qss->serv_addr.sin_family=AF_INET;
 qss->serv_addr.sin_port=htons(qss->port);
 qss->serv_addr.sin_addr.s_addr=INADDR_ANY;//inet_addr("127.0.0.1");
 if(WSAStartup(qss->wsaVersion,&qss->wsData)){  
  /*這里還有個插曲就是這個WSAStartup被調(diào)用的時候,它居然會啟動一條額外的線程,當(dāng)然稍后這條線程會自動退出的.不知WSAClearup又會如何?......*/

  SOlogger("WSAStartup failed.\n",0,0);
  return 0;
 }
 qss->server_s=socket(AF_INET,SOCK_STREAM,IPPROTO_IP);
 if(qss->server_s==INVALID_SOCKET){  
  SOlogger("socket failed.\n",0,1);
  return 0;
 }
 if(bind(qss->server_s,(LPSOCKADDR)&qss->serv_addr,sizeof(SOCKADDR_IN))==SOCKET_ERROR){  
  SOlogger("bind failed.\n",qss->server_s,1);
  return 0;
 }
 if(listen(qss->server_s,SOMAXCONN)==SOCKET_ERROR)/*這里來談?wù)?strong>backlog,很多人不知道設(shè)成何值,我見到過1,5,50,100的,有人說設(shè)定的越大越耗資源,的確,這里設(shè)成SOMAXCONN不代表windows會真的使用SOMAXCONN,而是" If set to SOMAXCONN, the underlying service provider responsible for socket s will set the backlog to a maximum reasonable value. ",同時在現(xiàn)實環(huán)境中,不同操作系統(tǒng)支持TCP緩沖隊列有所不同,所以還不如讓操作系統(tǒng)來決定它的值。像Apache這種服務(wù)器:
#ifndef DEFAULT_LISTENBACKLOG
#define DEFAULT_LISTENBACKLOG 511
#endif
*/
    {       
  SOlogger("listen failed.\n",qss->server_s,1);
        return 0;
    }
 qss->iocpHandle=CreateIoCompletionPort(INVALID_HANDLE_VALUE,NULL,0,/*NumberOfConcurrentThreads-->*/qss->maxThreads);
 //initialize worker for completion routine.
 addQSSWorker(qss,qss->minThreads);  
 qss->lifecycleStatus=2;
 {
  QSSWORKER_PARAM * pParam=malloc(sizeof(QSSWORKER_PARAM));
  pParam->qss=qss;
  pParam->th=NULL;
  if(qss->passive){
   DWORD threadId;
   pParam->th=CreateThread(NULL,0,(LPTHREAD_START_ROUTINE)acceptorRoutine,pParam,0,&threadId); 
  }else
   return acceptorRoutine(pParam);
 }
 return 1;
}

DWORD shutdownSocketServer(SocketServer *ss){
 QSocketServer * qss=(QSocketServer *)ss;
 if(qss==NULL||InterlockedCompareExchange(&qss->lifecycleStatus,3,2)!=2)
  return 0; 
 closesocket(qss->server_s/*listen-socket*/);//..other accepted-sockets associated with the listen-socket will not be closed,except WSACleanup is called.. 
 if(qss->CSocketsCounter==0)
  qss->lifecycleStatus=4,PostQueuedCompletionStatus(qss->iocpHandle,0,-1,NULL);
 WSACleanup();  
 return 1;
}

DWORD  acceptorRoutine(LPVOID ss){
 QSSWORKER_PARAM * pParam=(QSSWORKER_PARAM *)ss;
 QSocketServer * qss=pParam->qss;
 HANDLE curThread=pParam->th;
 QSSOverlapped *qssOl=NULL;
 SOCKADDR_IN client_addr;
 int client_addr_leng=sizeof(SOCKADDR_IN);
 SOCKET cs; 
 free(pParam);
 while(1){  
  printf("accept starting.....\n");
  cs/*Accepted-socket*/=accept(qss->server_s,(LPSOCKADDR)&client_addr,&client_addr_leng);
  if(cs==INVALID_SOCKET)
        {
   printf("accept failed:%d\n",GetLastError());   
            break;
        }else{//SO_KEEPALIVE,SIO_KEEPALIVE_VALS 這里是利用系統(tǒng)的"心跳探測",keepalive probes.linux:setsockopt,SOL_TCP:TCP_KEEPIDLE,TCP_KEEPINTVL,TCP_KEEPCNT
            struct tcp_keepalive alive,aliveOut;
            int so_keepalive_opt=1;
            DWORD outDW;
            if(!setsockopt(cs,SOL_SOCKET,SO_KEEPALIVE,(char *)&so_keepalive_opt,sizeof(so_keepalive_opt))){
               alive.onoff=TRUE;
               alive.keepalivetime=QSS_SIO_KEEPALIVE_VALS_TIMEOUT;
               alive.keepaliveinterval=QSS_SIO_KEEPALIVE_VALS_INTERVAL;
               if(WSAIoctl(cs,SIO_KEEPALIVE_VALS,&alive,sizeof(alive),&aliveOut,sizeof(aliveOut),&outDW,NULL,NULL)==SOCKET_ERROR){
                    printf("WSAIoctl SIO_KEEPALIVE_VALS failed:%d\n",GetLastError());   
                    break;
                }

            }else{
                     printf("setsockopt SO_KEEPALIVE failed:%d\n",GetLastError());   
                     break;
            }  
  }
  
  CreateIoCompletionPort((HANDLE)cs,qss->iocpHandle,cs,0);
  if(qssOl==NULL){
   qssOl=malloc(sizeof(QSSOverlapped));   
  }
  qssOl->client_s=cs;
  qssOl->wsaBuf.len=MAX_BUF_SIZE,qssOl->wsaBuf.buf=qssOl->buf,qssOl->numberOfBytesTransferred=0,qssOl->flags=0;//initialize WSABuf.
  memset(&qssOl->overlapped,0,sizeof(WSAOVERLAPPED));  
  {
   DWORD lastErr=GetLastError();
   int ret=0;
   SetLastError(0);
   ret=WSARecv(cs,&qssOl->wsaBuf,1,&qssOl->numberOfBytesTransferred,&qssOl->flags,&qssOl->overlapped,NULL);
   if(ret==0||(ret==SOCKET_ERROR&&GetLastError()==WSA_IO_PENDING)){
    InterlockedIncrement(&qss->CSocketsCounter);//Accepted-socket計數(shù)遞增.
    if(qss->cslifecb)
     qss->cslifecb(cs,0);
    qssOl=NULL;
   }    
   
   if(!GetLastError())
    SetLastError(lastErr);
  }
  
  printf("accept flags:%d ,cs:%d.\n",GetLastError(),cs);
 }//end while.

 if(qssOl)
  free(qssOl);
 if(qss)
  shutdownSocketServer((SocketServer *)qss);
 if(curThread)
  CloseHandle(curThread);

 return 1;
}

static int postRecvCompletionPacket(QSSOverlapped * qssOl,int SOErrOccurredCode){ 
 int SOErrOccurred=0; 
 DWORD lastErr=GetLastError();
 SetLastError(0);
 //SOCKET_ERROR:-1,WSA_IO_PENDING:997
 if(WSARecv(qssOl->client_s,&qssOl->wsaBuf,1,&qssOl->numberOfBytesTransferred,&qssOl->flags,&qssOl->overlapped,NULL)==SOCKET_ERROR
  &&GetLastError()!=WSA_IO_PENDING)//this case lastError maybe 64, 10054 
 {
  SOErrOccurred=SOErrOccurredCode;  
 }      
 if(!GetLastError())
  SetLastError(lastErr); 
 if(SOErrOccurred)
  printf("worker[%d] postRecvCompletionPacket SOErrOccurred=%d,preErr:%d,postedErr:%d\n",GetCurrentThreadId(),SOErrOccurred,lastErr,GetLastError());
 return SOErrOccurred;
}

DWORD  completionWorkerRoutine(LPVOID ss){
 QSSWORKER_PARAM * pParam=(QSSWORKER_PARAM *)ss;
 QSocketServer * qss=pParam->qss;
 HANDLE curThread=pParam->th;
 QSSOverlapped * qssOl=NULL;
 DWORD numberOfBytesTransferred=0;
 ULONG_PTR completionKey=0;
 int postRes=0,handleCode=0,exitCode=0,SOErrOccurred=0; 
 free(pParam);
 while(!exitCode){
  SetLastError(0);
  if(GetQueuedCompletionStatus(qss->iocpHandle,&numberOfBytesTransferred,&completionKey,(LPOVERLAPPED *)&qssOl,qss->workerWaitTimeout)){
   if(completionKey==-1&&qss->lifecycleStatus>=4)
   {
    printf("worker[%d] completionKey -1:%d \n",GetCurrentThreadId(),GetLastError());
    if(qss->workerCounter>1)
     PostQueuedCompletionStatus(qss->iocpHandle,0,-1,NULL);
    exitCode=1;
    break;
   }
   if(numberOfBytesTransferred>0){   
    
    InterlockedIncrement(&qss->currentBusyWorkers);
    addQSSWorker(qss,1);
    handleCode=qss->protoHandler((LPWSAOVERLAPPED)qssOl);    
    InterlockedDecrement(&qss->currentBusyWorkers);    
    
    if(handleCode>=0){
     SOErrOccurred=postRecvCompletionPacket(qssOl,1);
    }else
     SOErrOccurred=2;    
   }else{
    printf("worker[%d] numberOfBytesTransferred==0 ***** closesocket servS or cs *****,%d,%d ,ol is:%d\n",GetCurrentThreadId(),GetLastError(),completionKey,qssOl==NULL?0:1);
    SOErrOccurred=3;     
   }  
  }else{ //GetQueuedCompletionStatus rtn FALSE, lastError 64 ,995[timeout worker thread exit.] ,WAIT_TIMEOUT:258        
   if(qssOl){
    SOErrOccurred=postRecvCompletionPacket(qssOl,4);
   }else {    

    printf("worker[%d] GetQueuedCompletionStatus F:%d \n",GetCurrentThreadId(),GetLastError());
    if(GetLastError()!=WAIT_TIMEOUT){
     exitCode=2;     
    }else{//wait timeout     
     if(qss->lifecycleStatus!=4&&qss->currentBusyWorkers==0&&qss->workerCounter>qss->minThreads){
      EnterCriticalSection(&qss->QSS_LOCK);
      if(qss->lifecycleStatus!=4&&qss->currentBusyWorkers==0&&qss->workerCounter>qss->minThreads){
       qss->workerCounter--;//until qss->workerCounter decrease to qss->minThreads
       exitCode=3;      
      }
      LeaveCriticalSection(&qss->QSS_LOCK);
     }
    }    
   }    
  }//end GetQueuedCompletionStatus.

  if(SOErrOccurred){   
   if(qss->cslifecb)
    qss->cslifecb(qssOl->client_s,-1);
   /*if(qssOl)*/{
    closesocket(qssOl->client_s);
    free(qssOl);
   }
   if(InterlockedDecrement(&qss->CSocketsCounter)==0&&qss->lifecycleStatus>=3){    
    //for qss workerSize,PostQueuedCompletionStatus -1
    qss->lifecycleStatus=4,PostQueuedCompletionStatus(qss->iocpHandle,0,-1,NULL);        
    exitCode=4;
   }
  }
  qssOl=NULL,numberOfBytesTransferred=0,completionKey=0,SOErrOccurred=0;//for net while.
 }//end while.

 //last to do
 if(exitCode!=3){ 
  int clearup=0;
  EnterCriticalSection(&qss->QSS_LOCK);
  if(!--qss->workerCounter&&qss->lifecycleStatus>=4){//clearup QSS
    clearup=1;
  }
  LeaveCriticalSection(&qss->QSS_LOCK);
  if(clearup){
   DeleteCriticalSection(&qss->QSS_LOCK);
   CloseHandle(qss->iocpHandle);
   free(qss); 
  }
 }
 CloseHandle(curThread);
 return 1;
}
------------------------------------------------------------------------------------------------------------------------
    對于IOCP的LastError的辨別和處理是個難點,所以請注意我的completionWorkerRoutine的while結(jié)構(gòu),
結(jié)構(gòu)如下:
while(!exitCode){
    if(completionKey==-1){...break;}
    if(GetQueuedCompletionStatus){/*在這個if體中只要你投遞的OVERLAPPED is not NULL,那么這里你得到的就是.*/
        if(numberOfBytesTransferred>0){
               /*在這里handle request,記得要繼續(xù)投遞你的OVERLAPPED哦! */
        }else{
              /*這里可能客戶端或服務(wù)端closesocket(the socket),但是OVERLAPPED is not NULL,只要你投遞的不為NULL!*/
        }
    }else{/*在這里的if體中,雖然GetQueuedCompletionStatus return FALSE,但是不代表OVERLAPPED一定為NULL.特別是OVERLAPPED is not NULL的情況下,不要以為LastError發(fā)生了,就代表當(dāng)前的socket無用或發(fā)生致命的異常,比如發(fā)生lastError:995這種情況下此時的socket有可能是一切正常的可用的,你不應(yīng)該關(guān)閉它.*/
        if(OVERLAPPED is not NULL){
             /*這種情況下,請不管37,21繼續(xù)投遞吧!在投遞后再檢測錯誤.*/
        }else{ 

        }
    }
  if(socket error occured){

  }
  prepare for next while.

    行文倉促,難免有錯誤或不足之處,希望大家踴躍指正評論,謝謝!

    這個模型在性能上還是有改進的空間哦!

posted on 2010-06-26 17:30 adapterofcoms 閱讀(3904) 評論(1)  編輯 收藏 引用

FeedBack:
# re: 一個基于完成端口的TCP Server,淺析IOCP
2010-06-28 13:00 | 學(xué)習(xí)
挺好,多謝共享  回復(fù)  更多評論
  

只有注冊用戶登錄后才能發(fā)表評論。
網(wǎng)站導(dǎo)航: 博客園   IT新聞   BlogJava   博問   Chat2DB   管理



<2010年6月>
303112345
6789101112
13141516171819
20212223242526
27282930123
45678910

常用鏈接

留言簿

隨筆檔案(2)

搜索

  •  

最新評論

閱讀排行榜

評論排行榜

青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <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>
            一区二区三区不卡视频在线观看| 午夜精品一区二区三区在线播放| 久久在线91| 在线精品亚洲| 欧美国产日韩视频| 欧美大片免费看| 日韩视频在线一区| 99天天综合性| 国产午夜精品久久久| 男人插女人欧美| 欧美激情综合网| 午夜亚洲福利在线老司机| 欧美一区国产在线| 亚洲欧洲一区二区在线播放| 夜夜精品视频一区二区| 国产日韩欧美一区二区三区在线观看| 久久久久99精品国产片| 模特精品在线| 亚洲欧美一区二区精品久久久| 午夜视频一区| 99国产精品久久久久久久久久| 亚洲一区二区三区激情| 亚洲第一网站免费视频| 在线视频免费在线观看一区二区| 国产日韩精品一区| 亚洲激情在线| 国产欧美精品va在线观看| 欧美激情黄色片| 国产精品日韩欧美一区二区三区| 免费视频一区| 国产精品外国| 亚洲经典在线看| 国产综合视频| 这里是久久伊人| 亚洲欧洲免费视频| 亚洲欧美日韩一区在线观看| 亚洲精品美女久久久久| 性伦欧美刺激片在线观看| 一本色道久久综合亚洲精品小说| 欧美一区三区二区在线观看| 亚洲网友自拍| 欧美欧美午夜aⅴ在线观看| 久久香蕉国产线看观看av| 欧美日韩中文在线观看| 亚洲高清精品中出| 精品999久久久| 午夜精品久久久久久久久| 亚洲网在线观看| 欧美片网站免费| 亚洲电影在线观看| 亚洲国产精品999| 久久久999成人| 午夜精品亚洲| 国产精品久久久久999| 亚洲欧洲一区二区在线播放| 伊人色综合久久天天| 欧美在线亚洲| 久久免费99精品久久久久久| 国产乱码精品一区二区三| 一本高清dvd不卡在线观看| 亚洲美女诱惑| 欧美激情在线| 亚洲精选在线| 亚洲一区二区久久| 欧美日韩一区二区三区四区五区| 亚洲福利免费| 夜夜夜精品看看| 欧美欧美全黄| 亚洲视频专区在线| 午夜精品久久久久99热蜜桃导演| 国产精品成人一区二区| 亚洲午夜国产成人av电影男同| 亚洲综合导航| 国产精品视频成人| 亚洲欧美日韩另类| 久久资源在线| 亚洲日本电影| 欧美视频在线观看一区| 亚洲一区二区三区四区中文| 欧美一区二区三区在线看| 国产在线观看91精品一区| 久久久九九九九| 欧美激情亚洲另类| 国产精品99久久久久久人| 国产精品久久7| 久久精品99| 91久久久久久久久久久久久| 亚洲伊人第一页| 国产性色一区二区| 久久综合狠狠综合久久综青草| 亚洲国产精品成人va在线观看| 一本久久a久久精品亚洲| 国产精品三级久久久久久电影| 欧美一区久久| 亚洲欧洲一区二区在线播放| 亚洲女女做受ⅹxx高潮| 国内外成人在线视频| 欧美精品亚洲一区二区在线播放| 亚洲天堂免费观看| 欧美插天视频在线播放| 亚洲一区国产一区| 在线观看日韩www视频免费 | 一本色道久久| 国产精品永久| 欧美成人免费小视频| 中文在线资源观看视频网站免费不卡| 久久久国产精彩视频美女艺术照福利| 亚洲精品乱码久久久久久按摩观 | 欧美大片在线看| 亚洲综合社区| 亚洲国产精品成人va在线观看| 欧美一区影院| 亚洲每日在线| 激情久久五月天| 国产精品久久久久久av下载红粉| 老司机免费视频一区二区| 亚洲一区二区在| 亚洲人永久免费| 你懂的国产精品永久在线| 欧美一区二粉嫩精品国产一线天| 99v久久综合狠狠综合久久| 国产一区在线播放| 国产精品女主播一区二区三区| 牛夜精品久久久久久久99黑人| 欧美主播一区二区三区| 中国av一区| 亚洲美女av在线播放| 亚洲第一网站| 欧美69wwwcom| 老司机精品视频网站| 欧美中文字幕| 欧美一区二区日韩一区二区| 亚洲伊人一本大道中文字幕| 日韩亚洲欧美在线观看| 最新成人在线| 亚洲国产高清一区二区三区| 黄色一区二区三区四区| 国产一区二区三区电影在线观看 | 欧美激情精品久久久久| 久久香蕉国产线看观看av| 久久精品国产成人| 久久精品人人做人人爽电影蜜月| 性18欧美另类| 欧美伊久线香蕉线新在线| 欧美一级在线视频| 久久成人精品无人区| 欧美一区二区三区视频免费播放| 亚洲欧美一区二区精品久久久| 亚洲欧美激情视频在线观看一区二区三区| 日韩天堂在线观看| 一本综合精品| 亚洲免费一区二区| 欧美在线一二三区| 久久在线精品| 欧美日韩的一区二区| 国产精品爱啪在线线免费观看| 国产精品久久久久久av下载红粉| 国产精品视频第一区| 国产日产精品一区二区三区四区的观看方式 | 久久天天躁狠狠躁夜夜爽蜜月| 久久久久久999| 欧美成人亚洲成人| 亚洲国产美女精品久久久久∴| 99re热这里只有精品视频| 亚洲午夜在线观看| 久久国产欧美精品| 美女主播一区| 国产精品国产自产拍高清av| 国产视频一区三区| 亚洲国产成人久久综合一区| 一区二区欧美日韩视频| 性色av一区二区三区| 老司机67194精品线观看| 亚洲第一久久影院| 亚洲午夜激情| 久热精品视频在线观看一区| 欧美日韩情趣电影| 国内精品国产成人| 99亚洲视频| 久久夜色精品国产噜噜av| 亚洲激情一区二区三区| 午夜精品久久久久久久蜜桃app| 老司机aⅴ在线精品导航| 国产精品久久二区二区| 亚洲国产精品成人va在线观看| 亚洲网在线观看| 欧美大片91| 香蕉av777xxx色综合一区| 欧美激情va永久在线播放| 国产一区二区激情| 中文精品视频| 亚洲成在线观看| 欧美一区二区观看视频| 欧美日韩国产黄| 亚洲国产毛片完整版| 欧美一区影院| 亚洲图片欧洲图片日韩av| 欧美黑人多人双交| 伊人成年综合电影网| 校园激情久久|