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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > linux >内容正文

linux

再谈Linux epoll惊群问题的原因和解决方案

發布時間:2024/3/12 linux 39 豆豆
生活随笔 收集整理的這篇文章主要介紹了 再谈Linux epoll惊群问题的原因和解决方案 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

差別是什么?差別只是西裝!

緣起

近期排查了一個問題,epoll驚群的問題,起初我并不認為這是驚群導致,因為從現象上看,只是體現了CPU不均衡。一共fork了20個Server進程,在請求負載中等的時候,有三四個Server進程呈現出比較高的CPU利用率,其余的Server進程的CPU利用率都是非常低。

中斷,軟中斷都是均衡的,網卡RSS和CPU之間進行了bind之后依然如故,既然系統層面查不出個所以然,只能從服務的角度來查了。

自上而下的排查首先就想到了strace,沒想到一下子就暴露了原形:

accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable)

如果僅僅strace accept,即加上“-e trace=accept”參數的話,偶爾會有accept成功的現象:

accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, {sa_family=AF_INET, sin_port=htons(39306), sin_addr=inet_addr("172.16.1.202")}, [16]) = 19 accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable) accept(4, 0x9ecd930, [16]) = -1 EAGAIN (Resource temporarily unavailable)

大量的CPU空轉,進一步加大請求負載,CPU空轉明顯降低,這說明在預期的空轉期間,新來的請求降低了空轉率…現象明顯偏向于這就是驚群導致的之判斷!

本文將詳細說一下關于epoll的細節。現在開始!


關于epoll的文章,我很早前寫過一篇總結性的,可以參考這里:
Linux內核中網絡數據包的接收-第二部分 select/poll/epoll:https://blog.csdn.net/dog250/article/details/50528373
不過這篇文章主要是原理性的介紹,對于一開始并不充分理解epoll機制的人來講,可讀性并不強,所以我準備寫一篇稍微接地氣的,比如帶有一些“源碼分析”的文章,雖然我并不是很喜歡源碼分析,但有時對于快速理解為什么這樣還是必要的。

題目中為什么是“再談”,因為這個話題別人已經聊過很多了,我順勢繼續下去而已。

簡單介紹驚群和事件模型

關于什么是驚群,這里不再做概念上的解釋,能搜到這篇文章的想必已經有所了解,如果仍有概念上的疑惑,自行百度或者谷歌。


驚群問題一般出現在那些web服務器上,曾經Linux系統有個經典的accept驚群問題困擾了大家非常久的時間,這個問題現在已經在內核曾經得以解決,具體來講就是當有新的連接進入到accept隊列的時候,內核喚醒且僅喚醒一個進程來處理,這是通過以下的代碼來實現的:

list_for_each_entry_safe(curr, next, &q->task_list, task_list) {unsigned flags = curr->flags;if (curr->func(curr, mode, wake_flags, key) &&(flags & WQ_FLAG_EXCLUSIVE) && !--nr_exclusive)break;}

是的,添加了一個WQ_FLAG_EXCLUSIVE標記,告訴內核進行排他性的喚醒,即喚醒一個進程后即退出喚醒的過程,問題得以解決。

然而,沒有哪個web服務器會傻到多個進程直接阻塞在accept上準備接收請求,在更高層次上,多路復用的需求讓select,poll,epoll等事件模型更為受到歡迎,所謂的事件模型即阻塞在事件上而不是阻塞在事務上。內核僅僅通知發生了某件事,具體發生了什么事,則有處理進程或者線程自己來poll。如此一來,這個事件模型(無論其實現是select,poll,還是epoll)便可以一次搜集多個事件,從而滿足多路復用的需求。

好了,基本原理就介紹到這里,下面我將來詳細談一下Linux epoll中的驚群問題,我們知道epoll在實際中要比直接accept實用性強很多,據我所知,除非編程學習或者驗證性小demo,幾乎沒有直接accept的代碼,所有的線上代碼幾乎都使用了事件模型。然而由于select,poll沒有可擴展性,存在O(n)O(n)問題,因此在帶寬越來越高,服務器性能越來越強的趨勢下,越來越多的代碼將收斂到使用epoll的情形,所以有必要對其進行深入的討論。


Linux epoll驚群問題

知乎上有一個問題:
Linux 3.x 中epoll的驚群問題?:https://www.zhihu.com/question/24169490/answers/created
建議先看一下,但不要看回答,因為知乎上上的很多回答往往會讓事情變得更加混亂,除非你自己對這個問題已經有了自己的答案或者觀點,否則還是不要去指望在諸多的答案中選一個自己滿意的來用,還是要自己先思考。

下面我來就這個問題給一個答案,這也是我自己思考的答案:

  • 在ep_poll的睡眠中加入WQ_FLAG_EXCLUSIVE標記,確實實實在在解決了epoll的驚群問題
  • epoll_wait返回后確實也還有多個進程被喚醒只有一個進程能正確處理其他進程無事可做的情況發生,但這不是因為驚群,而是你的使用方法不對。
  • What?使用方法不對?

    是的,使用方法不對。若想了解Why,則必須對epoll的實現細節以及其對外提供的API的語義有充分的理解,接下來我們就循著這個思路來擼個所以然。請繼續閱讀。

    Linux epoll的實現機制

    說起實現原理,很多人喜歡擼源碼分析,我并不喜歡,我認為源碼是自己看看就行了,搞這個行業的能看懂代碼是一個最最基本的能力,我比較在意的是對某種機制內在邏輯的深入理解,而這個通過代碼是體現不出來的,我一般會做下面幾件事:

    • 運行起來并測得預期的數據
    • 看懂代碼并畫出原理圖
    • 自己重新實現一版(時間精力允許的情況下)
    • 寫個demo驗證一些具體邏輯細節

    不多說。

    下面是我總結的一張關于Linux epoll的原理圖:

    要說代碼實現上,其實也比較簡單,大致有以下的幾個邏輯:

  • 創建epoll句柄,初始化相關數據結構
  • 為epoll句柄添加文件句柄,注冊睡眠entry的回調
  • 事件發生,喚醒相關文件句柄睡眠隊列的entry,調用其回調
  • 喚醒epoll睡眠隊列的task,搜集并上報數據
  • 來,一個一個說

    1.創建epoll句柄,初始化相關數據結構

    這里主要就是創建一個epoll文件描述符,注意,后面操作epoll的時候,就是用這個epoll的文件描述符來操作的,所以這就是epoll的句柄,精簡過后的epoll結構如下:

    struct eventpoll {// 阻塞在epoll_wait的task的睡眠隊列wait_queue_head_t wq;// 存在就緒文件句柄的list,該list上的文件句柄事件將會全部上報給應用struct list_head rdllist;// 存放加入到此epoll句柄的文件句柄的紅黑樹容器struct rb_root rbr;// 該epoll結構對應的文件句柄,應用通過它來操作該epoll結構struct file *file; };

    2.為epoll句柄添加文件句柄,注冊睡眠entry的回調

    這個步驟中其實有兩個子步驟:
    1). 添加文件句柄
    將一個文件句柄,比如socket添加到epoll的rbr紅黑樹容器中,注意,這里的文件句柄最終也是一個包裝結構,和epoll的結構體類似:

    struct epitem {// 該字段鏈接入epoll句柄的紅黑樹容器struct rb_node rbn;// 當該文件句柄有事件發生時,該字段鏈接入“就緒鏈表”,準備上報給用戶態struct list_head rdllink;// 該字段封裝實際的文件,我已經將其展開struct epoll_filefd {struct file *file;int fd;} ffd;// 反向指向其所屬的epoll句柄struct eventpoll *ep; };

    以上結構實例就是epi,將被添加到epoll的rbr容器中的邏輯如下:

    struct eventpoll *ep = 待加入文件句柄所屬的epoll句柄; struct file *tfile = 待加入的文件句柄file結構體; int fd = 待加入的文件描述符ID;struct epitem *epi = kmem_cache_alloc(epi_cache, GFP_KERNEL); INIT_LIST_HEAD(&epi->rdllink); INIT_LIST_HEAD(&epi->fllink); INIT_LIST_HEAD(&epi->pwqlist); epi->ep = ep; ep_set_ffd(&epi->ffd, tfile, fd); ... ep_rbtree_insert(ep, epi);

    2). 注冊睡眠entry回調并poll文件句柄
    在第一個子步驟的代碼邏輯中,我有一段“…”省略掉了,這部分比較關鍵,所以我單獨抽取了出來作為第二個子步驟。
    我們知道,Linux內核的sleep/wakeup機制非常重要,幾乎貫穿了所有的內核子系統,值得注意的是,這里的sleep/wakeup依然采用了OO的思想,并沒有限制睡眠的entry一定要是一個task,而是將睡眠的entry做了一層抽象,即:

    struct __wait_queue {unsigned int flags;// 至于這個private到底是什么,內核并不限制,顯然,它可以是task,也可以是別的。void *private;wait_queue_func_t func;struct list_head task_list; };

    以上的這個entry,最終要睡眠在下面的數據結構實例化的一個鏈表上:

    struct __wait_queue_head {spinlock_t lock;struct list_head task_list; };

    顯然,在這里,一個文件句柄均有自己睡眠隊列用于等待自己發生事件的entry在沒有發生事件時來歇息,對于TCP socket而言,該睡眠隊列就是其sk_wq,通過以下方式取到:

    static inline wait_queue_head_t *sk_sleep(struct sock *sk) {return &rcu_dereference_raw(sk->sk_wq)->wait; }

    我們需要一個entry將來在發生事件的時候從上述wait_queue_head_t中被喚醒,執行特定的操作,即將自己放入到epoll句柄的“就緒鏈表”中。下面的函數可以完成該邏輯的框架:

    // 此處的whead就是上面例子中的sk_sleep返回的wait_queue_head_t實例。 static void ep_ptable_queue_proc(struct file *file, wait_queue_head_t *whead,poll_table *pt) {struct epitem *epi = ep_item_from_epqueue(pt);struct eppoll_entry *pwq;if (pwq = kmem_cache_alloc(pwq_cache, GFP_KERNEL)) {// 發生事件即調用ep_poll_callback回調函數,該回調函數會將自己這個epitem加入到epoll的“就緒鏈表”中去。init_waitqueue_func_entry(&pwq->wait, ep_poll_callback);// 是否排他喚醒取決于用戶的配置,有些IO是希望喚醒所有entry來處理,有些則不必。注意,這里是針對文件句柄IO而言的,并不是針對epoll句柄的。if (epi->event.events & EPOLLEXCLUSIVE)add_wait_queue_exclusive(whead, &pwq->wait);elseadd_wait_queue(whead, &pwq->wait);} }

    至于說什么時候調用上面的函數,Linux的poll機制仍然是采用了分層抽象的思想,即上述函數會作為另一個回調在相關文件句柄的poll函數中被調用。即:


    static inline unsigned int ep_item_poll(struct epitem *epi, poll_table *pt) {pt->_key = epi->event.events;return epi->ffd.file->f_op->poll(epi->ffd.file, pt) & epi->event.events; }

    對于TCP socket而言,其file_operations的poll回調即:

    unsigned int tcp_poll(struct file *file, struct socket *sock, poll_table *wait) {unsigned int mask;struct sock *sk = sock->sk;const struct tcp_sock *tp = tcp_sk(sk);// 此函數會調用poll_wait->wait._qproc// 而wait._qproc就是ep_ptable_queue_procsock_poll_wait(file, sk_sleep(sk), wait);... }

    現在,我們可以把子步驟1中的邏輯補全了:

    struct eventpoll *ep = 待加入文件句柄所屬的epoll句柄; struct file *tfile = 待加入的文件句柄file結構體; int fd = 待加入的文件描述符ID;struct epitem *epi = kmem_cache_alloc(epi_cache, GFP_KERNEL); INIT_LIST_HEAD(&epi->rdllink); INIT_LIST_HEAD(&epi->fllink); INIT_LIST_HEAD(&epi->pwqlist); epi->ep = ep; ep_set_ffd(&epi->ffd, tfile, fd); // 這里會將wait._qproc初始化成ep_ptable_queue_proc init_poll_funcptr(&epq.pt, ep_ptable_queue_proc); // 這里會調用wait._qproc即ep_ptable_queue_proc,安排entry的回調函數ep_poll_callback,并將entry“睡眠”在socket的sk_wq這個睡眠隊列上。 revents = ep_item_poll(epi, &epq.pt); ep_rbtree_insert(ep, epi); // 如果剛才的ep_item_poll取出了事件,隨即將該item掛入“就緒隊列”中,并且wakeup阻塞在epoll_wait系統調用中的task! if ((revents & event->events) && !ep_is_linked(&epi->rdllink)) {list_add_tail(&epi->rdllink, &ep->rdllist);if (waitqueue_active(&ep->wq))wake_up_locked(&ep->wq); }

    3.事件發生,喚醒相關文件句柄睡眠隊列的entry,調用其回調

    上面已經很詳細地描述了epoll的基礎設施了,現在我們假設一個TCP Listen socket上來了一個連接請求,已經完成了三次握手,內核希望通知epoll_wait返回,然后去取accept。
    內核在wakeup這個socket的sk_wq時,最終會調用到ep_poll_callback回調,這個函數我們說了好幾次了,現在看看它的真面目:

    static int ep_poll_callback(wait_queue_t *wait, unsigned mode, int sync, void *key) {unsigned long flags;struct epitem *epi = ep_item_from_wait(wait);struct eventpoll *ep = epi->ep;// 這個lock比較關鍵,操作“就緒鏈表”相關的,均需要這個lock,以防丟失事件。spin_lock_irqsave(&ep->lock, flags);// 如果發生的事件我們并不關注,則不處理直接返回即可。if (key && !((unsigned long) key & epi->event.events))goto out_unlock;// 實際將發生事件的epitem加入到“就緒鏈表”中。if (!ep_is_linked(&epi->rdllink)) {list_add_tail(&epi->rdllink, &ep->rdllist);}// 既然“就緒鏈表”中有了新成員,則喚醒阻塞在epoll_wait系統調用的task去處理。注意,如果本來epi已經在“就緒隊列”了,這里依然會喚醒并處理的。if (waitqueue_active(&ep->wq)) {wake_up_locked(&ep->wq);}out_unlock:spin_unlock_irqrestore(&ep->lock, flags);... }

    沒什么好多說的。現在“就緒鏈表”已經有epi了,接下來就要喚醒epoll_wait進程去處理了。

    4.喚醒epoll睡眠隊列的task,搜集并上報數據
    這個邏輯主要集中在ep_poll函數,精簡版如下:

    static int ep_poll(struct eventpoll *ep, struct epoll_event __user *events,int maxevents, long timeout) {unsigned long flags;wait_queue_t wait;// 當前沒有事件才睡眠if (!ep_events_available(ep)) {init_waitqueue_entry(&wait, current);__add_wait_queue_exclusive(&ep->wq, &wait);for (;;) {set_current_state(TASK_INTERRUPTIBLE);...// 例行的schedule timeout}__remove_wait_queue(&ep->wq, &wait);set_current_state(TASK_RUNNING);}// 往用戶態上報事件,即那些epoll_wait返回后能獲取的事件。ep_send_events(ep, events, maxevents); }

    其中關鍵在ep_send_events,這個函數實現了非常重要的邏輯,包括LT和ET的邏輯,我不打算深入去解析這個函數,只是大致說下流程:

    ep_scan_ready_list() {// 遍歷“就緒鏈表”ready_list_for_each() {// 將epi從“就緒鏈表”刪除list_del_init(&epi->rdllink);// 實際獲取具體的事件。// 注意,睡眠entry的回調函數只是通知有“事件”,具體需要每一個文件句柄的特定poll回調來獲取。revents = ep_item_poll(epi, &pt);if (revents) {if (__put_user(revents, &uevent->events) ||__put_user(epi->event.data, &uevent->data)) {// 如果沒有完成,則將epi重新加回“就緒鏈表”等待下次。list_add(&epi->rdllink, head);return eventcnt ? eventcnt : -EFAULT;}// 如果是LT模式,則無論如何都會將epi重新加回到“就緒鏈表”,等待下次重新再poll以確認是否仍然有未處理的事件。這也符合“水平觸發”的邏輯,即“只要你不處理,我就會一直通知你”。if (!(epi->event.events & EPOLLET)) {list_add_tail(&epi->rdllink, &ep->rdllist);}}}// 如果“就緒鏈表”上仍有未處理的epi,且有進程阻塞在epoll句柄的睡眠隊列,則喚醒它!(這將是LT驚群的根源)if (!list_empty(&ep->rdllist)) {if (waitqueue_active(&ep->wq))wake_up_locked(&ep->wq);} }

    這里的代碼邏輯的分析過程就到此為止了。以對這個代碼邏輯的充分理解為基礎,接下來我們就可以看具體的問題細節了。

    下面一小節先從LT(水平觸發模式)以及ET(即邊沿觸發模式)開始。

    epoll的LT和ET以及相關細節問題

    簡單點解釋:

    • LT水平觸發
      如果事件來了,不管來了幾個,只要仍然有未處理的事件,epoll都會通知你。
    • ET邊沿觸發
      如果事件來了,不管來了幾個,你若不處理或者沒有處理完,除非下一個事件到來,否則epoll將不會再通知你。

    理解了上面說的兩個模式,便可以很明確地展示可能會遇到的問題以及解決方案了,這將非常簡單。

    LT水平觸發模式的問題以及解決

    下面是epoll使用中非常常見的代碼框架,我將問題注釋于其中:

    // 否則會阻塞在IO系統調用,導致沒有機會再epoll set_socket_nonblocking(sd); epfd = epoll_create(64); event.data.fd = sd; epoll_ctl(epfd, EPOLL_CTL_ADD, sd, &event); while (1) {epoll_wait(epfd, events, 64, xx);... // 危險區域!如果有共享同一個epfd的進程/線程調用epoll_wait,它們也將會被喚醒!// 這個accept將會有多個進程/線程調用,如果并發請求數很少,那么將僅有幾個進程會成功:// 1. 假設accept隊列中有n個請求,則僅有n個進程能成功,其它將全部返回EAGAIN (Resource temporarily unavailable)// 2. 如果n很大(即增加請求負載),雖然返回EAGAIN的比率會降低,但這些進程也并不一定取到了epoll_wait返回當下的那個預期的請求。csd = accept(sd, &in_addr, &in_len); ... }

    這一切為什么會發生?

    我們結合理論和代碼一起來分析。

    再看一遍LT的描述“如果事件來了,不管來了幾個,只要仍然有未處理的事件,epoll都會通知你。”,顯然,epoll_wait剛剛取到事件的時候的時候,不可能馬上就調用accept去處理,事實上,邏輯在epoll_wait函數調用的ep_poll中還沒返回的,這個時候,顯然符合“仍然有未處理的事件”這個條件,顯然這個時候為了實現這個語義,需要做的就是通知別的同樣阻塞在同一個epoll句柄睡眠隊列上的進程!在實現上,這個語義由兩點來保證:

  • 保證1:在LT模式下,“就緒鏈表”上取出的epi上報完事件后會重新加回“就緒鏈表”;
  • 保證2:如果“就緒鏈表”不為空,且此時有進程阻塞在同一個epoll句柄的睡眠隊列上,則喚醒它。
  • ep_scan_ready_list() {// 遍歷“就緒鏈表”ready_list_for_each() {list_del_init(&epi->rdllink);revents = ep_item_poll(epi, &pt);// 保證1if (revents) {__put_user(revents, &uevent->events);if (!(epi->event.events & EPOLLET)) {list_add_tail(&epi->rdllink, &ep->rdllist);}}}// 保證2if (!list_empty(&ep->rdllist)) {if (waitqueue_active(&ep->wq))wake_up_locked(&ep->wq);} }

    我們來看一個情景分析。

    假設LT模式下有10個進程共享同一個epoll句柄,此時來了一個請求client進入到accept隊列,我們發現上述的1和2是一個循環喚醒的過程:

    1).假設進程a的epoll_wait首先被ep_poll_callback喚醒,那么滿足1和2,則喚醒了進程B;
    2).進程B在處理ep_scan_ready_list的時候,發現依然滿足1和2,于是喚醒了進程C….
    3).上面1)和2)的過程一直到之前某個進程將client取出,此時下一個被喚醒的進程在ep_scan_ready_list中的ep_item_poll調用中將得不到任何事件,此時便不會再將該epi加回“就緒鏈表”了,LT水平觸發結束,結束了這場悲傷的夢!

    問題非常明確了,但是怎么解決呢?也非常簡單,讓不同進程的epoll_waitI調用互斥即可。

    但是且慢!

    上面的情景分析所展示的是一個“驚群效應”嗎?其實并不是!對于Listen socket,當然要避免這種情景,但是對于很多其它的I/O文件句柄,說不定還指望著大家一起來read數據呢…所以說,要說互斥也僅僅要針對Listen socket的epoll_wait調用而言。

    換句話說,這里epoll LT模式下有進程被不必要喚醒,這一點并不是內核無意而為之的,內核肯定是知道這件事的,這個并不像之前accept驚群那樣算是內核的一個缺陷。epoll LT模式只是提供了一種模式,誤用這種模式將會造成類似驚群那樣的效應。但是不管怎么說,為了討論上的方便,后面我們姑且將這種效應稱作epoll LT驚群吧。

    除了epoll_wait互斥之外,還有一種解決問題的方案,即使用ET邊沿觸發模式,但是會遇到新的問題,我們接下來來描述。

    ET邊沿觸發模式的問題以及解決

    ET模式不滿足上述的“保證1”,所以不會將已經上報事件的epi重新鏈接回“就緒鏈表”,也就是說,只要一個“就緒隊列”上的epi上的事件被上報了,它就會被刪除出“就緒隊列”。

    由于epi entry的callback即ep_poll_callback所做的事情僅僅是將該epi自身加入到epoll句柄的“就緒鏈表”,同時喚醒在epoll句柄睡眠隊列上的task,所以這里并不對事件的細節進行計數,比如說,如果ep_poll_callback在將一個epi加入“就緒鏈表”之前發現它已經在“就緒鏈表”了,那么就不會再次添加,因此可以說,一個epi可能pending了多個事件,注意到這點非常重要!

    一個epi上pending多個事件,這個在LT模式下沒有任何問題,因為獲取事件的epi總是會被重新添加回“就緒鏈表”,那么如果還有事件,在下次check的時候總會取到。然而對于ET模式,僅僅將epi從“就緒鏈表”刪除并將事件本身上報后就返回了,因此如果該epi里還有事件,則只能等待再次發生事件,進而調用ep_poll_callback時將該epi加入“就緒隊列”。這意味著什么?

    這意味著,應用程序,即epoll_wait的調用進程必須自己在獲取事件后將其處理干凈后方可再次調用epoll_wait,否則epoll_wait不會返回,而是必須等到下次產生事件的時候方可返回。即,依然以accept為例,必須這樣做:

    // 否則會阻塞在IO系統調用,導致沒有機會再epoll set_socket_nonblocking(sd); epfd = epoll_create(64); event.data.fd = sd; // 添加ET標記 event.events |= EPOLLET; epoll_ctl(epfd, EPOLL_CTL_ADD, sd, &event); while (1) {epoll_wait(epfd, events, 64, xx);while ((csd = accept(sd, &in_addr, &in_len)) > 0) {do_something(...);} ... }

    好了,解釋完了。


    以上就是epoll的LT,ET相關的兩個問題和解決方案。接下來的一節,我將用一個小小的簡單Demo來重現上面描述的理論和代碼。

    測試demo

    是時候給出一個實際能run的代碼了:

    #include <stdlib.h> #include <stdio.h> #include <unistd.h> #include <sys/types.h> #include <sys/socket.h> #include <sys/epoll.h> #include <netdb.h> #include <fcntl.h> #include <sys/wait.h> #include <time.h> #include <signal.h>#define COUNT 1int mode = 0; int slp = 0;int pid[COUNT] = {0}; int count = 0;void server(int epfd) {struct epoll_event *events;int num, i;struct timespec ts;events = calloc(64, sizeof(struct epoll_event));while (1) {int sd, csd;struct sockaddr in_addr;num = epoll_wait(epfd, events, 64, -1);if (num <= 0) {continue;}/*ts.tv_sec = 0;ts.tv_nsec = 1;if(nanosleep(&ts, NULL) != 0) {perror("nanosleep");exit(1);}*/// 用于測試ET模式下丟事件的情況if (slp) {sleep(slp);}sd = events[0].data.fd;socklen_t in_len = sizeof(in_addr);csd = accept(sd, &in_addr, &in_len);if (csd == -1) {// 打印這個說明中了epoll LT驚群的招了。printf("shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:%d\n", getpid()); continue;}// 本進程一共成功處理了多少個請求。count ++;printf("get client:%d\n", getpid()); close(csd);} }static void siguser_handler(int sig) {// 在主進程被Ctrl-C退出的時候,每一個子進程均要打印自己處理了多少個請求。printf("pid:%d count:%d\n", getpid(), count);exit(0); }static void sigint_handler(int sig) {int i = 0;// 給每一個子進程發信號,要求其打印自己處理了多少個請求。for (i = 0; i < COUNT; i++) {kill(pid[i], SIGUSR1);} }int main (int argc, char *argv[]) {int ret = 0;int listener;int c = 0;struct sockaddr_in saddr;int port;int status;int flags;int epfd;struct epoll_event event;if (argc < 4) {exit(1);}// 0為LT模式,1為ET模式mode = atoi(argv[1]);port = atoi(argv[2]);// 是否在處理accept之前耽擱一會兒,這個參數更容易重現問題slp = atoi(argv[3]);signal(SIGINT, sigint_handler);listener = socket(PF_INET, SOCK_STREAM, 0);saddr.sin_family = AF_INET;saddr.sin_port = htons(port);saddr.sin_addr.s_addr = INADDR_ANY;bind(listener, (struct sockaddr*)&saddr, sizeof(saddr));listen(listener, SOMAXCONN);flags = fcntl (listener, F_GETFL, 0);flags |= O_NONBLOCK;fcntl (listener, F_SETFL, flags);epfd = epoll_create(64);if (epfd == -1) {perror("epoll_create");abort();}event.data.fd = listener;event.events = EPOLLIN;if (mode == 1) {event.events |= EPOLLET;} else if (mode == 2) {event.events |= EPOLLONESHOT;} ret = epoll_ctl(epfd, EPOLL_CTL_ADD, listener, &event);if (ret == -1) {perror("epoll_ctl");abort();}for(c = 0; c < COUNT; c++) {int child;child = fork();if(child == 0) {// 安裝打印count值的信號處理函數signal(SIGUSR1, siguser_handler);server(epfd);}pid[c] = child;printf("server:%d pid:%d\n", c+1, child);}wait(&status);sleep(1000000);close (listener); }

    編譯之,為a.out。

    測試客戶端選用了簡單webbench,首先我們看一下LT水平觸發模式下的問題:

    [zhaoya@shit ~/test]$ sudo ./a.out 0 112 0 server:1 pid:9688 server:2 pid:9689 server:3 pid:9690 server:4 pid:9691 server:5 pid:9692 server:6 pid:9693 server:7 pid:9694 server:8 pid:9695 server:9 pid:9696 server:10 pid:9697

    另起一個終端運行webbench,并發10,測試5秒:

    [zhaoya@shit ~/test]$ webbench -c 10 -t 5 http://127.0.0.1:112/ Webbench - Simple Web Benchmark 1.5 Copyright (c) Radim Kolar 1997-2004, GPL Open Source Software.Benchmarking: GET http://127.0.0.1:112/ 10 clients, running 5 sec.

    而a.out的終端有以下輸出:

    ... get client:9690 get client:9688 get client:9691 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9693 get client:9692 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9689 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9697 get client:9691 get client:9696 get client:9690 get client:9690 get client:9695 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9697 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9689 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9692 get client:9696 get client:9688 get client:9695 get client:9693 get client:9689 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9691 get client:9695 get client:9691 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:9692 get client:9690 get client:9694 get client:9693 ...

    所有的“shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:”的行均是被epoll LT驚群不必要喚醒的進程打印的。

    接下來用ET模式運行:

    [zhaoya@shit ~/test]$ sudo ./a.out 1 112 0

    對應的輸出如下:

    ... get client:14462 get client:14462 get client:14464 get client:14464 get client:14462 get client:14462 get client:14467 get client:14469 get client:14468 get client:14468 get client:14464 get client:14467 get client:14467 get client:14469 get client:14469 get client:14469 get client:14464 get client:14464 get client:14466 get client:14466 get client:14469 get client:14469 ...

    沒有任何一行是shit,即沒有被不必要喚醒的驚群現象發生。

    以上兩個case確認了epoll LT模式的驚群效應是可以通過改用ET模式來解決的,接下來我們確認ET模式非循環處理會丟失事件

    用ET模式運行a.out,這時將slp參數設置為1,即在epoll_wait返回和實際accept之間耽擱1秒,這樣可以讓一個epi在被加入到“就緒鏈表”中之后,在其被實際accept處理之前,積累更多的未決事件,即未處理的請求,而我們實驗的目的則是,epoll ET會丟失這些事件。

    webbench的參數依然如故,a.out的輸出如下:

    [zhaoya@shit ~/test]$ sudo ./a.out 1 114 1 server:1 pid:31161 server:2 pid:31162 server:3 pid:31163 server:4 pid:31164 server:5 pid:31165 server:6 pid:31166 server:7 pid:31167 server:8 pid:31168 server:9 pid:31169 server:10 pid:31170 get client:31170 get client:31170 get client:31167 get client:31169 get client:31166 get client:31165 get client:31170 get client:31167 get client:31169 get client:31165 get client:31168 get client:31170 get client:31167 get client:31165 get client:31169 get client:31170 get client:31167 get client:31169 get client:31170 get client:31167 get client:31169^Cpid:31170 count:6 pid:31169 count:5 pid:31163 count:0 pid:31168 count:1 pid:31167 count:5 pid:31165 count:3 pid:31166 count:1 pid:31161 count:0 pid:31162 count:0 pid:31164 count:0 User defined signal 1

    同樣的webbench參數,僅僅處理了十幾個請求,可見大多數都丟掉了。如果我們用LT模式,同樣在sleep 1秒導致事件擠壓的情況下,是不是會多處理一些呢?我們的預期應該是肯定的,因為LT模式在事件被處理完之前,會一直促使epoll_wait返回繼續處理,那么讓我們試一下:

    [zhaoya@shit ~/test]$ sudo ./a.out 0 115 1 server:1 pid:363 server:2 pid:364 server:3 pid:365 server:4 pid:366 server:5 pid:367 server:6 pid:368 server:7 pid:369 server:8 pid:370 server:9 pid:371 server:10 pid:372 get client:372 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:371 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:365 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:366 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:363 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:367 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:369 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:364 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:368 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:370 get client:370 get client:364 get client:367 get client:368 get client:369 get client:365 get client:371 get client:372 get client:363 get client:366 get client:370 get client:367 get client:364 get client:369 get client:371 get client:368 get client:366 get client:363 get client:365 get client:372 get client:370 get client:367 get client:364 get client:371 get client:369 get client:366 get client:368 get client:363 get client:365 get client:372 get client:370 get client:367 get client:371 get client:364 get client:369 get client:366 get client:365 get client:368 get client:363 get client:372 get client:370 get client:364 get client:371 get client:367 get client:366 get client:369 get client:365 get client:363 get client:368 get client:372 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:371 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:370 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:364 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:367 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:366 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:369 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:365 shit xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx:363 ^Cpid:363 count:5 pid:368 count:5 pid:372 count:6 pid:369 count:5 pid:366 count:5 pid:370 count:5 pid:367 count:5 pid:371 count:5 pid:365 count:5 pid:364 count:5 User defined signal 1

    是的,多處理了很多,但是出現了LT驚群,這也是意料之中的事。

    最后,讓我們把這個Demo代碼小改一下,改成循環處理,依然采用ET模式,sleep 1秒,看看情況會怎樣。修改后的代碼如下:

    void server(int epfd) {struct epoll_event *events;int num, i;struct timespec ts;events = calloc(64, sizeof(struct epoll_event));while (1) {int sd, csd;struct sockaddr in_addr;num = epoll_wait(epfd, events, 64, -1);if (num <= 0) {continue;}if (slp)sleep(slp);sd = events[0].data.fd;socklen_t in_len = sizeof(in_addr);// 這里循環處理,一直到空。while ((csd = accept(sd, &in_addr, &in_len)) > 0) {count ++;printf("get client:%d\n", getpid());close(csd);}} }

    改完代碼后,再做同樣參數的測試,結果大大不同:

    [zhaoya@shit ~/test]$ sudo ./a.out 0 116 1 ... get client:3640 get client:3645 get client:3640 get client:3641 get client:3641 get client:3641 ^Cpid:3642 count:14 pid:3647 count:33531 pid:3646 count:21824 pid:3648 count:22 pid:3644 count:32219 pid:3645 count:94449 pid:3641 count:8 pid:3640 count:85385 pid:3643 count:13 pid:3639 count:10 User defined signal 1

    可以看到,大多數的請求都得到了處理,同樣的邏輯,epoll_wait返回后的循環讀和一次讀結果顯然不同。

    問題和解決方案都很明確了,可以結單了嗎?我想是的,但是在終結這個話題之前,我還想說一些結論性的東西以供備忘和參考。

    結論

    曾經,為了實現并發服務器,出現了很多的所謂范式,比如下面的兩個很常見:

    • 范式1:設置多個IP地址,多個IP地址同時偵聽相同的端口,前端用4層負載均衡或者反向代理來對這些IP地址進行請求分發;
    • 范式2:Master進程創建一個Listen socket,然后fork出來N個worker進程,這N個worker進程同時偵聽這個socket。

    第一個范式與本文講的epoll無關,更多的體現一種IP層的技術,這里不談,這里僅僅說一下第二個范式。


    為了保證元組的唯一性以及處理的一致性,很長時間以來對于服務器而言,是不允許bind同一個IP地址和端口對的。然而為了可以并發處理多個連接請求,則必須采用某種多處理的方式,為了多個進程可以同時偵聽同一個IP地址端口對,便出現了create listener+fork這種模型,具體來講就是:

    sd = create_listen_socket(); for (i = 0; i < N; i++) {if (fork() == 0) {// 繼承了父進程的文件描述符server(sd);} }

    然而這種模式僅僅是做到了進程級的可擴展性,即一個進程在忙時,其它進程可以介入幫忙處理,底層的socket句柄其實是同一個!簡單點說,這是一個沙漏模型:

    這種模型在處理同一個socket的時候,必須互斥,同時內核必須防止潛在的驚群效應,因為互斥的要求,有且僅有一個進程可以處理特定的請求。這就對編程造成了極大的干擾。

    以本文所描述的case為例,如果不清楚epoll LT模式和ET模式潛在的問題,那么就很容易誤用epoll導致比較令人頭疼的后果。

    非常幸運,reuseport出現后,模型徹底變成了桶狀:

    于是乎,使用了reuseport,一切都變得明朗了:

    • 不再依賴mem模型
    • 不再擔心驚群

    為什么reuseport沒有驚群?首先我們要知道驚群發生的原因,就是同時喚醒了多個進程處理一個事件,導致了不必要的CPU空轉。為什么會喚醒多個進程,因為發生事件的文件描述符在多個進程之間是共享的。而reuseport呢,偵聽同一個IP地址端口對的多個socket本身在socket層就是相互隔離的,在它們之間的事件分發是TCP/IP協議棧完成的,所以不會再有驚群發生。

    所以,結論是什么?

    結論就是全部統一采用reuseport的方式吧,徹底解決驚群問題。

    后記

    浙江溫州皮鞋濕!

    總結

    以上是生活随笔為你收集整理的再谈Linux epoll惊群问题的原因和解决方案的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    国产精品久久久久久久免费大片 | 天天操人人干 | 国产一区二区三区网站 | 婷婷5月激情5月 | 韩日三级在线 | 国产成人精品国内自产拍免费看 | 国内99视频| 日韩欧美91| 日日碰狠狠添天天爽超碰97久久 | 免费观看版 | 国产麻豆果冻传媒在线观看 | 亚洲精品五月 | 特黄特色特刺激视频免费播放 | 在线观看中文字幕亚洲 | 911久久| 手机av看片 | 狠狠88综合久久久久综合网 | 九九欧美视频 | 国产一级免费在线 | 日日干av | 91久久奴性调教 | 亚洲精品国产精品乱码不99热 | 国产欧美三级 | 日韩高清成人 | 免费看wwwwwwwwwww的视频 久久久久久99精品 91中文字幕视频 | 九九在线免费视频 | 成人a免费| 久久艹人人| 在线观看视频三级 | 久久夜色精品国产欧美一区麻豆 | 色综合久久综合中文综合网 | 日韩精品视频免费专区在线播放 | av黄色亚洲 | 在线观看91网站 | 久久精品国产v日韩v亚洲 | 韩国精品在线观看 | 亚洲经典视频 | 成年人app网址 | 夜夜婷婷 | 久久久久久草 | 欧美日韩在线观看一区二区 | 日韩av网站在线播放 | 99精品视频免费 | 日韩精品在线免费观看 | 欧美日韩精品在线视频 | 国产色小视频 | 久久艹艹 | 天天舔天天搞 | 中文字幕中文字幕在线中文字幕三区 | 最新av网址在线观看 | 999男人的天堂 | 波多野结衣久久资源 | 在线免费观看麻豆视频 | 中文字幕在线免费观看视频 | av综合网址 | 特级毛片aaa | 亚洲第一香蕉视频 | 91tv国产成人福利 | 毛片基地黄久久久久久天堂 | 国产特级毛片aaaaaa高清 | 一区二区视频在线播放 | 天天操天天爱天天干 | 成人va在线观看 | 97视频免费 | 国产91av视频在线观看 | 成人91在线观看 | av丝袜在线 | 久草在线在线精品观看 | 久久er99热精品一区二区三区 | 国产精品美女免费视频 | 五月亚洲 | 午夜精品久久久久久久爽 | 九九热在线免费观看 | 国产99久久久国产精品免费看 | 天天综合91 | 久久免费精彩视频 | 国产色在线观看 | 丁香综合激情 | 日本丶国产丶欧美色综合 | 久久久精品久久日韩一区综合 | 麻豆播放 | 国产成人免费av电影 | 国产免费观看av | 天堂av在线中文在线 | 免费一区在线 | 中文字幕文字幕一区二区 | 亚洲粉嫩av| 国产精品国产三级国产不产一地 | 黄色一级大片在线免费看国产一 | 亚洲午夜小视频 | 精品国模一区二区三区 | 国产区精品区 | 亚洲国产精品va在线 | 亚洲精品在线免费观看视频 | 国产精品2018 | 亚洲欧美国产精品va在线观看 | 久久综合欧美 | 久久视频网址 | 激情丁香在线 | 麻豆系列在线观看 | 国产福利精品在线观看 | 久久精品久久99精品久久 | 成人一区不卡 | 午夜视频在线观看一区 | 欧美日韩在线观看一区 | 97成人精品区在线播放 | 成人黄色国产 | 亚洲午夜在线视频 | 亚洲精品在线免费 | 亚洲精品午夜久久久 | 91精品啪| www.五月天婷婷 | 视频国产 | 九九九九九国产 | 成人在线免费看视频 | 国产 一区二区三区 在线 | 99久久精品免费看国产四区 | 一区二区三区不卡在线 | 国产精品永久免费 | 日本高清dvd | 在线日韩av| 缴情综合网五月天 | 在线免费中文字幕 | 99自拍视频在线观看 | 欧美激情第28页 | 午夜手机电影 | 九七在线视频 | 亚洲在线视频播放 | 日韩高清毛片 | 亚洲第一中文字幕 | 96精品视频 | 亚洲一级性 | 狠狠色狠狠色综合日日小说 | 欧美精品久久久久久久久久丰满 | 久久er99热精品一区二区三区 | 99精品毛片 | 午夜三级影院 | 日韩黄色软件 | 久久久久国产精品免费免费搜索 | 国产精品久久久99 | 九九久久久 | 精品一区二区免费 | 超碰人人做 | 狠狠狠狠狠狠狠 | 欧美aⅴ在线观看 | 欧美极品久久 | 久久99电影| 中文字幕资源网在线观看 | 久久久免费在线观看 | 亚洲美女精品区人人人人 | 日本资源中文字幕在线 | 视频一区二区免费 | 成人资源在线观看 | 一级α片免费看 | 国产黄色特级片 | 黄色成品视频 | 日日夜夜天天久久 | 国产午夜精品福利视频 | 中文字幕亚洲高清 | 国产精品99久久久精品 | 91精品国产综合久久久久久久 | 碰超在线| 国产午夜精品在线 | 亚洲经典视频在线观看 | 四虎国产精品成人免费影视 | 天天久久夜夜 | 亚洲永久精品视频 | 色综合婷婷久久 | 日韩av在线小说 | 亚洲成人免费在线 | 免费观看91视频大全 | 国产精品欧美久久久久无广告 | av无限看| 日本中文字幕视频 | 亚洲成av人片| 国产97色在线 | 久久男人免费视频 | 久久精品一区二区三区国产主播 | 亚洲高清久久久 | 国产在线观看你懂的 | 国产字幕在线看 | 黄色资源在线 | 我要看黄色一级片 | 天天色图 | 午夜精品视频一区 | 日韩免费福利 | 亚洲黄色一级电影 | 国产精品日韩在线 | 不卡视频国产 | 91精品少妇偷拍99 | 麻豆国产视频下载 | 国产精品久久久久四虎 | 亚洲黑丝少妇 | 久久刺激视频 | 欧美伊人网| 欧美黄色软件 | 1024手机看片国产 | 亚洲第一香蕉视频 | 五月婷婷中文 | 97人人添人澡人人爽超碰动图 | 激情动态 | a天堂中文在线 | 成人app在线播放 | 久久久久久久综合色一本 | 99人成在线观看视频 | 日韩精品久久久久久中文字幕8 | 成人精品一区二区三区电影免费 | 国产一级做a | 亚洲精色 | 天天舔天天搞 | 免费观看www小视频的软件 | 国产精品美女视频网站 | 91视频网址入口 | 日韩一区正在播放 | 国产爽视频 | 精精国产xxxx视频在线播放 | 色婷婷a| 黄色大片免费播放 | 麻豆成人小视频 | 麻豆视频免费播放 | 久久久久久高潮国产精品视 | 91成人精品一区在线播放69 | 黄色一级片视频 | 成人在线免费av | 久久精品看 | 国产污视频在线观看 | 91香蕉视频 | 亚洲综合精品在线 | 在线免费观看视频一区二区三区 | 韩国av一区二区三区 | 欧美久久电影 | 国产群p | 欧美孕妇视频 | a电影在线观看 | 亚洲精品一区二区三区在线观看 | 一级c片| 亚洲免费av在线播放 | 欧洲性视频 | 日本久久久亚洲精品 | 444av| www.色午夜,com | 免费精品视频 | 成人网大片 | 91人人视频在线观看 | 亚洲影视九九影院在线观看 | 日韩欧美高清一区二区 | 日韩理论在线视频 | 一区二区在线电影 | 欧美a在线免费观看 | 久久久久国产精品视频 | 在线看中文字幕 | 久草手机视频 | 久久综合偷偷噜噜噜色 | 在线观看亚洲专区 | 91av大全| 综合天天 | 一色av | 欧美日韩中文国产一区发布 | 特级毛片网站 | 欧美极品xxxx | 91精品视频一区 | 国产精品一区在线 | 欧美日韩综合在线观看 | 亚洲激情 | 久久女同性恋中文字幕 | 久久视频国产 | 特级西西人体444是什么意思 | 亚洲dvd| 在线小视频你懂得 | 黄色成人av | 一级理论片在线观看 | 激情视频网页 | 日日夜夜精品视频天天综合网 | 久久高清免费观看 | 久久人人做| 色婷婷久久一区二区 | 超碰在线公开免费 | 四虎国产精品永久在线国在线 | 91免费网 | 91污在线| 亚洲天堂网站 | 天天摸夜夜操 | 在线观看视频一区二区三区 | 久久久精华网 | 久久成人精品 | 中文字幕日韩在线播放 | 国模一二三区 | 夜夜干夜夜 | 日本高清xxxx | 亚洲精品女人久久久 | 亚洲黄色精品 | 五月开心六月伊人色婷婷 | 最新黄色av网址 | www.玖玖玖 | 亚洲日本va在线观看 | 国产黄色一级大片 | 九九热免费视频在线观看 | 国产精品一区二区久久 | 手机在线中文字幕 | 96精品视频 | 18pao国产成视频永久免费 | 国产成人一区二区三区电影 | 国产精品精品国产 | 国产精品99久久99久久久二8 | 日韩精品一区二区在线观看 | 91视频电影 | 丁香5月婷婷久久 | 久草视频中文 | 国产精品 日韩 | 中文字幕资源网在线观看 | 国产小视频在线播放 | 黄色软件视频大全免费下载 | 在线观看日韩视频 | 最近中文字幕久久 | 成人黄色电影在线观看 | 精品视频一区在线观看 | 波多野结衣视频一区二区 | 国产做爰视频 | 国产精品日韩欧美一区二区 | 国产乱码精品一区二区三区介绍 | 久久艹在线观看 | 日韩免费在线视频观看 | 色欧美日韩 | 日本三级在线观看中文字 | 欧美另类sm图片 | 日韩专区中文字幕 | 天天色天天 | 国产精品永久免费视频 | 精品一区91 | 久久新 | 九九热在线播放 | 69av视频在线 | 精品国产精品国产偷麻豆 | 天天干天天拍天天操 | 麻豆视频国产精品 | 69久久夜色精品国产69 | 精品uu | 手机看片福利 | 久久综合欧美精品亚洲一区 | 久久天堂网站 | 爱爱av网站| 97av在线| 精品国产久 | 国产午夜精品久久 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 欧美日韩国内在线 | 96亚洲精品久久久蜜桃 | 人人cao| av中文字幕在线电影 | 98涩涩国产露脸精品国产网 | 国产高清不卡在线 | 在线免费观看国产黄色 | 精品国产成人在线影院 | 亚洲成人资源在线 | 福利一区二区 | 国产日韩精品久久 | 五月天久久狠狠 | 国产一区私人高清影院 | 成人污视频在线观看 | 天天插天天干天天操 | 一级免费片 | 国产一区二区三区四区大秀 | 亚洲精品在线网站 | 日韩精品在线看 | 狠狠色伊人亚洲综合成人 | 91免费高清观看 | 欧美一二三视频 | 91视频一8mav| 欧美电影黄色 | 欧美网址在线观看 | 在线看片91 | 中文字幕 国产视频 | 激情在线五月天 | 人交video另类hd | 久热免费在线观看 | 亚洲视频在线免费观看 | 日韩精品久久一区二区三区 | 激情av五月婷婷 | 人人射人人爱 | 黄色av影院 | 在线免费观看不卡av | 人人干人人干人人干 | 精品久久久久久久久久国产 | 一区二区免费不卡在线 | aⅴ视频在线 | 在线看国产一区 | 一区中文字幕电影 | 成人性生交大片免费观看网站 | 国产精品手机视频 | 最新日韩在线观看视频 | 成人一区二区在线观看 | 97看片网| 啪啪肉肉污av国网站 | 婷婷久月| 欧美日韩调教 | 欧美日韩精品影院 | 在线观看免费黄色 | 亚洲成人av一区 | 中文视频在线播放 | 欧美色图亚洲图片 | 亚洲jizzjizz日本少妇 | 亚洲,国产成人av | 综合国产视频 | 午夜色大片在线观看 | 国内精品福利视频 | 国产精品情侣视频 | 精品国产一二三 | 国产乱视频 | 黄色大片中国 | 香蕉视频在线播放 | 国产视频1区2区3区 久久夜视频 | 九色激情网 | 国产xvideos免费视频播放 | 久久激五月天综合精品 | 在线国产黄色 | 欧美精品天堂 | 最近中文字幕完整高清 | 奇米网网址| 91视视频在线直接观看在线看网页在线看 | 三级视频日韩 | 九九精品在线观看 | 麻豆91在线 | 久久国产热 | www.少妇| 亚洲一级电影视频 | 亚洲国产一区二区精品专区 | 天天综合视频在线观看 | 高清不卡一区二区三区 | 91网站在线视频 | 国产精品免费久久久久久久久久中文 | 色夜影院| 国产亚洲精品美女久久 | 五月婷婷激情综合 | 九九热国产视频 | 丁香激情综合久久伊人久久 | 日韩中文字幕视频在线观看 | 99精品国产兔费观看久久99 | 亚洲国产精品免费 | 黄网站色视频免费观看 | 91成人欧美 | 精品成人a区在线观看 | 91桃色免费观看 | 午夜 免费| 视频一区在线免费观看 | 久久久久久久久久久电影 | 亚洲精品在线观看不卡 | 国产精品精品久久久久久 | 操操综合| 国产一级在线看 | 91在线精品秘密一区二区 | 黄色在线成人 | 在线成人一区 | 日韩有码第一页 | 西西www444| 日韩欧美成人网 | 国产91在线 | 美洲 | 久久久这里有精品 | 久久视频一区 | 日韩区视频 | 欧美日韩三级在线观看 | 亚洲精品乱码久久久久久久久久 | 中文字幕人成不卡一区 | 黄色片软件网站 | 国产在线观看免费 | 国产成人黄色av | 最新av免费在线观看 | 黄色亚洲片| 狠狠色丁香婷婷综合橹88 | 久久国内精品视频 | 免费高清影视 | .精品久久久麻豆国产精品 亚洲va欧美 | 精品国产一区二区三区久久久 | 色综合咪咪久久网 | 在线a亚洲视频播放在线观看 | 国产视频美女 | 毛片网免费 | 国产日韩视频在线 | 超碰日韩在线 | 国产精品一区二区三区在线看 | 久久国产精品免费视频 | 久久久免费看片 | 精品久久久成人 | 99热手机在线 | 欧美一级片免费在线观看 | www.久草视频 | 色爽网站 | 国产精品久久久久影院日本 | 特级黄色片免费看 | 日韩美女一级片 | 人人干人人干人人干 | 国产日韩欧美视频 | 欧美最新另类人妖 | 99精品电影 | 69国产成人综合久久精品欧美 | 天天插狠狠干 | 国产日韩欧美精品在线观看 | av成人免费在线看 | 香蕉精品在线观看 | 亚洲高清在线 | 波多野结衣网址 | 六月激情 | 午夜骚影 | 国产96av| 777奇米四色 | 久久综合免费视频 | 在线不卡中文字幕播放 | 国产精品国产三级国产aⅴ无密码 | 字幕网资源站中文字幕 | 一区久久久 | 在线观看中文av | www天天干com| 亚洲久草网| 久久久免费视频播放 | 91成人精品一区在线播放69 | 国产成人免费高清 | 亚洲国产欧美在线看片xxoo | 天堂中文在线视频 | 一级欧美黄 | 久久高清| 日韩区视频 | 国产精品久久久av久久久 | 久久99免费观看 | 777xxx欧美 | 婷婷色中文字幕 | 日韩免费一区二区在线观看 | 色综合色综合色综合 | 99国产在线 | www.超碰97.com | 91大神dom调教在线观看 | 国产精品初高中精品久久 | 久久综合影音 | 韩国精品在线 | 五月婷婷.com| 国产精品网站 | 国产精品欧美 | 亚洲日本黄色 | 国产中文字幕视频在线观看 | 久久福利综合 | 在线观看免费国产小视频 | 91av在线免费观看 | 丁香六月婷婷激情 | 91色在线观看视频 | 在线观看免费黄色 | 日韩激情av在线 | 黄色精品网站 | 国产一二区免费视频 | av网在线观看 | 久艹在线免费观看 | 欧美日韩国产精品一区二区亚洲 | 免费在线一区二区 | 麻豆91精品 | 久久综合婷婷国产二区高清 | 亚洲精品久久久久58 | 99精品国产99久久久久久97 | 美女网站在线看 | 亚州国产精品 | 一本一道久久a久久综合蜜桃 | 亚洲在线综合 | 亚洲美女免费视频 | 国产精品白浆视频 | 色av色av色av | 日韩中文字幕亚洲一区二区va在线 | 成人国产在线 | 91福利国产在线观看 | 亚洲国产精品小视频 | 激情丁香综合五月 | 久久成人精品电影 | 免费a级观看| 欧洲高潮三级做爰 | 91精品国产高清自在线观看 | 日韩一区二区三区免费电影 | 精品国产一区二区三区在线观看 | 国产精品久久久区三区天天噜 | 久色伊人 | 91大神在线观看视频 | 国产伦理一区二区三区 | 日韩在线免费 | 成人av影院在线观看 | 日韩有码专区 | 国产精品va在线观看入 | 久久国产一二区 | 亚洲欧美日韩精品久久奇米一区 | 国产一二三在线视频 | 91色网址| 在线观看国产v片 | 色综合天 | 五月网婷婷 | 欧美福利视频一区 | 国产午夜影院 | 国产在线中文字幕 | 国产 日韩 在线 亚洲 字幕 中文 | .国产精品成人自产拍在线观看6 | 亚洲一区二区三区在线看 | 国产一区二区精品 | 中文字幕在线观看视频一区 | 久久成人午夜视频 | 色婷婷六月| 中文字幕色在线视频 | 人人爽人人爽 | 中文字幕丝袜 | 亚洲精品国偷拍自产在线观看蜜桃 | 免费观看一级成人毛片 | 99久久精品国产一区二区三区 | 字幕网av | 一级黄色免费网站 | 99久热 | 国产 亚洲 欧美 在线 | 国产精品成人国产乱 | 人人爽人人爽人人片 | 婷婷日韩| 一级黄色片在线免费观看 | 天天射天天 | 午夜狠狠干 | 6699私人影院| 国产一级免费在线观看 | 激情综合六月 | 亚洲欧洲一区二区在线观看 | 午夜婷婷在线观看 | 日av免费| 久久成年人视频 | 成人精品一区二区三区中文字幕 | 在线观看日韩av | 国产亚洲免费观看 | 最近在线中文字幕 | 日韩大片在线免费观看 | 美女免费网视频 | 日韩二区精品 | 国产精品区一区 | 一区二区中文字幕在线播放 | 日韩在线不卡 | 99久久婷婷国产 | 日韩免费电影 | 少妇视频在线播放 | 97视频在线观看免费 | 久久狠狠一本精品综合网 | 五月天婷婷免费视频 | 日韩欧美精品在线观看 | 色在线网站| 超碰在线日韩 | 精品久久免费 | 尤物97国产精品久久精品国产 | 免费日韩 精品中文字幕视频在线 | 亚洲成av人片在线观看 | 国产亚洲免费的视频看 | 久久免费播放视频 | 亚洲精品中文在线资源 | 日韩中文在线播放 | 日韩理论片中文字幕 | 91网址在线观看 | 国产精品一区二区三区电影 | 欧美日韩99 | 一区二区三区日韩精品 | 日韩午夜在线播放 | 久久午夜剧场 | 日韩电影久久 | www.久久久 | 国产精品欧美久久久久天天影视 | 国产一区私人高清影院 | av网站播放 | 亚洲精品久久激情国产片 | 久久综合九色综合久99 | 综合久久一本 | 日本久久久久久久久久 | 成人黄色毛片 | 亚洲日韩中文字幕在线播放 | 最近最新中文字幕 | 欧美激情视频一二区 | 99精品视频一区 | 国产福利一区二区三区视频 | 亚洲欧美国内爽妇网 | 91在线看网站 | 久草网在线观看 | 国产r级在线观看 | 日韩在线观看电影 | 国产在线观看免费观看 | 久久午夜电影 | 国产免费又爽又刺激在线观看 | 在线观看亚洲电影 | 国产精品久久精品国产 | 麻豆小视频在线观看 | 最新中文字幕在线资源 | 国产成人精品不卡 | 日韩免费区 | 干av在线 | 久章操| 热久久免费视频精品 | 免费看一级黄色 | 国产成人精品一二三区 | 人人草在线视频 | a视频免费 | 视频一区二区国产 | 超碰国产97 | 亚洲男模gay裸体gay | 国产精品男女 | 色婷婷av一区 | 亚洲一区欧美激情 | 天天色.com | 免费欧美高清视频 | 日韩中文字幕在线 | 99re视频在线观看 | 国产精品免费观看国产网曝瓜 | 久久精品香蕉 | 国产在线一区二区 | 91久久精品一区 | 91九色成人蝌蚪首页 | 日韩欧美视频一区二区三区 | av丝袜制服 | 日韩免费在线视频观看 | 日日干美女 | 久久黄页 | 中文字幕免费播放 | 国产在线一区观看 | 一本一本久久a久久 | 色国产精品一区在线观看 | 亚洲黄网址 | 96国产精品视频 | 天天摸天天操天天舔 | 精品国产成人在线 | 精品中文字幕视频 | 久久草精品 | 精品久久1| 91香蕉视频在线下载 | 欧美日韩一区三区 | 久久久国产毛片 | 黄色成人av| 国产精品免费观看在线 | 久久免费精品一区二区三区 | 日韩欧美网站 | 欧美日韩视频一区二区三区 | www.色综合.com | 日韩在线 一区二区 | 久章操 | 国产色爽| 国产女人免费看a级丨片 | 精品欧美一区二区三区久久久 | 久久精品在线视频 | 天天干天天天天 | 国产精品一区二区在线观看 | 中文字幕成人网 | 91原创在线观看 | 久操久| 免费一级片在线 | 69夜色精品国产69乱 | 丝袜精品视频 | 激情综合色综合久久 | 国产精品丝袜久久久久久久不卡 | 伊人夜夜 | 欧美精品一区二区免费 | 日韩国产精品毛片 | 久久精品中文字幕免费mv | 日韩在线视频免费看 | 日韩视频中文 | 黄色软件在线观看 | 怡红院成人在线 | 久久午夜影视 | 精品免费观看视频 | 正在播放国产一区二区 | 国产一区二区视频在线播放 | 精品在线视频一区二区三区 | 国产一区二区电影在线观看 | 黄污网站在线观看 | 国产精品久久久久久久久免费看 | 99国产精品视频免费观看一公开 | 久久精品中文 | 九九av| 中文字幕在线专区 | 色干干| 日韩有码中文字幕在线 | av日韩av| 成人在线免费观看网站 | 精品欧美小视频在线观看 | 最近中文字幕mv免费高清在线 | 国产 字幕 制服 中文 在线 | bbw av| 四虎影视av | 国产在线精品一区二区不卡了 | 91视频电影 | av中文在线影视 | 国产又粗又猛又色又黄网站 | 亚洲一区二区三区四区在线视频 | 手机成人av在线 | 青青河边草免费观看 | 亚洲清纯国产 | 久草视频一区 | a在线观看免费视频 | www.久久婷婷 | 久久国产电影院 | 国产中文字幕视频 | 天天操天天摸天天干 | 天天操天天色天天射 | 成年人电影免费在线观看 | 国产精品视频永久免费播放 | 久久精品免费电影 | 91精品影视 | 综合色亚洲 | 久久久国产一区二区 | 美女视频黄频大全免费 | 中文字幕在线资源 | 久草国产在线 | 成人毛片在线观看视频 | 99精品小视频 | 激情婷婷亚洲 | 亚洲欧美一区二区三区孕妇写真 | 一区二区三区四区影院 | 国内精品久久久精品电影院 | 国偷自产中文字幕亚洲手机在线 | 蜜臀av一区二区 | 亚洲成av人电影 | 欧美综合干 | 欧美午夜性生活 | 成人影视免费看 | 日韩午夜三级 | 又黄又刺激又爽的视频 | 91精品国自产拍天天拍 | 91av九色| 最近高清中文字幕在线国语5 | 少妇精品久久久一区二区免费 | 你操综合| 久久夜色电影 | 最近免费中文视频 | 五月激情av | 成人18视频| 夜夜操夜夜干 | 91一区啪爱嗯打偷拍欧美 | 97香蕉久久超级碰碰高清版 | 天天射天天射天天射 | 国产精品淫片 | 国产精品毛片久久久 | 欧美伦理电影一区二区 | 久久综合免费视频影院 | 久久久国产精品人人片99精片欧美一 | 在线观看视频福利 | 国产美女网站在线观看 | 亚洲精品91天天久久人人 | 中文字幕在线观看一区 | 亚洲黄色一级视频 | 91中文字幕在线视频 | 国产精品美女久久 | 欧美一区二区日韩一区二区 | 国产成人综合精品 | 999久久a精品合区久久久 | 91在线视频免费 | 国产精品久久人 | 国语精品免费视频 | 亚洲精品天天 | 四虎在线免费观看 | 中文字幕永久免费 | 日韩乱色精品一区二区 | 中文字幕123区 | 久久久久免费电影 | 国产精品麻豆免费版 | 亚洲精品456在线播放 | 日本黄色片一区二区 | 这里只有精彩视频 | 天天av资源 | 最近免费中文字幕大全高清10 | 亚洲国产成人精品在线 | 欧美一级免费片 | 五月天色中色 | 亚洲国内在线 | 狠狠色网 | 欧美成人91 | 久久综合九九 | 日韩精品观看 | 99精品免费久久久久久日本 | 色诱亚洲精品久久久久久 | 日韩欧美高清在线 | www.久久久精品 | 日韩av影片在线观看 | 欧美一级在线观看视频 | 日韩欧美在线免费观看 | 草久久影院 | 外国av网| 成人va天堂 | 精品一区二区免费 | 久久国产精品精品国产色婷婷 | 免费看国产一级片 | 美女久久一区 | av在线不卡观看 | 国内精品久久久久久久久久久久 | 国产久草在线 | 国产精品欧美精品 | 亚洲国产精品99久久久久久久久 | 天天射,天天干 | 香蕉视频久久 | 午夜国产福利在线 | 人人爽夜夜爽 | 97超碰.com| 亚洲精品97 | 天天躁天天躁天天躁婷 | 成人wwwxxx视频 | 国产成a人亚洲精v品在线观看 | 欧美日韩国产在线观看 | 日韩丝袜在线观看 | 欧美人人| 成年人在线电影 | 天天干天天操天天干 | 天天操天天操一操 | 毛片无卡免费无播放器 | 91精品国产综合久久婷婷香蕉 | 久久久久久黄色 | 免费精品在线观看 | 欧美午夜久久久 | 玖玖在线资源 | 视频国产在线 | 日韩视频一区二区三区在线播放免费观看 | 欧洲av不卡 | a v在线观看 | 天天操天天干天天摸 | 黄色片视频在线观看 | 97品白浆高清久久久久久 | 一本一道久久a久久精品 | 三上悠亚一区二区在线观看 | 中文字幕在线视频免费播放 | 亚洲成人精品影院 | av观看免费在线 | 国内精品久久久久影院一蜜桃 | 国产一区二区不卡视频 | 在线观看成人网 | 欧美日韩另类在线观看 | 在线国产视频一区 | 久久综合加勒比 | 久久er99热精品一区二区三区 | 精品久久久久一区二区国产 | 深夜免费福利网站 | 五月天,com| 国产亚洲精品女人久久久久久 | 日本成人黄色片 | 91九色蝌蚪国产 | 国产美女无遮挡永久免费 | 国产精品6999成人免费视频 | 成人黄色毛片 | 天天色天天干天天色 | 久草a在线 | 日韩精品一区二区三区三炮视频 | 丁香六月国产 | 四虎在线免费视频 | 日韩成片 | 97热在线观看 | 91毛片视频 | 91成人在线免费观看 | 二区精品视频 | 欧美在线观看视频一区二区三区 | 99在线高清视频在线播放 | 91成年人网站 | 免费中文字幕 | 四虎在线观看网址 | 国产3p视频| 在线免费黄色毛片 | 国产精品久久久av久久久 | 午夜久久久久 | 一级黄色片在线 | 国产黄色片免费观看 | 国产亚洲精品久久 | 国内小视频在线观看 | 亚洲精品美女在线 | 日韩午夜高清 | 色综合a| 欧美亚洲另类在线视频 | 国产精品毛片一区二区三区 | 久草视频99 | 精品国产亚洲一区二区麻豆 | 亚洲视频一区二区三区在线观看 | 日韩精品一区二区三区中文字幕 | 久久黄色精品视频 | 四虎国产永久在线精品 | 国产亚洲精品久久久久久移动网络 | 亚洲毛片在线观看. | 国外成人在线视频网站 | 超碰97中文 | 91av影视| 激情综合五月天 | 日韩成人在线一区二区 | 亚洲最新av网站 | 日日麻批40分钟视频免费观看 | 一级一片免费视频 | 黄色av网站在线观看免费 | 亚洲第一久久久 | 日韩天堂在线观看 | 亚洲欧美国内爽妇网 | 综合色在线观看 | 亚洲精品毛片一级91精品 | 国产黄在线 | 国产精品av免费在线观看 | 国产一区二区三区 在线 | 成年人免费在线观看网站 | 日本精a在线观看 | 精品国产伦一区二区三区免费 | 亚洲成人资源在线 | 天天草天天干天天 | 成人性生爱a∨ | 亚洲激情 | 久草成人在线 | 国产精品久久久久一区二区三区共 | 久草在线视频精品 | 色综合色综合色综合 | 水蜜桃亚洲一二三四在线 | 国产剧情一区二区在线观看 |