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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

UDT 最新源码分析(三) -- UDT Socket 相关函数

發布時間:2023/12/14 编程问答 33 豆豆
生活随笔 收集整理的這篇文章主要介紹了 UDT 最新源码分析(三) -- UDT Socket 相关函数 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

UDT 最新源碼分析 -- UDT Socket 相關函數

  • UDT socket 建立與使用
    • 主要流程
      • C/S 模式
      • Rendezvous 模式
      • UDT epoll
    • UDT socket 創建
    • UDT socket setsockopt/getsockopt
    • UDT socket bind
    • UDT::listen
    • UDT connect
    • UDT accept
  • 總結
    • C/S 模式--四次握手
    • Rendezvous模式--三次握手

UDT socket 建立與使用

主要流程

C/S 模式

  • UDT::socket -> UDT::setsockopt -> UDT::connect -> UDT::send -> UDT::close
  • UDT::socket -> UDT::setsockopt -> UDT::bind -> UDT::listen -> UDT::accept -> UDT::recv -> UDT::close

Rendezvous 模式

  • UDT::setsockopt(usock, 0, UDT_RENDEZVOUS, &rendezvous, sizeof(bool))

UDT epoll

詳細源碼分析在下一篇文章中。

  • UDT::epoll_create -> UDT::epoll_add_usock/epoll_add_ssock -> UDT::epoll_wait/epoll_wait2 -> UDT::epoll_release

UDT socket 創建

  • UDT::socket -> CUDT::socket -> CUDTUnited::newSocket

UDT socket 的創建過程主要分為以下幾步:

  • 如果GC 線程未啟動,則首先啟動;
  • 新建 CUDTSocket,初始化新建內在變量 m_pUDT,以及地址信息 m_pSelfAddr,UDT socket 的標識 m_SocketID,UDT socket的狀態 m_Status 設置為 INIT, m_ListenSocket 初始設置為0;
  • 將 UDT socket中信息注冊到m_pUDT, 包括 m_SocketID,m_iSockType,m_iIPversion,m_pCache;
  • 將 UDT socket加入 全局m_Sockets map;
  • 返回標識 m_SocketID。
  • 首先代碼分析還是從對外提供的接口調用開始。

    UDTSOCKET CUDT::socket(int af, int type, int) {if (!s_UDTUnited.m_bGCStatus)s_UDTUnited.startup(); // 如果GC 線程未啟動,那么首先啟動return s_UDTUnited.newSocket(af, type); //創建一個 UDT socket }

    接下來是通過 newSocket 創建 UDT socket 的過程。

    UDTSOCKET CUDTUnited::newSocket(int af, int type) {CUDTSocket* ns = NULL;try{ns = new CUDTSocket; //新建UDT socketns->m_pUDT = new CUDT; //CUDT 在創建socket時新建if (AF_INET == af){ns->m_pSelfAddr = (sockaddr*)(new sockaddr_in);((sockaddr_in*)(ns->m_pSelfAddr))->sin_port = 0;}else{ns->m_pSelfAddr = (sockaddr*)(new sockaddr_in6); //支持IPv6((sockaddr_in6*)(ns->m_pSelfAddr))->sin6_port = 0;}}catch (...) { ... }CGuard::enterCS(m_IDLock);ns->m_SocketID = -- m_SocketID; //在初始化的隨機數值上進行遞減,作為 UDT socket IDCGuard::leaveCS(m_IDLock);ns->m_Status = INIT; //設置為 INIT 狀態ns->m_ListenSocket = 0;ns->m_pUDT->m_SocketID = ns->m_SocketID; //將剛剛獲得 UDT socket ID 注冊到 CUDT中ns->m_pUDT->m_iSockType = (SOCK_STREAM == type) ? UDT_STREAM : UDT_DGRAM; ns->m_pUDT->m_iIPversion = ns->m_iIPversion = af;ns->m_pUDT->m_pCache = m_pCache;// protect the m_Sockets structure.CGuard::enterCS(m_ControlLock);try{m_Sockets[ns->m_SocketID] = ns; //將 UDT socket加入 全局m_Sockets map}catch (...){//failure and rollback...}CGuard::leaveCS(m_ControlLock);return ns->m_SocketID; }

    m_SocketID 的初始化,在CUDTUnited構造函數中被初始化為一個隨機數。構造函數在系統初始化 startup 時被調用。

    CUDTUnited::CUDTUnited(): m_SocketID(0), {// Socket ID MUST start from a random valuesrand((unsigned int)CTimer::getTime());m_SocketID = 1 + (int)((1 << 30) * (double(rand()) / RAND_MAX)); }

    UDT socket setsockopt/getsockopt

    UDT Socket 參數設置。

    • CUDT::setsockopt -> CUDT::setOpt
    • CUDT::getsockopt -> CUDT::getOpt

    UDT 可配置的參數中包括一些系統內部自定義參數,這些參數的定義如下所示:

    enum UDTOpt {UDT_MSS, // the Maximum Transfer UnitUDT_SNDSYN, // if sending is blockingUDT_RCVSYN, // if receiving is blockingUDT_CC, // custom congestion control algorithmUDT_FC, // Flight flag size (window size)UDT_SNDBUF, // maximum buffer in sending queueUDT_RCVBUF, // UDT receiving buffer sizeUDT_LINGER, // waiting for unsent data when closingUDP_SNDBUF, // UDP sending buffer sizeUDP_RCVBUF, // UDP receiving buffer sizeUDT_MAXMSG, // maximum datagram message sizeUDT_MSGTTL, // time-to-live of a datagram messageUDT_RENDEZVOUS, // rendezvous connection modeUDT_SNDTIMEO, // send() timeoutUDT_RCVTIMEO, // recv() timeoutUDT_REUSEADDR, // reuse an existing port or create a new oneUDT_MAXBW, // maximum bandwidth (bytes per second) that the connection can useUDT_STATE, // current socket state, see UDTSTATUS, read onlyUDT_EVENT, // current avalable events associated with the socketUDT_SNDDATA, // size of data in the sending bufferUDT_RCVDATA // size of data available for recv };

    UDT socket bind

    • CUDT::bind -> CUDTUnited::bind
    • 不同形式
      • int CUDTUnited::bind(UDTSOCKET u, UDPSOCKET udpsock)
      • int CUDTUnited::bind(const UDTSOCKET u, const sockaddr* name, int namelen)

    UDT bind 過程涉及到的模塊較多,總的來說,就是將創建的 UDT socket 的信息注冊到一個復用器上,如果復用器不存在則創建。每個復用器保證用于一個端口,每個復用器有一個 channel, 用于 udp socket 的創建,端口綁定等,修改 UDT socket 狀態, 從 INIT 遷移到 OPENED。

    也就是說,UDT socket通過UDT bind 與復用器 CMultiplexer 關聯在一起,channel 作為 udp socket 的真正執行者進行運行,通過發送接收的兩個工作線程完成數據的收發。發送接收的兩個隊列屬于復用器,但是通過復用器ID使得 UDT socket 發送數據時直接與 channel 打交道,不再需要查找復用器。

    int CUDTUnited::bind(const UDTSOCKET u, ...) {CUDTSocket* s = locate(u);CGuard cg(s->m_ControlLock);// cannot bind a socket more than onceif (INIT != s->m_Status)throw CUDTException(5, 0, 0);s->m_pUDT->open(); //m_pUDT中一堆參數初始化updateMux(s, name); //更新復用器s->m_Status = OPENED; //更新UDT socket 狀態為 OPENED// copy address information of local nodes->m_pUDT->m_pSndQueue->m_pChannel->getSockAddr(s->m_pSelfAddr); return 0; }

    每個端口可能被多個UDT socket復用,所以綁定端口實際上是注冊到端口唯一的復用器上

    • void CUDTUnited::updateMux(CUDTSocket* s, const CUDTSocket* ls)
    • void CUDTUnited::updateMux(CUDTSocket* s, const sockaddr* addr, const UDPSOCKET* udpsock)
    void CUDTUnited::updateMux(CUDTSocket* s, const sockaddr* addr, const UDPSOCKET* udpsock) {CGuard cg(m_ControlLock);if ((s->m_pUDT->m_bReuseAddr) && (NULL != addr)){int port = (AF_INET == s->m_pUDT->m_iIPversion) ? ntohs(((sockaddr_in*)addr)->sin_port) : ntohs(((sockaddr_in6*)addr)->sin6_port);// find a reusable addressfor (map<int, CMultiplexer>::iterator i = m_mMultiplexer.begin(); i != m_mMultiplexer.end(); ++ i){if ((i->second.m_iIPversion == s->m_pUDT->m_iIPversion) && (i->second.m_iMSS == s->m_pUDT->m_iMSS) && i->second.m_bReusable){if (i->second.m_iPort == port) //找到端口對應復用器{// reuse the existing multiplexer++ i->second.m_iRefCount; //引用計數加一s->m_pUDT->m_pSndQueue = i->second.m_pSndQueue; //發送隊列s->m_pUDT->m_pRcvQueue = i->second.m_pRcvQueue; //接收隊列s->m_iMuxID = i->second.m_iID; //將復用器 ID 告知 UDT socketreturn;}}}}// a new multiplexer is neededCMultiplexer m;m.m_iMSS = s->m_pUDT->m_iMSS;m.m_iIPversion = s->m_pUDT->m_iIPversion;m.m_iRefCount = 1;m.m_bReusable = s->m_pUDT->m_bReuseAddr;m.m_iID = s->m_SocketID;// 新建傳輸channel,設置IP 版本,以及發送接收緩沖大小,默認65536m.m_pChannel = new CChannel(s->m_pUDT->m_iIPversion);m.m_pChannel->setSndBufSize(s->m_pUDT->m_iUDPSndBufSize);m.m_pChannel->setRcvBufSize(s->m_pUDT->m_iUDPRcvBufSize);try{ //創建并綁定真正用于傳輸的 udp socket,保存于 channel 中if (NULL != udpsock)m.m_pChannel->open(*udpsock);elsem.m_pChannel->open(addr);}catch (CUDTException& e){m.m_pChannel->close();delete m.m_pChannel;throw e;}//復用器相關參數賦值sockaddr* sa = (AF_INET == s->m_pUDT->m_iIPversion) ? (sockaddr*) new sockaddr_in : (sockaddr*) new sockaddr_in6;m.m_pChannel->getSockAddr(sa);m.m_iPort = (AF_INET == s->m_pUDT->m_iIPversion) ? ntohs(((sockaddr_in*)sa)->sin_port) : ntohs(((sockaddr_in6*)sa)->sin6_port);if (AF_INET == s->m_pUDT->m_iIPversion) delete (sockaddr_in*)sa; else delete (sockaddr_in6*)sa;m.m_pTimer = new CTimer;m.m_pSndQueue = new CSndQueue; //新建發送隊列m.m_pSndQueue->init(m.m_pChannel, m.m_pTimer);m.m_pRcvQueue = new CRcvQueue; //新建接收隊列m.m_pRcvQueue->init(32, s->m_pUDT->m_iPayloadSize, m.m_iIPversion, 1024, m.m_pChannel, m.m_pTimer);m_mMultiplexer[m.m_iID] = m;s->m_pUDT->m_pSndQueue = m.m_pSndQueue;s->m_pUDT->m_pRcvQueue = m.m_pRcvQueue;s->m_iMuxID = m.m_iID; }

    創建 channel 以后,新建發送接收隊列,同時還需要分別調用 init 函數,該方法的主要作用除了對一部分參數初始化外,就是創建工作線程 worker 。

    以發送隊列舉例,接收隊列以及具體的使用過程后續文章介紹:

    void CSndQueue::init(CChannel* c, CTimer* t) {m_pChannel = c;m_pTimer = t;m_pSndUList = new CSndUList;m_pSndUList->m_pWindowLock = &m_WindowLock;m_pSndUList->m_pWindowCond = &m_WindowCond;m_pSndUList->m_pTimer = m_pTimer;if (0 != pthread_create(&m_WorkerThread, NULL, CSndQueue::worker, this)){m_WorkerThread = 0;throw CUDTException(3, 1);} }

    對于發送隊列的工作線程:

    void* CSndQueue::worker(void* param) {CSndQueue* self = (CSndQueue*)param;while (!self->m_bClosing) //只要隊列處于正常狀態,就無限循環{uint64_t ts = self->m_pSndUList->getNextProcTime();if (ts > 0){// wait until next processing time of the first socket on the listuint64_t currtime;CTimer::rdtsc(currtime);if (currtime < ts) //未到時間,繼續等待self->m_pTimer->sleepto(ts);// it is time to send the next pktsockaddr* addr;CPacket pkt;if (self->m_pSndUList->pop(addr, pkt) < 0)continue;self->m_pChannel->sendto(addr, pkt);}else{// wait here if there is no sockets with data to be sent pthread_mutex_lock(&self->m_WindowLock);if (!self->m_bClosing && (self->m_pSndUList->m_iLastEntry < 0))pthread_cond_wait(&self->m_WindowCond, &self->m_WindowLock);//條件等待pthread_mutex_unlock(&self->m_WindowLock);}}return NULL; }

    UDT::listen

    • CUDT::listen -> CUDTUnited::listen

    UDT listen 在 UDT socket 處于OPENED 狀態時,開始端口監聽,從 UDT bind 可知,此時已經 bind 成功。一個端口上只能有一個 listening socket。這里的 listen 用于 C/S 模式,不支持匯合模式。對于已經處于監聽狀態的 UDT socket, 不會多次監聽。

    執行 UDT listen 成功后,m_bListening 修改為 true, UDT socket 狀態 m_Status 變成 LISTENING。在 UDT socket 中新建兩個集合 m_pQueuedSockets 和 m_pAcceptSockets,分別存放接收但還未來得及處理接受的連接請求 ,或者已經接受的連接請求。使用集合,也是借用集合元素唯一的特性。

    實際上,執行 UDT listen 是設置監聽到復用器中的接收隊列 m_pRcvQueue。在隊列的工作線程中,將會根據到來的包的類型進行對應的響應,并發送。

    int CUDTUnited::listen(const UDTSOCKET u, int backlog) {CUDTSocket* s = locate(u);CGuard cg(s->m_ControlLock);// do nothing if the socket is already listeningif (LISTENING == s->m_Status)return 0;// a socket can listen only if is in OPENED statusif (OPENED != s->m_Status)throw CUDTException(5, 5, 0);// listen is not supported in rendezvous connection setupif (s->m_pUDT->m_bRendezvous)throw CUDTException(5, 7, 0);if (backlog <= 0)throw CUDTException(5, 3, 0);s->m_uiBackLog = backlog;try{ //新建 接收到但未接受的socket集合與以及接受的集合,使用 set保證每個socket唯一s->m_pQueuedSockets = new set<UDTSOCKET>; s->m_pAcceptSockets = new set<UDTSOCKET>;}catch (...) { ... }s->m_pUDT->listen();s->m_Status = LISTENING;return 0; }

    通過接收隊列設置UDT 實例中的Listener,實際上是設置到復用器中

    void CUDT::listen() {CGuard cg(m_ConnectionLock);if (!m_bOpened)throw CUDTException(5, 0, 0);if (m_bConnecting || m_bConnected)throw CUDTException(5, 2, 0);// listen can be called more than onceif (m_bListening)return;// if there is already another socket listening on the same portif (m_pRcvQueue->setListener(this) < 0) // 為CRcvQueue 中 設置 = thisthrow CUDTException(5, 11, 0);m_bListening = true; //修改監聽狀態 }

    通過 UDT listen 設置,為復用器設置 Listener,當接收到數據時,將數據分發到對應的 UDT 實例。同時修改UDT 的當前狀態。

    在 setListener 以后, 非空,在CRcvQueue中線程函數 worker 循環中,會調用recvfrom 接收到連接請求,檢查m_Packet.m_iID, 決定是否調用 connect。

    void* CRcvQueue::worker(void* param) {CRcvQueue* self = (CRcvQueue*)param;sockaddr* addr = ...CUDT* u = NULL;int32_t id;while (!self->m_bClosing){unit->m_Packet.setLength(self->m_iPayloadSize);// reading next incoming packet, recvfrom returns -1 is nothing has been receivedif (self->m_pChannel->recvfrom(addr, unit->m_Packet) < 0)goto TIMER_CHECK;id = unit->m_Packet.m_iID;// ID 0 is for connection request, which should be passed to the listening socket or rendezvous socketsif (0 == id){if (NULL != self->m_pListener)self->m_pListener->listen(addr, unit->m_Packet);else if (NULL != (u = self->m_pRendezvousQueue->retrieve(addr, id))){// asynchronous connect: call connect here// otherwise wait for the UDT socket to retrieve this packetif (!u->m_bSynRecving)u->connect(unit->m_Packet);elseself->storePkt(id, unit->m_Packet.clone());}}} }

    對于到達的一個連接請求,如果非空,就可以調用 CUDT 中的私有 listen 方法。對到達的建立連接的包進行解析,生成 coockie字符串,如果時正常的連接請求,則調用發送隊列 sendto 發送包。如果是響應消息,且通過cookie 驗證,則建立新連接。

    int CUDT::listen(sockaddr* addr, CPacket& packet) {CHandShake hs;hs.deserialize(packet.m_pcData, packet.getLength());// SYN cookiechar clienthost[NI_MAXHOST];char clientport[NI_MAXSERV];getnameinfo(addr, (AF_INET == m_iVersion) ? sizeof(sockaddr_in) : sizeof(sockaddr_in6), clienthost, sizeof(clienthost), clientport, sizeof(clientport), NI_NUMERICHOST|NI_NUMERICSERV);int64_t timestamp = (CTimer::getTime() - m_StartTime) / 60000000; // secret changes every one minutestringstream cookiestr;cookiestr << clienthost << ":" << clientport << ":" << timestamp;unsigned char cookie[16];CMD5::compute(cookiestr.str().c_str(), cookie);// connection request type: 1: regular connection request, 0: rendezvous connection request, -1/-2: responseif (1 == hs.m_iReqType) {hs.m_iCookie = *(int*)cookie;packet.m_iID = hs.m_iID;int size = packet.getLength();hs.serialize(packet.m_pcData, size);m_pSndQueue->sendto(addr, packet);return 0;}else{if (hs.m_iCookie != *(int*)cookie){timestamp --;cookiestr << clienthost << ":" << clientport << ":" << timestamp;CMD5::compute(cookiestr.str().c_str(), cookie);if (hs.m_iCookie != *(int*)cookie)return -1;}}int32_t id = hs.m_iID;// When a peer side connects in...if ((1 == packet.getFlag()) && (0 == packet.getType())){ // 控制包,且當前為 Connection Handshakeif ((hs.m_iVersion != m_iVersion) || (hs.m_iType != m_iSockType)){// mismatch, reject the requesths.m_iReqType = 1002;int size = CHandShake::m_iContentSize;hs.serialize(packet.m_pcData, size);packet.m_iID = id;m_pSndQueue->sendto(addr, packet);}else{ int result = s_UDTUnited.newConnection(m_SocketID, addr, &hs);if (result == -1)hs.m_iReqType = 1002;// send back a response if connection failed or connection already existed// new connection response should be sent in connect()if (result != 1){int size = CHandShake::m_iContentSize;hs.serialize(packet.m_pcData, size);packet.m_iID = id;m_pSndQueue->sendto(addr, packet);}else{// a new connection has been created, enable epoll for write s_UDTUnited.m_EPoll.update_events(m_SocketID, m_sPollID, UDT_EPOLL_OUT, true);}}}return hs.m_iReqType; }

    在上面的代碼中,newConnection 主要作用是建立一個新的連接,實質上是將連接的對端信息加入到 UDT 中存放對端連接socket 記錄的 m_PeerRec map,并在 m_Sockets 加入新建立的 UDT socket。

    首先檢查 這個連接是否已經建立,如果已經建立,則返回已經存在 UDT socket 信息;如果當前處于 BROKEN 狀態,會修改狀態到 CLOSED, 并進行一些清理工作。其余的清理流程在 UDT 關閉過程中處理。

    如果是一個新的連接,新建 UDT socket, 初始化部分參數,包括 m_pSelfAddr, m_SocketID, m_ListenSocket,m_PeerID, m_iISN等,然后綁定新的地址到監聽socket。修改狀態為 CONNECTED。修改 m_PeerRec 與 m_Sockets, 插入socket 進入 m_pQueuedSockets, 更新本地節點信息, 更新事件與定時器。等待 accept 事件到來。

    int CUDTUnited::newConnection(const UDTSOCKET listen, const sockaddr* peer, CHandShake* hs) {CUDTSocket* ns = NULL;CUDTSocket* ls = locate(listen); //在 m_Sockets中查找本地 UDT socket。if (NULL == ls)return -1;// if this connection has already been processedif (NULL != (ns = locate(peer, hs->m_iID, hs->m_iISN))) //在 m_PeerRec 與 m_Sockets中查找。{if (ns->m_pUDT->m_bBroken){// last connection from the "peer" address has been broken...}else{// connection already exist, this is a repeated connection request// respond with existing HS information...return 0;//except for this situation a new connection should be started}}// exceeding backlog, refuse the connection requestif (ls->m_pQueuedSockets->size() >= ls->m_uiBackLog)return -1;try{ns = new CUDTSocket;ns->m_pUDT = new CUDT(*(ls->m_pUDT));...ns->m_pSelfAddr = ......}catch (...) { ... }CGuard::enterCS(m_IDLock);ns->m_SocketID = -- m_SocketID;CGuard::leaveCS(m_IDLock);ns->m_ListenSocket = listen;ns->m_iIPversion = ls->m_iIPversion;ns->m_pUDT->m_SocketID = ns->m_SocketID;ns->m_PeerID = hs->m_iID;ns->m_iISN = hs->m_iISN;int error = 0;try{// bind to the same addr of listening socketns->m_pUDT->open();updateMux(ns, ls);ns->m_pUDT->connect(peer, hs);}catch (...){error = 1;goto ERR_ROLLBACK;}ns->m_Status = CONNECTED;// copy address information of local nodens->m_pUDT->m_pSndQueue->m_pChannel->getSockAddr(ns->m_pSelfAddr);CIPAddress::pton(ns->m_pSelfAddr, ns->m_pUDT->m_piSelfIP, ns->m_iIPversion);// protect the m_Sockets structure.CGuard::enterCS(m_ControlLock);try{m_Sockets[ns->m_SocketID] = ns;m_PeerRec[(ns->m_PeerID << 30) + ns->m_iISN].insert(ns->m_SocketID);}catch (...){error = 2;}CGuard::leaveCS(m_ControlLock);CGuard::enterCS(ls->m_AcceptLock);try{ls->m_pQueuedSockets->insert(ns->m_SocketID);}catch (...){error = 3;}CGuard::leaveCS(ls->m_AcceptLock);// acknowledge users waiting for new connections on the listening socketm_EPoll.update_events(listen, ls->m_pUDT->m_sPollID, UDT_EPOLL_IN, true);CTimer::triggerEvent();ERR_ROLLBACK:if (error > 0){ns->m_pUDT->close();ns->m_Status = CLOSED;ns->m_TimeStamp = CTimer::getTime();return -1;}// wake up a waiting accept() call#ifndef WIN32pthread_mutex_lock(&(ls->m_AcceptLock));pthread_cond_signal(&(ls->m_AcceptCond));pthread_mutex_unlock(&(ls->m_AcceptLock));#elseSetEvent(ls->m_AcceptCond);#endifreturn 1; }

    UDT connect

    • CUDT::connect( api.cpp, CUDT::public method) -> CUDTUnited::connect -> CUDT::connect( core.cpp, CUDT::private method)

    如果UDT socket 能夠 connect,首先應該處于 INIT 或者 OPENED 狀態。如果處于 INIT狀態,表明為新創建的UDT Socket,需要初始化 m_pUDT 內參數并注冊到復用器,修改狀態為 OPENED。如果處于OPENED 狀態,可能已經被 bind 過,則可以進入 CONNECTING 狀態,并調用 m_pUDT->connect。記錄對端地址 m_pPeerAddr到該 UDT socket內部結構。

    int CUDTUnited::connect(const UDTSOCKET u, const sockaddr* name, int namelen) {CUDTSocket* s = locate(u);CGuard cg(s->m_ControlLock);// a socket can "connect" only if it is in INIT or OPENED statusif (INIT == s->m_Status){if (!s->m_pUDT->m_bRendezvous){s->m_pUDT->open();updateMux(s);s->m_Status = OPENED;}elsethrow CUDTException(5, 8, 0);}else if (OPENED != s->m_Status)throw CUDTException(5, 2, 0);// connect_complete() may be called before connect() returns.// So we need to update the status before connect() is called,// otherwise the status may be overwritten with wrong value (CONNECTED vs. CONNECTING).s->m_Status = CONNECTING;try{s->m_pUDT->connect(name);}catch (CUDTException e){s->m_Status = OPENED;throw e;}// record peer addressdelete s->m_pPeerAddr;if (AF_INET == s->m_iIPversion){s->m_pPeerAddr = (sockaddr*)(new sockaddr_in);memcpy(s->m_pPeerAddr, name, sizeof(sockaddr_in));}else{s->m_pPeerAddr = (sockaddr*)(new sockaddr_in6);memcpy(s->m_pPeerAddr, name, sizeof(sockaddr_in6));}return 0; }

    在 CUDT 中, 在private 方法中,connect 有三種形式:

    // Functionality: // Connect to a UDT entity listening at address "peer". // Parameters: // 0) [in] peer: The address of the listening UDT entity. // Returned value: // None. void CUDT::connect(const sockaddr* serv_addr) // Functionality: // Process the response handshake packet. // Parameters: // 0) [in] pkt: handshake packet. // Returned value: // Return 0 if connected, positive value if connection is in progress, otherwise error code. int CUDT::connect(const CPacket& response) throw ()// Functionality: // Connect to a UDT entity listening at address "peer", which has sent "hs" request. // Parameters: // 0) [in] peer: The address of the listening UDT entity. // 1) [in/out] hs: The handshake information sent by the peer side (in), negotiated value (out). // Returned value: // None. void CUDT::connect(const sockaddr* peer, CHandShake* hs)

    在 connect 中調用的 connect 參數為 sockaddr,即第一種形式,這也是從外部接口 UDT::connect 調用后進入的函數:

    void CUDT::connect(const sockaddr* serv_addr) {CGuard cg(m_ConnectionLock);if (!m_bOpened) // UDT socket 處于 OPENED 狀態throw CUDTException(5, 0, 0);if (m_bListening) //不能同時 listen 與 connectthrow CUDTException(5, 2, 0);if (m_bConnecting || m_bConnected) //以前沒有進行 connect 過throw CUDTException(5, 2, 0);m_bConnecting = true; //修改狀態,防止被多次 connect// record peer/server addressdelete m_pPeerAddr;m_pPeerAddr = (AF_INET == m_iIPversion) ? (sockaddr*)new sockaddr_in : (sockaddr*)new sockaddr_in6;memcpy(m_pPeerAddr, serv_addr, (AF_INET == m_iIPversion) ? sizeof(sockaddr_in) : sizeof(sockaddr_in6));// register this socket in the rendezvous queue// RendezevousQueue is used to temporarily store incoming handshake, non-rendezvous connections also require this functionuint64_t ttl = 3000000; //這也是關閉時需要額外等3s的原因if (m_bRendezvous)ttl *= 10;ttl += CTimer::getTime();//將 UDT socket 插入一個鏈表 m_lRendezvousID,臨時存儲。不管是否匯合模式,都會保存。此處可能導致誤解。m_pRcvQueue->registerConnector(m_SocketID, this, m_iIPversion, serv_addr, ttl); // This is my current configurationsm_ConnReq.m_iVersion = m_iVersion;m_ConnReq.m_iType = m_iSockType;m_ConnReq.m_iMSS = m_iMSS;m_ConnReq.m_iFlightFlagSize = (m_iRcvBufSize < m_iFlightFlagSize)? m_iRcvBufSize : m_iFlightFlagSize;m_ConnReq.m_iReqType = (!m_bRendezvous) ? 1 : 0;m_ConnReq.m_iID = m_SocketID;CIPAddress::ntop(serv_addr, m_ConnReq.m_piPeerIP, m_iIPversion);// Random Initial Sequence Numbersrand((unsigned int)CTimer::getTime());m_iISN = m_ConnReq.m_iISN = (int32_t)(CSeqNo::m_iMaxSeqNo * (double(rand()) / RAND_MAX));m_iLastDecSeq = m_iISN - 1;m_iSndLastAck = m_iISN;m_iSndLastDataAck = m_iISN;m_iSndCurrSeqNo = m_iISN - 1;m_iSndLastAck2 = m_iISN;m_ullSndLastAck2Time = CTimer::getTime();// Inform the server my configurations.CPacket request;char* reqdata = new char [m_iPayloadSize];request.pack(0, NULL, reqdata, m_iPayloadSize); //建包// ID = 0, connection requestrequest.m_iID = 0;int hs_size = m_iPayloadSize;m_ConnReq.serialize(reqdata, hs_size); //寫入請求request.setLength(hs_size);m_pSndQueue->sendto(serv_addr, request); // 發送請求m_llLastReqTime = CTimer::getTime(); //更新定時器// asynchronous connect, return immediatelyif (!m_bSynRecving){delete [] reqdata;return;}// Wait for the negotiated configurations from the peer side.CPacket response;char* resdata = new char [m_iPayloadSize];response.pack(0, NULL, resdata, m_iPayloadSize);CUDTException e(0, 0);while (!m_bClosing) // 等待 connect 返回,最多等待3s, 如果沒有響應,會重復發送請求{// avoid sending too many requests, at most 1 request per 250msif (CTimer::getTime() - m_llLastReqTime > 250000){m_ConnReq.serialize(reqdata, hs_size);request.setLength(hs_size);if (m_bRendezvous)request.m_iID = m_ConnRes.m_iID;m_pSndQueue->sendto(serv_addr, request);m_llLastReqTime = CTimer::getTime();}response.setLength(m_iPayloadSize);if (m_pRcvQueue->recvfrom(m_SocketID, response) > 0){if (connect(response) <= 0)break;// new request/response should be sent out immediately on receving a responsem_llLastReqTime = 0;}if (CTimer::getTime() > ttl){// timeoute = CUDTException(1, 1, 0);break;}}delete [] reqdata;delete [] resdata;if (e.getErrorCode() == 0){if (m_bClosing) // if the socket is closed before connection...e = CUDTException(1);else if (1002 == m_ConnRes.m_iReqType) // connection request rejectede = CUDTException(1, 2, 0);else if ((!m_bRendezvous) && (m_iISN != m_ConnRes.m_iISN)) // secuity checke = CUDTException(1, 4, 0);}if (e.getErrorCode() != 0)throw e; }

    發出 connect 連接請求以后,如果屬于同步方式,將等待返回,超時時間設置為3s,并會間隔250ms 不斷發送請求,直至收到響應。接收到響應以后, connect(response), 即為 第二種 connect 方法。

    建立連接的過程可以參考 TCP 的半連接的思想。這第二次的connect 實際上就是 第二個半連接的建立過程,也是最后一個協商過程。對于非匯合模式,從 m_lRendezvousID 中移除,重新配置所有的連接參數, 為UDT socket 建立對應的各種數據結構,包括 發送接收buffer,丟失鏈表,窗口等,這些是數據傳輸過程中需要使用的內部結構,服務于 UDT 的核心傳輸算法,包括擁塞避免,重傳等。所以也會初始化擁塞控制相關參數,最后,設置當前狀態 為已連接狀態。通過 connect_complete 與 update_events 通知管理模塊與epool 狀態更新。

    int CUDT::connect(const CPacket& response) throw () {// this is the 2nd half of a connection request. If the connection is setup successfully this returns 0.// returning -1 means there is an error.// returning 1 or 2 means the connection is in process and needs more handshakeif (!m_bConnecting)return -1;if (m_bRendezvous && ((0 == response.getFlag()) || (1 == response.getType())) && (0 != m_ConnRes.m_iType)){//a data packet or a keep-alive packet comes, which means the peer side is already connected// in this situation, the previously recorded response will be usedgoto POST_CONNECT;}if ((1 != response.getFlag()) || (0 != response.getType()))return -1;m_ConnRes.deserialize(response.m_pcData, response.getLength());if (m_bRendezvous){// regular connect should NOT communicate with rendezvous connect// rendezvous connect require 3-way handshakeif (1 == m_ConnRes.m_iReqType)return -1;if ((0 == m_ConnReq.m_iReqType) || (0 == m_ConnRes.m_iReqType)){m_ConnReq.m_iReqType = -1;// the request time must be updated so that the next handshake can be sent out immediately.m_llLastReqTime = 0;return 1;}}else{// set cookieif (1 == m_ConnRes.m_iReqType){m_ConnReq.m_iReqType = -1;m_ConnReq.m_iCookie = m_ConnRes.m_iCookie;m_llLastReqTime = 0;return 1;}}POST_CONNECT:// Remove from rendezvous queuem_pRcvQueue->removeConnector(m_SocketID);// Re-configure according to the negotiated values.m_iMSS = m_ConnRes.m_iMSS;m_iFlowWindowSize = m_ConnRes.m_iFlightFlagSize;m_iPktSize = m_iMSS - 28;m_iPayloadSize = m_iPktSize - CPacket::m_iPktHdrSize;m_iPeerISN = m_ConnRes.m_iISN;m_iRcvLastAck = m_ConnRes.m_iISN;m_iRcvLastAckAck = m_ConnRes.m_iISN;m_iRcvCurrSeqNo = m_ConnRes.m_iISN - 1;m_PeerID = m_ConnRes.m_iID;memcpy(m_piSelfIP, m_ConnRes.m_piPeerIP, 16);// Prepare all data structurestry{m_pSndBuffer = new CSndBuffer(32, m_iPayloadSize);m_pRcvBuffer = new CRcvBuffer(&(m_pRcvQueue->m_UnitQueue), m_iRcvBufSize);// after introducing lite ACK, the sndlosslist may not be cleared in time, so it requires twice space.m_pSndLossList = new CSndLossList(m_iFlowWindowSize * 2);m_pRcvLossList = new CRcvLossList(m_iFlightFlagSize);m_pACKWindow = new CACKWindow(1024);m_pRcvTimeWindow = new CPktTimeWindow(16, 64);m_pSndTimeWindow = new CPktTimeWindow();}catch (...){throw CUDTException(3, 2, 0);}CInfoBlock ib;ib.m_iIPversion = m_iIPversion;CInfoBlock::convert(m_pPeerAddr, m_iIPversion, ib.m_piIP);if (m_pCache->lookup(&ib) >= 0){m_iRTT = ib.m_iRTT;m_iBandwidth = ib.m_iBandwidth;}m_pCC = m_pCCFactory->create();m_pCC->m_UDT = m_SocketID;m_pCC->setMSS(m_iMSS);m_pCC->setMaxCWndSize(m_iFlowWindowSize);m_pCC->setSndCurrSeqNo(m_iSndCurrSeqNo);m_pCC->setRcvRate(m_iDeliveryRate);m_pCC->setRTT(m_iRTT);m_pCC->setBandwidth(m_iBandwidth);m_pCC->init();m_ullInterval = (uint64_t)(m_pCC->m_dPktSndPeriod * m_ullCPUFrequency);m_dCongestionWindow = m_pCC->m_dCWndSize;// And, I am connected too.m_bConnecting = false;m_bConnected = true;// register this socket for receiving data packetsm_pRNode->m_bOnList = true;m_pRcvQueue->setNewEntry(this);// acknowledge the management module.s_UDTUnited.connect_complete(m_SocketID); // 更新m_pSndQueue->m_pChannel本地節點信息,設置狀態為 CONNECTED// acknowledde any waiting epolls to writes_UDTUnited.m_EPoll.update_events(m_SocketID, m_sPollID, UDT_EPOLL_OUT, true);return 0; }

    UDT accept

    • UDT::accept -> CUDT::accept -> CUDTUnited::accept

    UDT accept 與 TCP socket 中 的accept一樣,在socket bind 以后就可以使用,等待其他連接到來。所以當前的 UDT Listener 的狀態為 LISTENING。 僅在非匯合模式下使用。

    主要的過程是一個while 循環,等待accept事件。當到來以后,刪除 m_pQueuedSockets中的節點,插入 m_pAcceptSockets。update_events 發給 epool事件更新。存儲對端地址。

    UDTSOCKET CUDTUnited::accept(const UDTSOCKET listen, sockaddr* addr, int* addrlen) {CUDTSocket* ls = locate(listen);// the "listen" socket must be in LISTENING statusif (LISTENING != ls->m_Status)throw CUDTException(5, 6, 0);// no "accept" in rendezvous connection setupif (ls->m_pUDT->m_bRendezvous)throw CUDTException(5, 7, 0);UDTSOCKET u = CUDT::INVALID_SOCK;bool accepted = false;// !!only one conection can be set up each time!!#ifndef WIN32while (!accepted) //循環等待連接到來 accepted = true 時退出{pthread_mutex_lock(&(ls->m_AcceptLock));if ((LISTENING != ls->m_Status) || ls->m_pUDT->m_bBroken){// This socket has been closed.accepted = true;}else if (ls->m_pQueuedSockets->size() > 0){ //更新 m_pAcceptSockets 和 m_pQueuedSocketsu = *(ls->m_pQueuedSockets->begin()); ls->m_pAcceptSockets->insert(ls->m_pAcceptSockets->end(), u);ls->m_pQueuedSockets->erase(ls->m_pQueuedSockets->begin());accepted = true;}else if (!ls->m_pUDT->m_bSynRecving){accepted = true;}if (!accepted && (LISTENING == ls->m_Status))pthread_cond_wait(&(ls->m_AcceptCond), &(ls->m_AcceptLock));if (ls->m_pQueuedSockets->empty())m_EPoll.update_events(listen, ls->m_pUDT->m_sPollID, UDT_EPOLL_IN, false);pthread_mutex_unlock(&(ls->m_AcceptLock));}#elsewhile (!accepted){WaitForSingleObject(ls->m_AcceptLock, INFINITE);if (ls->m_pQueuedSockets->size() > 0){u = *(ls->m_pQueuedSockets->begin());ls->m_pAcceptSockets->insert(ls->m_pAcceptSockets->end(), u);ls->m_pQueuedSockets->erase(ls->m_pQueuedSockets->begin());accepted = true;}else if (!ls->m_pUDT->m_bSynRecving)accepted = true;ReleaseMutex(ls->m_AcceptLock);if (!accepted & (LISTENING == ls->m_Status))WaitForSingleObject(ls->m_AcceptCond, INFINITE);if ((LISTENING != ls->m_Status) || ls->m_pUDT->m_bBroken){// Send signal to other threads that are waiting to accept.SetEvent(ls->m_AcceptCond);accepted = true;}if (ls->m_pQueuedSockets->empty())m_EPoll.update_events(listen, ls->m_pUDT->m_sPollID, UDT_EPOLL_IN, false);}#endifif (u == CUDT::INVALID_SOCK){// non-blocking receiving, no connection availableif (!ls->m_pUDT->m_bSynRecving)throw CUDTException(6, 2, 0);// listening socket is closedthrow CUDTException(5, 6, 0);}// 存儲對端的地址if ((addr != NULL) && (addrlen != NULL)){if (AF_INET == locate(u)->m_iIPversion)*addrlen = sizeof(sockaddr_in);else*addrlen = sizeof(sockaddr_in6);// copy address information of peer nodememcpy(addr, locate(u)->m_pPeerAddr, *addrlen);}return u; }

    總結

    UDT支持兩種連接模式:C/S 模式和匯合模式。UDT client 發送一個握手消息(type為 0 的控制報文)給 server 或者 peer。消息攜帶信息格式見文章 UDT最新協議分析 。

    C/S 模式–四次握手

    如果一個UDT socket 作為server,會建立一個UDT實體, 并作為 Listener 監聽綁定的端口,當有新的連接請求到來時,就會新創建一個 UDT socket,并初始化相關信息,并將新的 UDT socket 相關的信息寫入到 Listener。這就是一個連接的建立過程,和TCP的連接過程比較相似。

  • 當 UDT client 要對一個UDT server建立連接的時候,會在3s內每間隔 250ms 連續發送握手報文,直到收到server反饋回來的握手的報文或者連接超時。
  • 當 UDT server 第一次接收到來自 UDT client 的握手連接請求的時候,它會根據 client 的 address 和一個 secret key 產生一個 cookie 值,然后發送給 client。
  • 當 UDT client 收到回應以后,必須把收到的 cookie 再返回發送給 server。
  • 當 UDT server 接收到一個握手報文和正確的 cookie 時,協商包大小與最大窗口,并把協商結果發送給 client。
    • UDT server 將握手報文中的 packet size 和 maximum window size 信息提取出來,并同 server 端自己的 packet size 和 maximum window size信息相比較,將較小的 packet size 和 maximum window size 信息賦值給自己。
    • UDT server 把包大小與最大窗口等結果發送給client端,并攜帶上 server 的版本號和初始序列號。為防止丟包,如果后續還接收到同一對端其他握手消息時,仍需要繼續發送響應。
    • UDT server 準備接收發送數據。
    • UDT client 收到 server 發送的握手包,開始發送接收數據,如果還有其他握手消息,不再回應。

    Rendezvous模式–三次握手

    匯合模式下,兩端均為客戶端,需要兩端同時調用udt::connect, 主要用于NAT穿透的情況。

  • 兩個UDT client 同時向對方發起連接請求,發送握手包。連接類型初始值為0。
  • UDT client 收到對端發送的連接請求后,檢查連接類型
    • 如果連接類型為0,那么響應報文中會被設置成-1。
    • 如果連接類型為-1,那么響應報文中會被設置成-2。
    • 如果連接類型為-2,那么將不會有任何反饋信息。

    總結

    以上是生活随笔為你收集整理的UDT 最新源码分析(三) -- UDT Socket 相关函数的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    国产黄色特级片 | 精品在线视频播放 | 日韩乱色精品一区二区 | 欧美不卡视频在线 | 免费在线观看av网站 | 91 在线视频 | 亚洲国产97在线精品一区 | 国产精品久久久久av福利动漫 | 四虎影视成人 | 中国精品一区二区 | 91一区一区三区 | 久久国产精品视频免费看 | 国产伦精品一区二区三区高清 | 天天色天天射天天操 | 丁香六月激情 | 免费观看版 | 99久久www | 超级碰99 | 日韩欧美在线中文字幕 | 高清av免费观看 | 三级黄色理论片 | 国产中文字幕一区二区 | 日韩免费成人 | 中文视频在线 | 久草热久草视频 | 久久99视频免费观看 | 日韩视频免费观看高清完整版在线 | 91在线播放综合 | 久久草在线精品 | 在线 视频 一区二区 | 免费在线观看av网址 | 欧美亚洲国产精品久久高清浪潮 | 日韩特黄av| av成人免费网站 | 韩国精品一区二区三区六区色诱 | 日韩在线 | 中国美女一级看片 | 久久国产福利 | 夜色资源站wwwcom | 亚洲综合丁香 | 伊人狠狠色丁香婷婷综合 | 国产精品欧美一区二区三区不卡 | 操碰av | 国产97av| 久久国产精品99久久久久久老狼 | 久久综合偷偷噜噜噜色 | 五月激情姐姐 | 深夜免费福利视频 | 久久久久久久久久久高潮一区二区 | 成人在线中文字幕 | 91精品国产自产91精品 | 国产在线观看免 | av网站在线免费观看 | 国产成人久久久久 | 国产精品99久久免费观看 | 国产精品你懂的在线观看 | 综合中文字幕 | 91精品高清| 亚洲成人免费在线 | 日本天天操 | 婷婷成人在线 | www.亚洲精品在线 | 久草在线视频网站 | 亚洲春色成人 | 青青河边草免费直播 | 日韩试看| 国产成人区 | 日本黄色一级电影 | 国产精品久久久久亚洲影视 | av大全在线免费观看 | 亚洲第一中文字幕 | 超碰97国产精品人人cao | 日韩成人免费在线观看 | 国产麻豆精品传媒av国产下载 | 亚洲精品午夜久久久久久久久久久 | 中文在线免费一区三区 | 久久精品国产免费 | 天天鲁天天干天天射 | 女人高潮一级片 | 天天草视频 | 国产v在线播放 | 精品黄色片 | 亚洲动漫在线观看 | 伊人伊成久久人综合网小说 | 精品在线观看视频 | 日本中文字幕视频 | 中文字幕一区二区三区四区在线视频 | 精品久久久999 | 美女视频网站久久 | 国产精品小视频网站 | 97在线观看视频 | 国产日韩欧美视频 | www.色综合.com | 夜夜躁天天躁很躁波 | 日本中文一区二区 | 国产成人免费av电影 | 免费看成人片 | 久久久久精 | 国产精品久久99综合免费观看尤物 | 精品黄色在线 | 成人免费电影 | 黄色一级片视频 | 亚洲免费视频在线观看 | 免费a v观看 | 亚洲精品国产精品国自产在线 | 久久国产精品99国产精 | 欧美色图狠狠干 | 好看的国产精品视频 | 91自拍91| 91精品国产麻豆国产自产影视 | 中文字幕在线影院 | 久久视频这里只有精品 | 天天操夜操 | 久久国产高清视频 | 亚洲欧美日韩一区二区三区在线观看 | 黄色一级免费电影 | 国产麻豆精品95视频 | 国产成人精品一二三区 | 精品国产1区2区3区 国产欧美精品在线观看 | 黄污视频大全 | 久久久精品 | 天天狠狠操 | 四虎在线观看 | 久久久久久久久久久久久久av | 久久激情视频 | 一区二区电影在线观看 | 成人免费观看在线视频 | 又爽又黄又刺激的视频 | 久久精品伊人 | 午夜av免费看 | 国产夫妻性生活自拍 | 国产又粗又猛又色又黄网站 | 久久电影日韩 | 中文字幕在线视频国产 | 欧美日本国产在线观看 | 国产又粗又猛又黄又爽 | 51久久夜色精品国产麻豆 | 欧美a√在线| 久久99国产精品免费 | 亚洲网站在线 | 九九久久国产精品 | 在线视频麻豆 | 久久久久免费电影 | japanese黑人亚洲人4k | 99精品视频精品精品视频 | 91麻豆国产福利在线观看 | 日色在线视频 | 人人射网站| www.色五月| 天天射天天射天天 | 婷婷在线综合 | 99riav1国产精品视频 | 亚洲高清在线精品 | 久久激情五月激情 | 日韩在线视频免费观看 | 亚洲国产欧美在线人成大黄瓜 | 久久人人精品 | 国产视频精品久久 | 亚洲综合视频在线观看 | 精壮的侍卫呻吟h | av 一区二区三区 | 久久夜色精品国产欧美乱 | 国产丝袜制服在线 | 午夜免费在线观看 | 国产97在线观看 | 一级片色播影院 | 色射爱 | 日韩欧美精品在线视频 | 97在线观看视频 | 婷婷六月久久 | 色操插 | 久久精品欧美一区二区三区麻豆 | 在线观看亚洲成人 | 亚洲aⅴ一区二区三区 | 免费一区在线 | 亚洲精品播放 | 成人午夜影院 | 久久久久国产成人免费精品免费 | 婷婷五月在线视频 | 久久黄色免费 | 免费网址你懂的 | 中文字幕在线电影 | 天天操操 | 久久草精品 | 久久久久久久影院 | 欧美一二三在线 | 中文字幕一区三区 | 狠狠色丁香九九婷婷综合五月 | 99精品热视频只有精品10 | 精品夜夜嗨av一区二区三区 | 日韩毛片在线播放 | 男女拍拍免费视频 | 黄色大片免费网站 | 在线观看国产www | 99热在线精品观看 | 91精品国产麻豆国产自产影视 | 欧美日韩亚洲在线观看 | 国产资源| 97超级碰碰碰碰久久久久 | 日韩久久久久久久久久 | 日本bbbb摸bbbb | www.夜夜草| 亚洲国产成人在线观看 | 国产在线观看高清视频 | 精品国产区 | 午夜精品福利一区二区 | 久久天堂影院 | 亚洲,国产成人av | 日韩欧美一区二区三区免费观看 | 中文在线a天堂 | 日韩天堂在线观看 | 日本中文字幕在线播放 | 在线观看一级视频 | 久草视频在线新免费 | 日日干夜夜干 | 亚洲欧洲国产日韩精品 | 日韩av中文字幕在线免费观看 | 在线看国产一区 | 久久爱综合 | 日本久久久久久 | 亚洲精品av中文字幕在线在线 | 久久9999久久 | 国产精品美女免费视频 | 91成人在线视频 | 精品一区二区日韩 | 日韩中文字幕免费视频 | 人人干人人添 | 在线国产片 | 中文字幕 二区 | 精品久久久久久亚洲综合网站 | 欧美一级日韩免费不卡 | 精品久久久久一区二区国产 | 亚洲欧洲视频 | 青青啪 | 天天爽夜夜爽精品视频婷婷 | 91在线观 | 精品在线免费视频 | 久久久久久久久久久精 | 免费a级观看 | 国产精品密入口果冻 | 最近中文字幕在线中文高清版 | 久久精品视频国产 | 中文字幕之中文字幕 | 手机av观看 | 91在线免费公开视频 | 在线观看91久久久久久 | 亚洲精品国偷自产在线91正片 | 成人黄色大片网站 | 国产视频不卡一区 | 91高清免费在线观看 | 欧美二区视频 | 一级性视频 | 免费在线观看视频a | 波多野结衣资源 | 欧美日韩亚洲第一 | 99精品国产成人一区二区 | 一区 二区电影免费在线观看 | 91亚洲精品久久久 | 91视频免费观看 | 中文字幕资源网在线观看 | 亚洲人成免费 | 色婷婷啪啪免费在线电影观看 | 91精品日韩 | 欧美夫妻生活视频 | 最近久乱中文字幕 | 亚洲国产av精品毛片鲁大师 | 欧美 激情 国产 91 在线 | 国产精品美女久久久久久久网站 | 91麻豆看国产在线紧急地址 | 国产高清视频在线 | 欧美日韩中文字幕在线视频 | 丁香婷婷在线 | 高潮毛片无遮挡高清免费 | 国产999在线观看 | 中文字幕在线专区 | 中文字幕 国产 一区 | 色综合天天天天做夜夜夜夜做 | 日本69hd | 精一区二区 | 在线观看午夜av | 久久婷亚洲五月一区天天躁 | 99久久精品午夜一区二区小说 | 亚洲天堂网在线视频观看 | 韩国视频一区二区三区 | 91一区一区三区 | 丁香五香天综合情 | 久久精品国产久精国产 | 超碰人人在线 | 国产成人三级在线 | 九九亚洲视频 | 日本性生活一级片 | 亚洲高清视频在线观看 | 日韩在线播放av | 亚洲黄色免费观看 | 香蕉视频国产在线观看 | 在线国产99 | 国产淫片免费看 | 婷婷精品在线 | 欧美激情另类 | 在线观看资源 | 天天艹天天 | 美女黄频视频大全 | 国产尤物视频在线 | 国产在线看一区 | 五月婷婷狠狠 | 日本超碰在线 | 日韩中文字 | 日韩黄色软件 | 男女视频91| 国产精品久久久久久久免费观看 | 69久久夜色精品国产69 | 欧美在线aa| 国内综合精品午夜久久资源 | 精品久久久影院 | 美女网站视频色 | 国产婷婷在线观看 | 欧美亚洲久久 | 久久综合视频网 | 国产中文字幕在线播放 | 综合色婷婷 | 午夜精品久久久久 | 免费视频在线观看网站 | 99热99re6国产在线播放 | 国产96在线视频 | 久要激情网 | 丁香在线视频 | 国产日韩精品一区二区三区在线 | 国产在线999| 人人干人人添 | 久久都是精品 | 在线成人小视频 | 三级黄色免费片 | 日本精品视频一区二区 | 国产打女人屁股调教97 | 久久亚洲精品国产亚洲老地址 | 99视频网站 | 蜜桃视频日本 | 在线观看av小说 | 91免费在线视频 | 免费看成人片 | 天天干,天天操 | 欧美激情va永久在线播放 | 亚洲日日日 | 亚洲丁香久久久 | 激情欧美丁香 | 国产手机视频精品 | 黄色av三级在线 | 中文字幕二区在线观看 | 狠狠色丁香婷婷综合欧美 | 欧美日韩二三区 | 精品视频在线观看 | 亚洲一级二级三级 | 亚洲成人精品久久久 | 久章操 | 中文字幕av免费观看 | 久久久久久久网 | 成人97视频一区二区 | 欧美精品在线观看免费 | 久久99国产综合精品 | av九九九| 狠狠久久伊人 | 在线之家官网 | 亚洲精品在线一区二区 | 精品一区二区三区四区在线 | 国产一区在线观看视频 | 天天爽夜夜操 | 国产中文字幕视频在线观看 | 米奇狠狠狠888 | 国产欧美精品一区二区三区 | 久久香蕉影视 | 99久久精品一区二区成人 | 成人性生爱a∨ | 色诱亚洲精品久久久久久 | 国产成人亚洲在线观看 | 成人av网页 | 天天操天天摸天天干 | 日韩欧美视频一区二区 | 婷婷在线网| 国产日韩欧美网站 | 天天av资源 | 黄色在线观看污 | 97视频免费在线 | 天天插一插 | 91在线观 | 99热只有精品在线观看 | 免费精品国产 | 97av视频 | 精品国产精品国产偷麻豆 | 中文字幕乱在线伦视频中文字幕乱码在线 | 日本精品久久久一区二区三区 | 玖玖在线看| 成人网444ppp | 精品久久久免费视频 | 国产在线观看av | 亚洲婷婷丁香 | 亚洲精品ww | 国产一区在线看 | 成人在线视频论坛 | 久草免费在线视频观看 | 91精品国产欧美一区二区 | 在线最新av | 在线视频婷婷 | 国产成人综合在线观看 | 国产精品一区二区62 | 西西大胆啪啪 | 午夜12点 | 成人精品一区二区三区电影免费 | 亚洲精品www久久久 www国产精品com | 成人午夜电影免费在线观看 | 黄色视屏免费在线观看 | 久久综合狠狠综合久久综合88 | 国产亚洲高清视频 | 丁香5月婷婷 | 国产成人久久久久 | 久久久久久影视 | 欧美a级在线 | 96久久久 | 久久久久久国产精品美女 | 97人人艹| 91在线视频免费播放 | 国产精品国产三级在线专区 | 99久久www免费 | 国产伦精品一区二区三区四区视频 | 伊人天天狠天天添日日拍 | 欧美大片www | 在线国产视频观看 | 午夜精品中文字幕 | 涩涩资源网 | 天天操夜夜做 | 99免费在线播放99久久免费 | 在线视频麻豆 | 亚洲va韩国va欧美va精四季 | 久草视频免费播放 | 日韩com| 日韩在线一区二区免费 | 精品伊人久久久 | 天天操天天干天天 | 在线看小早川怜子av | 免费在线观看黄网站 | 久久亚洲免费视频 | 成人免费大片黄在线播放 | 99一级片 | 日韩在线视频免费看 | 国产精品综合av一区二区国产馆 | 色wwwww| 久久免费公开视频 | 日本精品久久久久久 | 久久精品这里精品 | 韩国av免费在线观看 | 免费视频久久久久 | 啪啪免费试看 | 免费福利在线观看 | 国产精品日韩欧美 | 亚洲少妇自拍 | 中文字幕成人在线观看 | 色com| 日本美女xx | 婷婷激情影院 | 日韩av免费一区 | www.久久色.com | 国产精品mv在线观看 | 91看片在线观看 | 日日爱夜夜爱 | 九九欧美| 天天射天天操天天干 | 91中文在线视频 | 看av在线| 91天堂在线观看 | 亚洲二区精品 | 蜜桃传媒一区二区 | av高清免费在线 | 国产成人区 | 色综合久久久久久久 | 亚洲色综合 | 欧美最猛性xxxxx亚洲精品 | 麻豆国产精品永久免费视频 | 国产小视频在线看 | 久久五月天综合 | 中文字幕中文字幕在线中文字幕三区 | 亚洲免费在线 | 久热av| 久久一区精品 | 久久久久久久久黄色 | 国产精品第一页在线 | 国产精品久久久久av | 波多野结衣在线视频一区 | 久久天堂精品视频 | 一区二区三区在线免费观看视频 | 欧美激情h| www.狠狠 | 日日操狠狠干 | 亚洲黄色影院 | 91久久奴性调教 | 午夜视频一区二区三区 | 亚洲精品资源在线 | 丝袜美腿在线视频 | 精品久久久久久久久久岛国gif | 91免费在线看片 | 欧美另类tv | 成人午夜剧场在线观看 | 毛片的网址 | 色婷婷国产| 日韩视频在线一区 | 午夜丁香网 | 成人av在线亚洲 | 久久视频在线观看中文字幕 | 中文字幕国产一区 | 欧美va天堂在线电影 | 探花视频在线观看+在线播放 | 日韩欧美视频 | www.久久色.com | 国产综合在线观看视频 | 在线看不卡av | 五月开心婷婷网 | 日p在线观看 | 久久国内精品99久久6app | www.天天色 | 午夜三级理论 | 狠狠夜夜| 免费人成在线观看 | 亚洲精品久久久蜜桃直播 | 三级黄色在线观看 | 成人午夜免费福利 | 国产视频一区二区在线 | 久久久久久久久久久久久影院 | 波多野结衣电影久久 | 久久精品99北条麻妃 | 亚洲一级性 | 国产精品美女999 | 国产精品av免费在线观看 | www.狠狠干 | 久色网| 在线 国产一区 | 国产精品igao视频网入口 | 国产精品成人久久久久 | 激情五月五月婷婷 | 亚洲欧洲精品一区二区精品久久久 | 伊人影院99 | 99视频在线免费观看 | 精品久久久久久综合 | 色综合久久中文字幕综合网 | 国产精品99久久免费观看 | 狠狠色丁香婷婷 | 欧美精品亚洲精品 | 久久免费电影网 | 中文字幕电影高清在线观看 | 午夜久久网 | 天天干天天色2020 | 四虎影视精品永久在线观看 | 韩国三级在线一区 | 成x99人av在线www | 色香天天 | 人人爽人人| 91刺激视频 | 久久影院精品 | 久久综合九色99 | 日本中文字幕在线 | 91av原创| 国产一区二区精品 | 日本性生活一级片 | 福利视频精品 | 久久精品视频在线观看 | 欧美与欧洲交xxxx免费观看 | 久久夜夜夜| 国产美女免费视频 | 狠狠色丁香婷婷综合最新地址 | 91九色蝌蚪在线 | 国产99精品 | 激情开心色| 国产不卡网站 | 欧美激情综合五月色丁香 | 五月婷婷丁香网 | 免费h精品视频在线播放 | 蜜臀久久99精品久久久酒店新书 | 国产午夜在线 | 成片免费 | 国产精品门事件 | 色婷婷免费视频 | 亚洲一级片免费观看 | 国产毛片久久久 | 久久亚洲电影 | 午夜av日韩 | 欧美久久成人 | 精品一区二区三区电影 | 日韩精品免费一区二区在线观看 | 狠狠干成人综合网 | 久久精品一二三区白丝高潮 | 国产一级电影网 | 在线观看日韩免费视频 | 亚洲国产高清视频 | 99视屏 | 久久免费成人精品视频 | 亚洲v欧美v国产v在线观看 | 欧美激情第十页 | 日韩av午夜在线观看 | 久久最新网址 | 国产精品黑丝在线观看 | 丁香花在线视频观看免费 | 亚洲专区免费观看 | 免费激情在线电影 | 久久99欧美 | 中文字幕一区二区三区在线播放 | 夜夜澡人模人人添人人看 | 少妇bbbb搡bbbb桶 | 天天精品视频 | 玖玖综合网 | 国产视频二区三区 | 成人毛片在线观看 | 韩国av免费在线 | 日韩视频在线观看视频 | 九九色在线 | 欧美日韩在线视频免费 | 91精品中文字幕 | 日韩簧片在线观看 | 波多野结衣在线观看一区 | 在线看一区 | 久久精品男人的天堂 | 久久97视频 | 国产美腿白丝袜足在线av | 久久久精品国产一区二区 | av在线小说| 久久精品这里热有精品 | 超碰在线公开免费 | 波多野结衣视频一区 | 日日夜夜天天干 | 91探花国产综合在线精品 | 国产亚洲精品久久久久久电影 | 国产成人福利在线观看 | 操操日| 日本中文字幕在线电影 | 韩国一区二区在线观看 | 黄色大片网 | 久久久久久久久久亚洲精品 | 色的网站在线观看 | 精品国产乱码 | 日韩三级av | 亚洲成色777777在线观看影院 | 中文字幕亚洲欧美日韩2019 | 精品久久久久久久久中文字幕 | 奇米影视8888 | 久久免费看 | 九九热精| 日韩有码网站 | 99精品乱码国产在线观看 | 一本一本久久a久久 | 97在线精品视频 | 欧美日韩精品网站 | 97超碰免费在线观看 | 婷婷视频导航 | 日韩a在线看 | av丝袜在线 | 亚洲精品玖玖玖av在线看 | 永久av免费在线观看 | 国内三级在线观看 | 国产精品美女久久久久久 | 国产免费亚洲高清 | 麻豆视频观看 | 日韩在线视 | 国产91精品看黄网站在线观看动漫 | 欧美日韩观看 | 中文久草| 香蕉在线视频观看 | 国产精品久久久电影 | 久久综合狠狠狠色97 | 444av| 国产精品成人自拍 | 国产高清视频网 | 国产在线精品二区 | 开心激情久久 | 一本色道久久综合亚洲二区三区 | 在线观看aa | 日韩中出在线 | 久久久综合香蕉尹人综合网 | 午夜精品久久久99热福利 | 五月婷婷六月丁香 | 最近中文字幕大全中文字幕免费 | 婷婷色综 | 91传媒在线观看 | 日本精品在线 | 欧美一级xxxx | 国产精品 9999 | 九九热国产视频 | 久久综合婷婷国产二区高清 | 夜夜夜影院 | 欧美日韩一二三四区 | av在线成人 | 99久久国产免费免费 | 精品国产区 | 中文字幕 婷婷 | 国产区精品在线 | www.看片网站 | 精品视频 | 夜夜夜夜夜夜操 | 久久a免费视频 | 美女性爽视频国产免费app | 欧美欧美 | 色香蕉在线 | 91在线网址 | 人人搞人人爽 | 伊人天天综合 | 粉嫩一区二区三区粉嫩91 | 国产成人精品在线观看 | 精品国产观看 | 国产午夜精品免费一区二区三区视频 | 久久精品这里精品 | 黄色片亚洲 | 国产剧情一区在线 | 亚洲最大av | 丁香电影小说免费视频观看 | 亚洲精品视频免费看 | 成人毛片一区 | 日本精品视频一区二区 | 国产成人一二片 | 久久精品高清视频 | www.天天射 | 国产精品一区电影 | 一级黄色在线免费观看 | 亚洲精品国产成人av在线 | 国产视频每日更新 | 午夜精品久久久久久久99无限制 | 日韩va亚洲va欧美va久久 | 国产视频91在线 | 人人爽人人爱 | 亚洲美女精品 | 日韩在线国产 | 成人免费视频网站在线观看 | 在线亚洲观看 | 久久成人国产精品免费软件 | 欧美日韩亚洲第一 | 黄色电影网站在线观看 | 国产久草在线观看 | 99热精品国产一区二区在线观看 | 久 久久影院 | 免费av在线播放 | 国产黄色精品在线 | 国产福利精品在线观看 | 久久综合欧美精品亚洲一区 | av中文天堂在线 | 91av在线电影 | 麻豆免费观看视频 | 中文在线字幕观看电影 | 国产麻豆成人传媒免费观看 | 中午字幕在线观看 | 久久人人爽人人爽人人片 | 亚洲天天看 | 国产视频观看 | 99999精品 | 夜夜操天天操 | 精品视频免费在线 | 91资源在线观看 | 日本精品视频一区二区 | 中文字幕一区二区在线播放 | 亚洲国产成人久久 | 99热播精品| 在线91视频 | 国产精品视频免费 | 婷婷激情在线 | 亚洲免费在线视频 | 三级午夜片 | 日韩在线观看精品 | 狠狠插狠狠干 | 日韩免费三级 | 日韩黄色中文字幕 | 国产亚洲人 | 欧美激情精品久久久久久 | 在线国产视频一区 | 国产成人99久久亚洲综合精品 | 国产精品一区二区无线 | 狠狠的干狠狠的操 | 在线观看激情av | 国产精品一区二区视频 | 国产夫妻av在线 | 成人动图 | 色天天 | 精品在线播放视频 | 91精品少妇偷拍99 | 99精品视频免费在线观看 | 中文av网站 | 亚洲国产午夜 | 青青河边草免费视频 | 国内视频一区二区 | 国产一区二区久久久 | www.国产精品 | 久久五月情影视 | 中文字幕精品三区 | 日日躁夜夜躁xxxxaaaa | 亚洲激情视频在线观看 | 久久久久欠精品国产毛片国产毛生 | 久久久久久久久久久免费 | 久久一区国产 | 成人黄在线观看 | 91av视频| 久久国产电影 | 四虎在线观看视频 | 午夜av免费看 | 草久热| 久久久久电影网站 | 毛片精品免费在线观看 | 91麻豆精品国产91久久久久 | 婷婷在线网站 | 亚洲最大色 | 一区二区中文字幕在线 | 日韩网站在线观看 | 免费看一级黄色大全 | 99精品视频精品精品视频 | 99热.com| 国产最新91 | 99中文视频在线 | 超碰97在线资源 | 国产精品毛片久久 | 2024国产精品视频 | 国产毛片久久久 | 天天干天天操天天拍 | 天天搞天天干 | 日韩啪视频 | 成片免费观看视频 | 国产九九精品视频 | 欧美国产日韩在线观看 | 黄色精品网站 | 在线v| 国产视频欧美视频 | 美女网站久久 | 日韩欧美在线视频一区二区三区 | 亚洲视屏 | 91中文字幕在线视频 | 国产精品视频在线看 | 91在线超碰| 永久免费精品视频 | 天天射天天色天天干 | 亚洲精品高清在线 | 免费看的av片 | 国产亚洲精品久久久久动 | 欧美男同视频网站 | 国产高清视频在线播放 | 欧美成人tv| 久久不卡国产精品一区二区 | 欧美日韩不卡一区二区三区 | 狠狠色丁香久久婷婷综合丁香 | 九九免费精品 | 最新色站| 在线 日韩 av| 中文字幕在线看视频国产中文版 | 欧美性爽爽 | 国产原创91 | 99精品免费网 | 日韩高清在线一区二区三区 | 91精品1区2区| 中文字幕久久亚洲 | 亚洲视频免费在线观看 | 日韩色一区二区三区 | 九九99靖品 | 国产精品伦一区二区三区视频 | av三级av| 国内一级片在线观看 | 午夜av免费看 | 在线观看久 | 伊人狠狠色丁香婷婷综合 | 91在线观看视频网站 | 久久久久久高潮国产精品视 | 夜夜躁天天躁很躁波 | 亚洲成人一区 | 天天摸天天舔 | 99精品在线免费在线观看 | 九九一级片 | 亚洲自拍偷拍色图 | 免费黄色特级片 | 在线观看麻豆av | 久草在线免费在线观看 | 国产美腿白丝袜足在线av | 中文字幕av日韩 | 国产精品毛片一区二区在线看 | 色九九影院 | 一级性视频 | 美女网站色在线观看 | 在线看黄色的网站 | 久久久久久久久久久影视 | 欧美日韩一区二区在线观看 | 国产91丝袜在线播放动漫 | 中文字幕婷婷 | 最新日本中文字幕 | 日韩av电影免费在线观看 | 欧美射射射 | 91精品国产99久久久久 | 久久久精品国产免费观看一区二区 | 国产一区视频免费在线观看 | 欧美黄色免费 | 国产成人精品一区二三区 | 国内久久精品 | 视频二区 | 免费一级特黄录像 | 日韩中文字幕在线观看 | 草樱av| 亚洲日韩中文字幕在线播放 | 91免费高清 | 国内综合精品午夜久久资源 | 激情视频网页 | 国产精品久久久久久久久久了 | 日韩丝袜 | 亚洲国产网站 | 欧美日韩免费看 | 99这里都是精品 | 欧美人人| 精品久久久久免费极品大片 | 婷婷丁香在线 | 又黄又爽的视频在线观看网站 | 91精品国产麻豆国产自产影视 | 国产精品久久久久久久久软件 | 黄毛片在线观看 | 99日精品 | 亚洲精品中文字幕在线观看 | 久久国产精品一国产精品 | 狠狠干在线 | 久久久www成人免费毛片 | av网站在线观看播放 | 免费精品视频在线观看 | 亚洲欧美日本一区二区三区 | 国产特级毛片aaaaaaa高清 | 国内丰满少妇猛烈精品播放 | 亚洲国产欧美在线人成大黄瓜 | 操操操人人| 中文字幕在线观看1 | 日韩高清二区 | 国产日韩欧美在线观看 | 日韩精品观看 | 成人av电影免费 | 欧美激情综合五月色丁香 | 国产精品国产自产拍高清av | 国产精品久久久一区二区三区网站 | 免费大片av | 国产成人av网站 | 免费在线观看av不卡 | 亚洲人精品午夜 | 国产精品区在线观看 | 天天操天天干天天插 | 久久精品中文视频 | 国产成人av电影 | 日日干av| 欧美日韩一区二区在线观看 | 久草在线视频免费资源观看 | 久久天天拍 | 涩五月婷婷 | 一区二区三区在线观看免费 | 亚洲国产视频在线 | 福利电影久久 | 国产伦理久久精品久久久久_ | 麻豆91精品视频 | 黄色网在线播放 | 欧美 日韩 成人 | 夜又临在线观看 | 国产成人在线一区 | 黄色影院在线播放 | 日本久久久久久久久久 | 综合久久精品 | 国产不卡一 | 国产精品自拍av | 操操操天天操 | 亚洲一区二区三区毛片 | 九九视频一区 | 日本久久影视 | 欧美一区二区三区免费观看 | 99 精品 在线 | 久久国产精品免费一区二区三区 | 亚洲午夜精品久久久久久久久 | 这里只有精品视频在线观看 | 久久久久久久久久免费视频 | 日韩欧美视频免费观看 | 久久艹国产视频 | 天天操天天干天天操天天干 | 日韩高清dvd | 精品资源在线 | 日本中文字幕在线 | 欧美日韩精品在线免费观看 | 国产精品av在线 | 色老板在线视频 | 91| 日韩精品亚洲专区在线观看 | 久草在线视频中文 | 色99久久 | 亚洲精品乱码久久久久久蜜桃欧美 | 精品亚洲男同gayvideo网站 | 色就色,综合激情 | 欧美成人aa | 91九色视频观看 | 精品国产1区 | 欧美在线aa | av字幕在线| 日韩精品在线看 | av在线观| 国产成人福利在线观看 | 成人免费网视频 | 中文字幕在线不卡国产视频 | 91高清视频在线 | 五月激情片 | www.天天色 | 少妇18xxxx性xxxx片 | 高清国产午夜精品久久久久久 | www.久久91 | 手机av网站 |