精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Go BIO/NIO探討:Go netpoll 是如何工作的

開發 前端
一般我們聊到 netpoll 時,是指 Go runtime 中借助于epoll對套接字進行批量監聽、數據到來時喚醒特定goroutine的機制。

一般我們聊到 netpoll 時,是指 Go runtime 中借助于epoll對套接字進行批量監聽、數據到來時喚醒特定goroutine的機制。對應的代碼存放在runtime/netpoll.go 和 runtime/netpoll_epoll.go (只考慮linux) 中。為此 runtime 提供了兩大類函數:

第一類:調用方是 Go Runtime。

  1. netpoll: 檢查有事件發生的套接字,并返回處于pdReady狀態的goroutine列表,基于epoll_wait。
  2. netpollBreak: 向 netpollBreakWr 寫入一個字節數據,通過管道傳到 netpollBreakRd,epoll_wait 監聽到read pipe上的event,立即返回。

第二類:調用方是internal/poll、net、net/http等。

  1. poll_runtime_pollServerInit(netpollGenericInit): 初始化poller,基于epoll_create1。
  2. poll_runtime_pollOpen: 將套接字添加到監聽列表,基于 epoll_ctl。
  3. poll_runtime_pollWait: 等待套接字上的事件,可以休眠(gopark)當前goroutine, 借助于netpollblock函數。
  4. poll_runtime_pollUnblock: 使用Unblock模式進行poll。
  5. poll_runtime_pollClose: 將套接字從監聽列表刪除,基于 epoll_ctl。
  6. poll_runtime_pollReset: nonblock模式下 prepareRead/prepareWrite 使用。

這些函數都會被link到 internal/poll.runtime_xxx, xxx 可以是。
runtime_pollServerInit/runtime_pollOpen等。

后面我們挑一些主要的函數來說一下。

netpollGenericInit 初始化 poller

netpollGenericInit 保證 poller 被初始化,原子變量netpollInited保證其僅被初始化一次。

func netpollGenericInit() {
if atomic.Load(&netpollInited) == 0 {
lockInit(&netpollInitLock, lockRankNetpollInit)
lock(&netpollInitLock)
if netpollInited == 0 {
netpollinit()
atomic.Store(&netpollInited, 1)
}
unlock(&netpollInitLock)
}
}

這個函數只是一個殼,初始化邏輯封裝在netpollinit函數中,依賴于平臺具體的實現。linux下,init的邏輯是:

  1. 通過epoll_create1系統調用創建 epoll fd。
  2. 創建一對 read/write pipe。pipe的一個特性是向 write pipe寫入數據,read pipe 就能收到同樣的數據。
  3. 通過epoll_ctl將 write pipe 對應的fd 加入到監聽列表。

單獨創建一對pipe后,runtime就能夠按需中斷epoll_wait,讓netpoll函數立即返回。

func netpollinit() {
epfd = epollcreate1(_EPOLL_CLOEXEC)
if epfd < 0 {
epfd = epollcreate(1024)
if epfd < 0 {
println("runtime: epollcreate failed with", -epfd)
throw("runtime: netpollinit failed")
}
closeonexec(epfd)
}
r, w, errno := nonblockingPipe()
if errno != 0 {
println("runtime: pipe failed with", -errno)
throw("runtime: pipe failed")
}
ev := epollevent{
events: _EPOLLIN,
}
*(**uintptr)(unsafe.Pointer(&ev.data)) = &netpollBreakRd
errno = epollctl(epfd, _EPOLL_CTL_ADD, r, &ev)
if errno != 0 {
println("runtime: epollctl failed with", -errno)
throw("runtime: epollctl failed")
}
netpollBreakRd = uintptr(r)
netpollBreakWr = uintptr(w)
}

netpoll函數

netpoll函數的功能是檢查可用的網絡連接,它的工作流程是(happy path):

  1. 創建size=128的epollevent數組, 以接收事件。
  2. 調用epollwait等待事件: 依賴epoll_wait系統調用。
  3. 遍歷epoll events,對于每個event創建一個pollDesc對象調用netpollready,找到對應的goroutine,并將其狀態從pdWait修改為pdReady。
  4. 返回pdReady狀態的 goroutine列表 (gList)。

struct pollDesc中包含兩個信號量字段,可以表示四種狀態:

  1. pdReady: io ready信號等待被接收,goroutine可以消費這個信號,邏輯上是把信號量改成nil。
  2. pdWait: goroutine已經準備好在該信號量上阻塞,但還沒有阻塞;如果goroutine通過gopark阻塞,狀態會變成G pointer如果并發的io ready信號到達,狀態會改成pdReady如果并發的timeout/close信號到達,狀態會被改成nil。
  3. G pointer: goroutine被阻塞在信號量上,可以被下面兩類事件喚醒:io ready信號到來時,狀態被修改好pdReadytimeout/close信號到來時,狀態被修改為nil。
  4. nil: 不是上面三種狀態。

對應一些輔助函數:

  • netpollblock 函數將goroutine狀態從 pdReady 轉化成 pdWait,并gopark當前goroutine。
  • netpollunblock 函數將goroutine狀態從 pdWait 轉換為 pdReady 或 nil。

netpoll函數的代碼在runtime/netpoll_epoll.go中,部分代碼如下:

func netpoll(delay int64) gList {
// epoll fd 為-1,說明不需要poll
if epfd == -1 {
return gList{}
}
var waitms int32
// ...省略一段代碼
var events [128]epollevent
retry:
n := epollwait(epfd, &events[0], int32(len(events)), waitms)
if n < 0 {
if n != -_EINTR {
println("runtime: epollwait on fd", epfd, "failed with", -n)
throw("runtime: netpoll failed")
}
// If a timed sleep was interrupted, just return to
// recalculate how long we should sleep now.
if waitms > 0 {
return gList{}
}
goto retry
}
var toRun gList
for i := int32(0); i < n; i++ {
ev := &events[i]
if ev.events == 0 {
continue
}

if *(**uintptr)(unsafe.Pointer(&ev.data)) == &netpollBreakRd {
// ... read pipe 有數據
// 不需要喚醒任何goroutine
}

var mode int32
if ev.events&(_EPOLLIN|_EPOLLRDHUP|_EPOLLHUP|_EPOLLERR) != 0 {
mode += 'r'
}
if ev.events&(_EPOLLOUT|_EPOLLHUP|_EPOLLERR) != 0 {
mode += 'w'
}
if mode != 0 {
pd := *(**pollDesc)(unsafe.Pointer(&ev.data))
pd.setEventErr(ev.events == _EPOLLERR)
// 將goroutine置為 pdReady
// 并添加到toRun *gList
netpollready(&toRun, pd, mode)
}
}
return toRun
}

備注: netpollready 函數借助于netpollunblock修改goroutine狀態,并將其加到 io ready 的 goroutine list。

runtime在調用 netpoll 時,通常采用的是 nonblock 模式(delay=0), 只有在 findrunnable 的最后一個環節,會檢查是否有單獨的M(GMP中的M)進行net polling,如果沒有,會block等待delay參數指定的時間。

netpollBreak 函數

netpollBreak函數的功能比較簡單,但實現比較有意思。它和netpoll函數通過變量netpollWakeSig進行交互,由于在不同的goroutine中,所以對于該變量的操作都是原子?操作。

// netpollBreak interrupts an epollwait.
func netpollBreak() {
if atomic.Cas(&netpollWakeSig, 0, 1) {
for {
var b byte
n := write(netpollBreakWr, unsafe.Pointer(&b), 1)
if n == 1 {
break
}
if n == -_EINTR {
continue
}
if n == -_EAGAIN {
return
}
println("runtime: netpollBreak write failed with", -n)
throw("runtime: netpollBreak write failed")
}
}
}

poll_runtime_pollOpen 函數

poll_runtime_pollOpen 的邏輯分為三塊:

  1. 給 pollDesc 分配內存。
  2. 初始化 pollDesc 對象。
  3. 借助于 netpollopen 注冊epoll監聽(netpollopen在linux下是 epoll_ctl)。
  4. 返回 pollDesc 對象。

poll_runtime_pollOpen函數的實現位于 runtime/netpoll.go 中, 主要邏輯如下:

//go:linkname poll_runtime_pollOpen internal/poll.runtime_pollOpen
func poll_runtime_pollOpen(fd uintptr) (*pollDesc, int) {
pd := pollcache.alloc()
lock(&pd.lock)
wg := pd.wg.Load()
if wg != 0 && wg != pdReady {
throw("runtime: blocked write on free polldesc")
}
rg := pd.rg.Load()
if rg != 0 && rg != pdReady {
throw("runtime: blocked read on free polldesc")
}
pd.fd = fd
// ... 省略部分初始化邏輯
unlock(&pd.lock)

errno := netpollopen(fd, pd)
if errno != 0 {
pollcache.free(pd)
return nil, int(errno)
}
return pd, 0
}

// 位于net/netpoll_epoll.go
func netpollopen(fd uintptr, pd *pollDesc) int32 {
var ev epollevent
ev.events = _EPOLLIN | _EPOLLOUT | _EPOLLRDHUP | _EPOLLET
*(**pollDesc)(unsafe.Pointer(&ev.data)) = pd
return -epollctl(epfd, _EPOLL_CTL_ADD, int32(fd), &ev)
}

poll_runtime_pollWait 函數

poll_runtime_pollWait 函數只是對 netpollblock 函數的封裝,增加了容錯。值得注意的是,該函數不是runtime觸發的,而是用戶程序觸發的。

func poll_runtime_pollWait(pd *pollDesc, mode int) int {
errcode := netpollcheckerr(pd, int32(mode))
if errcode != pollNoError {
return errcode
}
// As for now only Solaris, illumos, and AIX use level-triggered IO.
if GOOS == "solaris" || GOOS == "illumos" || GOOS == "aix" {
netpollarm(pd, mode)
}
for !netpollblock(pd, int32(mode), false) {
errcode = netpollcheckerr(pd, int32(mode))
if errcode != pollNoError {
return errcode
}
// Can happen if timeout has fired and unblocked us,
// but before we had a chance to run, timeout has been reset.
// Pretend it has not happened and retry.
}
return pollNoError
}

下面我們看下用戶程序如何觸發 poll_runtime_xxx 系列的函數。首先,套接字分為兩類:LISTEN套接字(Server套接字) 和 ESTABLISHED套接字(TCPConn);

  • LISTEN 套接字通過系統調用 socket/bind/listen 去生成。
  • ESTABLISHED 套接字通過系統調用 accept 去生成。

LISTEN套接字(Server套接字)

從http server的角度來看,LISTEN套接字注冊epoll監聽的鏈路如下:

// net/http/server.go
func ListenAndServe(addr string, handler Handler) error

// net/http/server.go
func (srv *Server) ListenAndServe() error

// net/dial.go
func Listen(network, address string) (Listener, error) {
var lc ListenConfig
return lc.Listen(context.Background(), network, address)
}

// net/dial.go
func (lc *ListenConfig) Listen(ctx context.Context, network, address string) (Listener, error)

// net/tcpsock_posix.go
func (sl *sysListener) listenTCP(ctx context.Context, laddr *TCPAddr) (*TCPListener, error)

// net/ipsock_posix.go
func internetSocket(ctx context.Context, net string, laddr, raddr sockaddr, sotype, proto int, mode string, ctrlFn func(string, string, syscall.RawConn) error) (fd *netFD, err error)

// net/sock_posix.go
func socket(ctx context.Context, net string, family, sotype, proto int, ipv6only bool, laddr, raddr sockaddr, ctrlFn func(string, string, syscall.RawConn) error) (fd *netFD, err error)

// net/sock_posix.go
func (fd *netFD) listenStream(laddr sockaddr, backlog int, ctrlFn func(string, string, syscall.RawConn) error)
if err = fd.init(); err != nil {
return err
}

// net/fd_unix.go
func (fd *netFD) init() error {
// fd.pfd 類型是 poll.FD
return fd.pfd.Init(fd.net, true)
}

// internal/poll/fd_unix.go
func (fd *FD) Init(net string, pollable bool) error {
// We don't actually care about the various network types.
if net == "file" {
fd.isFile = true
}
if !pollable {
fd.isBlocking = 1
return nil
}
err := fd.pd.init(fd)
if err != nil {
// If we could not initialize the runtime poller,
// assume we are using blocking mode.
fd.isBlocking = 1
}
return err
}

// internal/poll/fd_poll_runtime.go
func (pd *pollDesc) init(fd *FD) error {
serverInit.Do(runtime_pollServerInit)
ctx, errno := runtime_pollOpen(uintptr(fd.Sysfd))
if errno != 0 {
return errnoErr(syscall.Errno(errno))
}
pd.runtimeCtx = ctx
return nil
}

ESTABLISHED套接字(TCPConn)

http server accept 新的tcp conn。

// net/http/server.go
func (srv *Server) Serve(l net.Listener) error {
for {
rw, err := l.Accept()

// net/tcpsock.go
func (l *TCPListener) Accept() (Conn, error)

func (ln *TCPListener) accept() (*TCPConn, error) {
fd, err := ln.fd.accept()

// net/fd_posix.go
func (fd *netFD) accept() (netfd *netFD, err error) {
d, rsa, errcall, err := fd.pfd.Accept()
// 省略部分代碼
if err = netfd.init(); err != nil
// 省略部分代碼


// internal/poll/fd_unix.go
func (fd *FD) Init(net string, pollable bool) error

// internal/poll/fd_poll_runtime.go
func (pd *pollDesc) init(fd *FD) error

關于 net.netFD struct

netFD是對套接字(網絡文件描述符)的封裝。對于Server套接字而言,可以通過accept方法從Server套接字(LISTEN套接字)獲取新的TCP連接(或ESTABLISHED套接字)。Linux的accept系統調用返回的ESTABLISHED套接字是一個int值,通過 newFD 和 init 函數將其封裝為一個完整的 netFD結構,后面會被封裝為一個net.TCPConn。

對于操作系統而言,LISTEN套接字和ESTABLISHED套接字都只是一個int類型的文件描述符,沒有本質區別。系統調用accept和read都是從套接字讀取數據,所以epoll里會放到一個batch里去監聽。

這是 netFD 的定義和accept方法的實現:

// Network file descriptor.
type netFD struct {
pfd poll.FD

// immutable until Close
family int
sotype int
isConnected bool // handshake completed or use of association with peer
net string
laddr Addr
raddr Addr
}

func (fd *netFD) accept() (netfd *netFD, err error) {
d, rsa, errcall, err := fd.pfd.Accept()
if err != nil {
if errcall != "" {
err = wrapSyscallError(errcall, err)
}
return nil, err
}

if netfd, err = newFD(d, fd.family, fd.sotype, fd.net); err != nil {
poll.CloseFunc(d)
return nil, err
}
if err = netfd.init(); err != nil {
netfd.Close()
return nil, err
}
lsa, _ := syscall.Getsockname(netfd.pfd.Sysfd)
netfd.setAddr(netfd.addrFunc()(lsa), netfd.addrFunc()(rsa))
return netfd, nil
}

net.netFD 依賴 poll.FD 實現poll功能。區別正如名字所展示,net.netFD是封裝了網絡相關的功能,而 poll.FD是更為通用的FD,封裝了文件描述符上能進行的操作。其定義如下:

// FD is a file descriptor. The net and os packages use this type as a
// field of a larger type representing a network connection or OS file.
type FD struct {
// Lock sysfd and serialize access to Read and Write methods.
fdmu fdMutex

// System file descriptor. Immutable until Close.
Sysfd int

// I/O poller.
pd pollDesc

// Writev cache.
iovecs *[]syscall.Iovec

// Semaphore signaled when file is closed.
csema uint32

// Non-zero if this file has been set to blocking mode.
isBlocking uint32

// Whether this is a streaming descriptor, as opposed to a
// packet-based descriptor like a UDP socket. Immutable.
IsStream bool

// Whether a zero byte read indicates EOF. This is false for a
// message based socket connection.
ZeroReadIsEOF bool

// Whether this is a file rather than a network socket.
isFile bool
}

poll.FD 依賴 poll.pollDesc 實現poll功能。poll.pollDesc 實現了 IO polling 的功能。poll.pollDesc 有一系列的方法,比如 init、wait、close、prepare 等都是對 runtime_pollXXX 函數系列的封裝,下面詩pollDesc的部分邏輯:

type pollDesc struct {
runtimeCtx uintptr
}

var serverInit sync.Once

func (pd *pollDesc) init(fd *FD) error {
serverInit.Do(runtime_pollServerInit)
ctx, errno := runtime_pollOpen(uintptr(fd.Sysfd))
if errno != 0 {
return errnoErr(syscall.Errno(errno))
}
pd.runtimeCtx = ctx
return nil
}


責任編輯:姜華 來源: 今日頭條
相關推薦

2023-03-06 08:37:58

JavaNIO

2023-03-31 07:49:51

syscall庫Echo Serve

2023-03-09 08:22:57

Go net庫Socket

2025-06-20 09:57:42

2011-03-31 10:41:49

BIONIOIO

2023-04-03 08:02:16

切片擴容GO

2024-08-20 16:27:54

2025-09-11 02:00:00

2014-11-10 10:52:33

Go語言

2023-11-29 08:19:45

Go泛型缺陷

2020-04-16 15:20:43

PHP前端BIO

2022-04-16 16:52:24

Netty網絡服務器客戶端程序

2023-04-03 06:53:04

Go開發架構

2021-11-02 11:31:47

Go代碼模式

2023-09-26 01:21:34

2023-08-07 08:01:15

2020-08-05 12:27:18

Go語言碼農

2025-04-02 05:23:00

GoChannel數據

2023-08-10 08:00:42

令牌限流器計數器

2018-09-19 14:53:02

NIOBIO運行
點贊
收藏

51CTO技術棧公眾號

国产精品丝袜久久久久久app| 9国产精品视频| 欧美一区二区三区视频免费| av在线免费观看国产| 日韩资源在线| 黑人巨大精品欧美一区| 午夜精品一区二区三区在线播放| 男女黄床上色视频| gogo大尺度成人免费视频| 亚洲一二三区在线观看| 欧美激情论坛| 国产黄a三级三级三级| 老鸭窝毛片一区二区三区 | 国内外免费激情视频| 自拍视频在线网| 成人99免费视频| 国产有码一区二区| 中文字幕第四页| 中文字幕日韩一区二区不卡| 亚洲人成五月天| 国产91在线免费观看| 裤袜国产欧美精品一区| 一区二区三区中文在线观看| 日韩啊v在线| 秋霞网一区二区| 精品一区二区三区在线播放视频 | 国产精品一区二区男女羞羞无遮挡| 久久久久久亚洲精品中文字幕| 欧美成人短视频| 亚州av日韩av| 亚洲高清av在线| www.久久com| 国产一区二区色噜噜| 狠狠躁夜夜躁人人躁婷婷91 | 91高清在线观看视频| 国产欧美日韩在线| 麻豆久久久av免费| 无码国产精品高潮久久99| 国产剧情在线观看一区二区| 国产精品视频yy9099| 香蕉影院在线观看| 亚洲专区一区二区三区| 久久久久亚洲精品| 久久精品www| 综合五月婷婷| 久久久999精品免费| 亚洲不卡的av| 围产精品久久久久久久| 色综合伊人色综合网| 极品尤物一区二区| 成人在线电影在线观看视频| 一本久久综合亚洲鲁鲁| 久久久久久亚洲中文字幕无码| 欧美电影完整版在线观看| 亚洲国产精品美女| 日韩无码精品一区二区| 卡通动漫国产精品| 亚洲激情在线观看| 极品粉嫩小仙女高潮喷水久久| 国产精品sss在线观看av| 精品国产亚洲在线| 大尺度做爰床戏呻吟舒畅| 国偷自产视频一区二区久| 亚洲第一福利在线观看| 国产精品无码一区二区三区免费| 欧美丝袜美腿| 国产婷婷成人久久av免费高清| 泷泽萝拉在线播放| 第一sis亚洲原创| 日韩一区二区欧美| 成人免费精品动漫网站| 欧美精品偷拍| 97超级碰碰碰久久久| 欧美 日韩 精品| 免费精品视频在线| 成人激情视频在线播放| 亚洲AV无码精品自拍| 成人av资源站| 日韩中文字幕av在线| 欧美jizzhd欧美| 亚洲一级二级三级| 亚洲熟女乱色一区二区三区| 日韩中文视频| 日韩欧美中文字幕制服| 中文字幕 亚洲一区| 欧美日韩水蜜桃| 欧美精品中文字幕一区| 国产午夜久久久| 久久综合九色| 国产日韩在线观看av| 丁香六月色婷婷| 国产目拍亚洲精品99久久精品| 亚洲国产一二三精品无码| 色多多在线观看| 777色狠狠一区二区三区| 中文字幕一区二区人妻电影丶| 国产免费av一区二区三区| 伦理中文字幕亚洲| 天天操夜夜操视频| 国产精品一卡二卡在线观看| 久久久久久久久久久一区| 欧美激情办公室videoshd| 天天影视涩香欲综合网| 天天干天天玩天天操| 国产丝袜一区| 久久久99久久精品女同性| 中文字幕激情小说| 国产一区三区三区| 日韩福利视频| 美女搞黄视频在线观看| 欧美一区二区免费视频| 天天躁日日躁aaaa视频| 欧美激情视频一区二区三区在线播放| 奇门遁甲1982国语版免费观看高清| 国产精品久久久久久免费免熟| 97久久超碰精品国产| 激情视频小说图片| 国产成人亚洲一区二区三区| 精品在线小视频| 国产一级一片免费播放| 麻豆极品一区二区三区| 美女黄毛**国产精品啪啪| 久久av色综合| 欧美日韩国产高清一区二区三区| 狠狠人妻久久久久久综合蜜桃| 欧美精品日本| 亚洲影院色在线观看免费| 香蕉视频在线看| 欧美自拍偷拍一区| 三级网站在线免费观看| 一区二区日本视频| 国产成人一区二区三区免费看| 嫩草在线视频| 欧美色图在线观看| a级在线免费观看| 麻豆亚洲精品| 欧美一区1区三区3区公司| 成人影院在线视频| 精品av综合导航| 久久久久99精品| 高清久久久久久| 黄色特一级视频| 欧美a级大片在线| 久久99久久久久久久噜噜| 国产精品女同一区二区| 国产精品免费人成网站| www.com黄色片| 色琪琪久久se色| 成人性生交大片免费看视频直播| 91在线看黄| 欧美乱妇一区二区三区不卡视频| 黄色国产在线播放| 久久se精品一区精品二区| 伊人狠狠色丁香综合尤物| 亚洲欧美专区| 欧美大尺度激情区在线播放| 成人乱码一区二区三区| 亚洲国产精品一区二区尤物区| 在线观看一区二区三区四区| 亚洲狠狠婷婷| 久久综合九色欧美狠狠| 欧美国产大片| 日韩视频中文字幕| 99久久婷婷国产一区二区三区| 伊人婷婷欧美激情| 国产视频精品视频| 亚洲欧美日韩专区| 欧洲亚洲一区二区| 欧美啪啪网站| 欧美日韩国产成人在线| 婷婷五月综合激情| 在线视频亚洲一区| 男女做暖暖视频| hitomi一区二区三区精品| 成年人视频网站免费观看| 国产麻豆一区二区三区精品视频| 国产精品永久免费视频| av理论在线观看| 亚洲黄色av女优在线观看| 日日夜夜操视频| 亚洲品质自拍视频网站| 四虎精品一区二区| 日日夜夜精品免费视频| 99精品视频网站| 国产精品115| 国产激情久久久久| 国产剧情在线| 国产午夜精品久久久| 国产又粗又猛视频免费| 亚洲国产精品欧美一二99| 亚洲久久久久久久| 国产精品88888| 爆乳熟妇一区二区三区霸乳| 中文在线日韩| 热re99久久精品国99热蜜月| 99久久999| 欧美一级淫片aaaaaaa视频| 麻豆tv在线| 精品亚洲男同gayvideo网站 | 秋霞伦理一区| 麻豆国产精品va在线观看不卡 | 337p粉嫩大胆噜噜噜噜噜91av| www.涩涩涩| 中国女人久久久| 国产成人精品免费看在线播放| 美女少妇全过程你懂的久久| 亚洲影院在线看| 丁香婷婷久久| 青草青草久热精品视频在线网站 | 国产裸体视频网站| 日本中文在线一区| 怡红院av亚洲一区二区三区h| 国产精品久久久久无码av| 久久久久久国产精品mv| 亚洲国产精品免费视频| 国产精品日韩欧美大师| 一二三四视频在线中文| 色综合久综合久久综合久鬼88| www.国产精品.com| 亚洲男人第一av网站| 亚洲老妇色熟女老太| 4438x亚洲最大成人网| 波多野结衣黄色| 欧美日韩免费在线| 国产一二三四在线| 最近日韩中文字幕| 中国美女黄色一级片| 国产欧美一区二区精品性| 欧美做受喷浆在线观看| 成人91在线观看| 美女伦理水蜜桃4| 国产成人亚洲综合a∨婷婷| 九九精品久久久| 看国产成人h片视频| 成人黄色一区二区| 久久精品一区| 99免费视频观看| 石原莉奈一区二区三区在线观看| 国产深夜男女无套内射| 亚洲一级黄色| 国产成人在线小视频| 欧美日韩综合| www.好吊操| 中文日韩在线| 日本一区二区黄色| 久久成人亚洲| 久久综合久久色| 日韩在线a电影| wwwwxxxx日韩| 美女精品一区二区| 色乱码一区二区三区在线| 日韩va亚洲va欧美va久久| av观看免费在线| 日产国产高清一区二区三区| 亚洲国产日韩欧美在线观看| 精品在线观看免费| 特种兵之深入敌后| 成人毛片在线观看| 日本xxxx裸体xxxx| 国产无人区一区二区三区| 亚洲综合欧美综合| 中文字幕中文在线不卡住| 日韩精品一区二区三区在线视频| 亚洲少妇屁股交4| 妺妺窝人体色www聚色窝仙踪| 香蕉成人伊视频在线观看| 91视频免费网址| 欧美天天综合网| 国产伦精品一区二区三区视频痴汉| 欧美一区二区三区四区高清| 丰满人妻一区二区三区无码av| 亚洲娇小xxxx欧美娇小| 日本一区高清| 日韩一区av在线| 久草成色在线| 国产精品电影网站| 日本免费一区二区视频| 精品国产aⅴ麻豆| 北条麻妃国产九九九精品小说| 亚洲免费av网| 国产欧美日韩一级| 午夜精品久久久久久久99热影院| 国产精品88888| 一区二区精品免费| 亚洲免费观看高清完整版在线 | 亚洲国产精品免费| 成人影视在线播放| 欧美日韩第一视频| av有声小说一区二区三区| 成人激情视频免费在线| 日韩精选在线| 麻豆md0077饥渴少妇| 国产精品一页| 国模大尺度视频| 国产日产精品一区| 久久久久久久久久久97| 在线亚洲一区观看| 丁香六月色婷婷| 精品国模在线视频| 一根才成人网| 成人在线观看91| 色综合咪咪久久网| 日韩av资源在线| 国产91丝袜在线播放| 国产wwwwxxxx| 欧美性xxxx极品hd满灌| 精品国产无码一区二区| 一本色道久久综合狠狠躁篇的优点| 日本三级在线观看网站| 国产精品一区二区三区久久| 欧美男男freegayvideosroom| 国产一区一区三区| 日韩成人一级片| 国产制服丝袜在线| 亚洲国产成人av网| 国产欧美一级片| 日韩中文字幕久久| 欧美性片在线观看| 欧美高清视频一区二区三区在线观看 | 日韩三级在线免费观看| 午夜在线免费观看视频| 日韩av免费看| 日韩mv欧美mv国产网站| 2019日韩中文字幕mv| 国产精品77777| 欧美激情精品久久久久久免费 | 国产精品久久久久久久久晋中 | 先锋a资源在线看亚洲| 亚洲精品乱码久久久久久蜜桃欧美| 中文字幕中文字幕一区| 中文字幕欧美人妻精品一区蜜臀| 精品亚洲永久免费精品| 黄视频免费在线看| 久久99精品国产99久久| 99综合在线| 免费在线观看成年人视频| 亚洲a一区二区| 天堂av在线免费| 97超级碰碰人国产在线观看| 久久精品色综合| heyzo国产| 91麻豆蜜桃一区二区三区| 国产精品一区二区三区四| 日韩电影在线观看永久视频免费网站| av在线网页| 精品免费国产| 国产视频一区在线观看一区免费| 一级特黄a大片免费| 精品国产乱码久久久久久婷婷| 天天操天天干天天干| 91chinesevideo永久地址| 欧美理论电影在线精品| 国产淫片av片久久久久久| 久久精品网站免费观看| 亚洲色成人www永久网站| 一区二区三区亚洲| 999精品嫩草久久久久久99| 欧美另类videosbestsex日本| 成人深夜在线观看| 久草视频在线观| 国产亚洲欧洲黄色| **国产精品| 国产成人永久免费视频| 91在线国产观看| 国产裸体美女永久免费无遮挡| 精品久久久av| 日韩最新av| 黄色片久久久久| 中文字幕一区二区三区蜜月| 精品黑人一区二区三区国语馆| 久久久久久久久综合| 中文有码一区| 伊人精品视频在线观看| 亚洲va欧美va人人爽| 男操女在线观看| 91在线视频导航| 国产精品普通话对白| 国产黄色录像视频| 日韩午夜小视频| 伊人色综合一区二区三区影院视频| 色狠狠久久av五月综合| 国产馆精品极品| 久草视频一区二区| 欧美老少配视频| 亚洲系列另类av| gogo亚洲国模私拍人体| 色综合天天性综合| 午夜小视频在线观看| 欧美连裤袜在线视频| 国内精品久久久久影院色| 日本道在线观看| 久久精品电影网| 亚洲人成精品久久久 | 超碰在线观看91| 久久在精品线影院精品国产| 色老板在线视频一区二区| √天堂资源在线| 在线中文字幕一区| 免费不卡av| 中文字幕第一页亚洲|