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

Go 協(xié)程鎖機制的實現(xiàn)

開發(fā) 前端
鎖的實現(xiàn)方式通常是首先通過 CAS 嘗試獲得鎖,如果成功則直接返回,如果獲取失敗可以選擇自旋或者把當前的實體加入鎖的等待隊列并把當前實體改成阻塞狀態(tài),然后觸發(fā)重新調(diào)度,等待鎖的持有者解鎖然后喚醒等待者。

在日常的開發(fā)中,我們經(jīng)常會通過高并發(fā)來提高系統(tǒng)的處理能力,高并發(fā)帶來的一個問題就是對公共資源的訪問,這時候必須使用一些互斥機制保證數(shù)據(jù)的正確性。比如數(shù)據(jù)庫的鎖機制,或者我們基于 Redis、Zookeeper 等中間件實現(xiàn)的分布式鎖機制,線程/進程間的鎖機制等,而在 Go 中,我們還會看到有協(xié)程的鎖機制。本文主要分析 Go 中協(xié)程鎖機制的實現(xiàn)。

鎖的實現(xiàn)方式通常是首先通過 CAS 嘗試獲得鎖,如果成功則直接返回,如果獲取失敗可以選擇自旋或者把當前的實體加入鎖的等待隊列并把當前實體改成阻塞狀態(tài),然后觸發(fā)重新調(diào)度,等待鎖的持有者解鎖然后喚醒等待者。但是比較有意思的是"把當前的實體加入鎖的等待隊列"這個問題,比如有多個線程需要操作公共資源 A,當 A 被某個線程加鎖成功后,其余的加鎖失敗的多個線程都需要加入鎖 A 的等待隊列 Q1,這里也涉及到公共資源的訪問,所以對 Q1 的訪問也會執(zhí)行 CAS 加鎖,失敗后加入等待隊列 Q2,形成了套娃。后面我們會看到這個套娃是如何解決的。

協(xié)程間的鎖機制

在 Go 中,我們可以通過 Mutex 來實現(xiàn)多個協(xié)程對公共資源的訪問,它的使用方式很簡單,但是實現(xiàn)相對來說復雜很多,因為 Go 做了很多性能優(yōu)化,比如自旋,解鎖時喚醒的公平性。下面看一個例子。

package main

import (
 "sync"
)

func main() {
 var m sync.Mutex
 m.Lock()
  // 訪問公共資源
  m.Unlock()
}

鎖機制的 API 通常不會很復雜,一般就是 lock/unlock,多一點的就是 trylock 或帶 timeout 的 lock。接著看一下這簡單使用方式的背后 Mutex 是如何實現(xiàn)的。Mutex 的定義如下。

type Mutex struct {
 mu isync.Mutex
}

func (m *Mutex) Lock() {
 m.mu.Lock()
}

可以看到,Mutex 是對 isync.Mutex 的封裝,接著看 isync.Mutex Lock 的實現(xiàn)。

type Mutex struct {
 state int32
 sema  uint32
}

func (m *Mutex) Lock() {
  // 獲取鎖,成功就直接返回
 if atomic.CompareAndSwapInt32(&m.state, 0, mutexLocked) {
  return
 }
 m.lockSlow()
}

Lock 首先用 CAS 獲得鎖,成功則直接返回,失敗時,Go 在獲得鎖失敗時會做一系列的優(yōu)化處理,我們只關(guān)注阻塞協(xié)程的部分。

func (m *Mutex) lockSlow() {
 iter := 0
 old := m.state
for {
    // 鎖被其他協(xié)程持有 & 不是饑餓模式 & 可以自旋
if old&(mutexLocked|mutexStarving) == mutexLocked && runtime_canSpin(iter) {
   // 自旋
   runtime_doSpin()
   iter++
   old = m.state
   continue
  }
    // 嘗試獲得鎖
if atomic.CompareAndSwapInt32(&m.state, old, new) {
      // 成功
   if old&(mutexLocked|mutexStarving) == 0 {
    break// locked the mutex with CAS
   }
      // 獲取失敗則阻塞
   runtime_SemacquireMutex(&m.sema, queueLifo, 2)
      iter = 0
  }
 }
}

Go 在獲得鎖失敗時會先嘗試自旋,如果還是失敗則調(diào)用 runtime_SemacquireMutex(&m.sema, queueLifo, 2) 把協(xié)程加入等待隊列并阻塞協(xié)程。runtime_SemacquireMutex 對應(yīng)函數(shù)如下。

func semacquire1(addr *uint32, lifo bool, profile semaProfileFlags, skipframes int, reason waitReason) {
 gp := getg()

// 獲取一個 sudog
 s := acquireSudog()
// semtable 是一個數(shù)組,每個元素對應(yīng)一棵樹,根據(jù) addr 計算哈希出所屬的數(shù)組索引,并拿到該索引對應(yīng)的樹的根節(jié)點
 root := semtable.rootFor(addr)

// 對樹進行加鎖,因為需要把 sodog 插入樹中
  lockWithRank(&root.lock, lockRankRoot)
// 把 sudog 插入樹中,sudog 記錄了 addr 和當前的 g
  root.queue(addr, s, lifo)
// 把協(xié)程改成阻塞狀態(tài),調(diào)度其他協(xié)程執(zhí)行
  goparkunlock(&root.lock, reason, traceBlockSync, 4+skipframes)
}

semacquire1 的邏輯是首先獲得一個樹的鎖,然后把當前 addr 和 g 封裝成 sudog 插入樹中,最終把協(xié)程改成阻塞狀態(tài),重新觸發(fā)調(diào)度。鎖的持有者解鎖后會喚醒鎖的等待者。

func semrelease1(addr *uint32, handoff bool, skipframes int) {
 // 根據(jù)地址找到樹的根節(jié)點
  root := semtable.rootFor(addr)
  // 加鎖操作樹
 lockWithRank(&root.lock, lockRankRoot)
  // 從樹中獲取 addr 對應(yīng)的 sudog
 s, t0, tailtime := root.dequeue(addr)
 unlock(&root.lock)
  // 修改協(xié)程為 ready 狀態(tài),等待調(diào)度執(zhí)行
  readyWithTime(s, 5+skipframes)
}

通過前面的分析可以看到 Go Mutex 的實現(xiàn)并不是直接使用多線程的鎖機制,而是 Go 自己實現(xiàn)的,因為直接使用多線程的鎖會導致一個協(xié)程阻塞引起整個線程阻塞。Go Mutex 的實現(xiàn)原理大致是首先通過 CAS 獲取鎖,成功則返回,失敗則獲取一個全局數(shù)據(jù)結(jié)構(gòu)(樹)的鎖,把當前 g 插入樹中等待喚醒。那么這個全局數(shù)據(jù)結(jié)構(gòu)的鎖是如何獲取的呢?這就涉及到線程間的鎖機制了。

線程間的鎖機制

剛才加鎖全局數(shù)據(jù)結(jié)構(gòu)的函數(shù)是 lockWithRank。

func lockWithRank(l *mutex, rank lockRank) {
 lock2(l)
}

func lock2(l *mutex) {
 gp := getg()

 k8 := key8(&l.key)
// 直接加鎖成功
 v8 := atomic.Xchg8(k8, mutexLocked)
if v8&mutexLocked == 0 {
if v8&mutexSleeping != 0 {
   atomic.Or8(k8, mutexSleeping)
  }
return
 }
// 創(chuàng)建一個線程級的加鎖數(shù)據(jù)結(jié)構(gòu),比如線程互斥結(jié)構(gòu)體
 semacreate(gp.m)

 v := atomic.Loaduintptr(&l.key)
tryAcquire:
for i := 0; ; i++ {
    // 判斷加鎖可以加鎖并且加鎖成功
if v&mutexLocked == 0 {
   prev8 := atomic.Xchg8(k8, mutexLocked|mutexSleeping)
      if prev8&mutexLocked == 0 {
        timer.end()
        return
      }
   v = atomic.Loaduintptr(&l.key)
   continue tryAcquire
  }
    // 阻塞
  semasleep(-1)
  v = atomic.Loaduintptr(&l.key)
 }
}

可以看到 lockWithRank 最終類似協(xié)程加鎖的流程,先嘗試加鎖,成功則返回,失敗則進入阻塞流程,但是這個阻塞流程和協(xié)程的阻塞流程是不一樣的。看一下 semasleep 的實現(xiàn),semasleep 在不同系統(tǒng)下實現(xiàn)不一樣,比如 MacOS 下是:

func semasleep(ns int64) int32 {
 g := getg()
 mp := g.m
 pthread_mutex_lock(&mp.mutex)
for {
    // 已經(jīng)解鎖了,則返回
if mp.count > 0 {
   mp.count--
   pthread_mutex_unlock(&mp.mutex)
   return0
  }
    // 阻塞等待條件變量的喚醒
  pthread_cond_wait(&mp.cond, &mp.mutex)
 }
}

Linux 下是:

func semasleep(ns int64) int32 {
 mp := getg().m

for v := atomic.Xadd(&mp.waitsema, -1); ; v = atomic.Load(&mp.waitsema) {
ifint32(v) >= 0 {
   return0
  }
  futexsleep(&mp.waitsema, v, ns)
if ns >= 0 {
   ifint32(v) >= 0 {
    return0
   } else {
    return-1
   }
  }
 }
}

func futexsleep(addr *uint32, val uint32, ns int64) {
if ns < 0 {
  futex(unsafe.Pointer(addr), _FUTEX_WAIT_PRIVATE, val, nil, nil, 0)
return
 }

var ts timespec
 ts.setNsec(ns)
 futex(unsafe.Pointer(addr), _FUTEX_WAIT_PRIVATE, val, unsafe.Pointer(&ts), nil, 0)
}

MacOS 是通過 C 庫函數(shù)實現(xiàn),Linux 下則是通過系統(tǒng)調(diào)用實現(xiàn)的。那么 C 庫和 Futex 是如何實現(xiàn)線程的鎖機制的呢?它們的實現(xiàn)有什么區(qū)別?

暫時沒有找到 MacOS 下的 C 庫實現(xiàn),下面是早期 linuxthreads 庫實現(xiàn)的線程鎖機制。

int __pthread_mutex_lock(pthread_mutex_t * mutex)
{
pthread_t self;

while(1) {
    // 加鎖, acquire 實現(xiàn)是 while (testandset(spinlock)) sched_yield();
    acquire(&mutex->m_spinlock);
    switch(mutex->m_kind) {
    case PTHREAD_MUTEX_FAST_NP:
      // 如果還沒有加鎖則加鎖直接返回
      if (mutex->m_count == 0) {
        mutex->m_count = 1;
        release(&mutex->m_spinlock);
        return0;
      }
      self = thread_self();
      break;
    default:
      return EINVAL;
    }
    // 獲取失敗,需要阻塞,把當前線程插入該互斥鎖的等待隊列
    enqueue(&mutex->m_waiting, self);
    release(&mutex->m_spinlock);
    // 掛起等待喚醒
    suspend(self);
  }
}

linuxthreads 的實現(xiàn)是首先通過自旋鎖獲取一個鎖,這個鎖是用于互斥訪問 mutex 數(shù)據(jù)結(jié)構(gòu),對 mutex 加鎖成功則返回,失敗則把當前線程加入 mutex 的等待隊列,然后再通過 suspend 阻塞在 SIGUSR1 信號,借助操作系統(tǒng)掛起當前線程,然后解鎖時發(fā)送 SIGUSR1 信號喚醒等待線程。

int __pthread_mutex_unlock(pthread_mutex_t * mutex)
{
pthread_t th;
// 獲取 mutex 的鎖
  acquire(&mutex->m_spinlock);
switch (mutex->m_kind) {
case PTHREAD_MUTEX_FAST_NP:
    mutex->m_count = 0;
    break;
default:
    return EINVAL;
  }
// 取出一個被阻塞的線程
  th = dequeue(&mutex->m_waiting);
  release(&mutex->m_spinlock);
// 發(fā)送信號喚醒它
if (th != NULL) restart(th);
return0;
}

可以看到 linuxthreads 庫是通過自旋鎖解決了套娃的問題。而 glibc 的線程鎖機制則是借助 futex 實現(xiàn)的。

int pthread_mutex_lock (pthread_mutex_t *mutex)
{
// 獲取鎖類型
unsignedint type = PTHREAD_MUTEX_TYPE_ELISION (mutex);
// 普通鎖
if (__glibc_likely (type == PTHREAD_MUTEX_TIMED_NP))
  {
    LLL_MUTEX_LOCK_OPTIMIZED (mutex);
  }
// 加鎖成功,記錄鎖的持有者
pid_t id = THREAD_GETMEM (THREAD_SELF, tid);
  mutex->__data.__owner = id;
return0;
}

pthread_mutex_lock 的核心邏輯是通過 LLL_MUTEX_LOCK_OPTIMIZED 實現(xiàn)的。

# define LLL_MUTEX_LOCK_OPTIMIZED(mutex) lll_mutex_lock_optimized (mutex)

static inline void lll_mutex_lock_optimized (pthread_mutex_t *mutex)
{
intprivate = PTHREAD_MUTEX_PSHARED (mutex);
  lll_lock (mutex->__data.__lock, private);
}

#define lll_lock(futex, private) \
  __lll_lock (&(futex), private)

#define __lll_lock(futex, private)                                      \
  ((void)                                                               \
   ({                                                                   \
     int *__futex = (futex);                                            \
     if (__glibc_unlikely                                               \
         (atomic_compare_and_exchange_bool_acq (__futex, 1, 0)))        \
       {                                                                \
         if (__builtin_constant_p (private) && (private) == LLL_PRIVATE) \
           __lll_lock_wait_private (__futex);                           \
         else                                                           \
           __lll_lock_wait (__futex, private);                          \
       }                                                                \
   }))
   
void __lll_lock_wait (int *futex, intprivate)
{
if (atomic_load_relaxed (futex) == 2)
    goto futex;

while (atomic_exchange_acquire (futex, 2) != 0)
    {
    futex:
      futex_wait ((unsignedint *) futex, 2, private); /* Wait if *futex == 2.  */
    }
}

static inline int
futex_wait (int *futexp, int val)
{
#ifdef __NR_futex
return syscall (__NR_futex, futexp, FUTEX_WAIT, val);
#else
return syscall (__NR_futex_time64, futexp, FUTEX_WAIT, val);
#endif
}

pthread_mutex_lock 通過原子操作進行加鎖,成功則返回,失敗則通過 futex 實現(xiàn)阻塞和喚醒。

Futex

Futex 是通過系統(tǒng)實現(xiàn)線程阻塞喚醒的一種方式,基于 futex 實現(xiàn)的鎖邏輯為首先通過 CAS 加鎖,成功后直接返回,失敗則通過 futex 陷入系統(tǒng)把當前線程改成阻塞狀態(tài),然后鎖的持有者解鎖時再通過 futex 喚醒等待者。2.6.11 版本的 Futex 實現(xiàn)如下。

static int futex_wait(unsigned long uaddr, int val, unsigned long time)
{
// 等待隊列的節(jié)點
wait_queue_t wait = {    
    .task  = current, // 當前線程     
    .func  = default_wake_function,   
    .task_list = { NULL, NULL } 
  }
int ret, curval;
struct futex_q q;

 retry:
// 加鎖內(nèi)存映射區(qū)域
 down_read(¤t->mm->mmap_sem);
// 根據(jù) uaddr 計算 key
 ret = get_futex_key(uaddr, &q.key);
// 插入隊列
 queue_me(&q, -1, NULL);

 up_read(¤t->mm->mmap_sem);

// 把當前線程改成阻塞狀態(tài)
 __set_current_state(TASK_INTERRUPTIBLE);
// 把當前線程插入 futex 結(jié)構(gòu)體的等待隊列
 add_wait_queue(&q.waiters, &wait);

if (likely(!list_empty(&q.list)))
  time = schedule_timeout(time);
 __set_current_state(TASK_RUNNING);
}

futex_wait 首先把 futex 結(jié)構(gòu)體插入到隊列中,然后把當前線程插入到 futex 結(jié)構(gòu)體的隊列中,最后把當前線程改成阻塞狀態(tài)并重新進行調(diào)度。那么把 futex 結(jié)構(gòu)體插入到隊列中又必然會涉及到并發(fā)訪問公共資源的問題,看看 queue_me 是如何解決的。

static void queue_me(struct futex_q *q, int fd, struct file *filp)
{
	struct futex_hash_bucket *bh;

	q->fd = fd;
	q->filp = filp;

	init_waitqueue_head(&q->waiters);

	get_key_refs(&q->key);
	bh = hash_futex(&q->key);
	q->lock_ptr = &bh->lock;

	spin_lock(&bh->lock);
	bh->nqueued++;
	list_add_tail(&q->list, &bh->chain);
	spin_unlock(&bh->lock);
}

queue_me 先通過 key 從一個全局的結(jié)構(gòu)體數(shù)組中計算出對應(yīng)的索引,這個索引對應(yīng)的元素是一條鏈表(減少并發(fā)時的競爭),然后加鎖并把 futex 結(jié)構(gòu)體插入該鏈表中。這里使用的是自旋鎖,那么這里的鎖又是怎么加的呢?

#define spin_lock(lock)  _spin_lock(lock)

#define _spin_lock(lock) \
do { \
 preempt_disable(); \ // 關(guān)系統(tǒng)搶占
 _raw_spin_lock(lock); \
 __acquire(lock); \
} while(0)

_raw_spin_lock 在不同的架構(gòu)中實現(xiàn)不一樣。比如非 SMP 架構(gòu)下因為關(guān)閉了系統(tǒng)搶占,所以 _raw_spin_lock 的實現(xiàn)實際上什么都不需要做,因為不會發(fā)生并發(fā)問題。

#define _raw_spin_lock(lock)	do { (void)(lock); } while(0)

而 SMP 架構(gòu)下,可能存在多個 CPU 訪問該數(shù)據(jù)結(jié)構(gòu),所以需要真正的加鎖。

static inline void _raw_spin_lock(spinlock_t *lock)
{
 __asm__ __volatile__(
  spin_lock_string
  :"=m" (lock->slock) : : "memory");
}

#define spin_lock_string \
"\n1:\t" \
"lock ; decb %0\n\t" \
"jns 3f\n" \
"2:\t" \
"rep;nop\n\t" \
"cmpb $0,%0\n\t" \
"jle 2b\n\t" \
"jmp 1b\n" \
 "3:\n\t"

以上代碼大概是通過原子操作實現(xiàn)加鎖。

通過分析可以看到,在 Go 中,協(xié)程鎖機制的實現(xiàn)是由 Go 自己實現(xiàn)的,但是在實現(xiàn)中需要借助線程的鎖機制來實現(xiàn)協(xié)程的阻塞等待和喚醒,而線程的鎖機制又需要通過操作系統(tǒng)的 futex 來實現(xiàn)線程的阻塞等待和喚醒,而系統(tǒng)的等待和喚醒機制同樣需要通過鎖機制來實現(xiàn),具體實現(xiàn)在不同架構(gòu)下不一樣,比如在非 SMP 架構(gòu)中,只需要關(guān)系統(tǒng)搶占即可,這樣在執(zhí)行操作系統(tǒng)代碼時就不會有并發(fā)代碼訪問該公共數(shù)據(jù)結(jié)構(gòu)(需要保證中斷中也不會訪問該數(shù)據(jù)結(jié)構(gòu),否則也需要關(guān)中斷),而在 SMP 架構(gòu)下是通過自旋鎖來實現(xiàn)的,這樣就解決了套娃的問題。

責任編輯:姜華 來源: 編程雜技
相關(guān)推薦

2024-12-03 15:15:22

2016-10-28 17:39:47

phpgolangcoroutine

2013-12-12 16:44:25

Lua協(xié)程

2023-11-23 08:31:51

競爭鎖共享字段

2024-02-05 09:06:25

Python協(xié)程Asyncio庫

2024-06-27 07:56:49

2018-12-04 14:00:41

協(xié)程編程模式PHP

2021-04-25 09:36:20

Go協(xié)程線程

2021-09-16 09:59:13

PythonJavaScript代碼

2023-07-27 13:46:10

go開源項目

2017-05-02 11:38:00

PHP協(xié)程實現(xiàn)過程

2024-08-27 09:46:39

Go協(xié)程效率

2023-11-17 11:36:59

協(xié)程纖程操作系統(tǒng)

2023-07-13 08:06:05

應(yīng)用協(xié)程阻塞

2025-02-28 09:04:08

2024-05-29 08:05:15

Go協(xié)程通信

2021-09-27 23:28:29

Go多協(xié)程并發(fā)

2023-04-19 21:20:49

Tars-Cpp協(xié)程

2022-10-28 10:45:22

Go協(xié)程GoFrame

2021-05-21 08:21:57

Go語言基礎(chǔ)技術(shù)
點贊
收藏

51CTO技術(shù)棧公眾號

日韩资源av在线| 国产成人精品久久| 中文字幕第10页| 色老头在线观看| 97久久精品人人做人人爽50路| 91av在线播放视频| 五月天婷婷丁香网| 国产精品tv| 欧美三级电影网| 国产视频在线观看网站| 每日更新在线观看av| 黑人精品欧美一区二区蜜桃| 97在线视频精品| 激情五月激情综合| 老司机在线精品视频| 欧美三级午夜理伦三级中视频| 成人在线视频一区二区三区| 二区三区在线| 成人激情小说乱人伦| 国产精品免费在线免费 | 鲁大师精品99久久久| 欧亚洲嫩模精品一区三区| 国产亚洲精品久久久久久久| 国产黄在线播放| 成人av在线电影| 成人中文字幕在线观看| 日本视频在线观看免费| 国产一区激情| 久久精品在线视频| 国产成人免费观看网站| 女仆av观看一区| 欧美一区二区三区四区在线观看| 亚洲欧美另类动漫| 九色porny自拍视频在线观看| 亚洲视频资源在线| 亚洲国产精品123| 久久这里精品| 久久久久国产一区二区三区四区 | 欧美精品第三页| av蜜臀在线| 亚洲精品少妇30p| 亚洲一区三区| 大片免费播放在线视频| 26uuu精品一区二区 | 一本久久青青| 亚洲精品久久久久中文字幕欢迎你| 三年中文在线观看免费大全中国| 91亚洲视频| 色88888久久久久久影院野外| 免费欧美一级视频| 电影在线观看一区| 亚洲成人777| 久久国产午夜精品理论片最新版本| 爆操欧美美女| 亚洲人一二三区| 一级黄色片播放| 1769免费视频在线观看| 亚洲精品久久久蜜桃| 久久综合亚洲精品| heyzo在线| 午夜精品久久久久| www.爱色av.com| 最近高清中文在线字幕在线观看1| 午夜精品久久久久久久99樱桃| 国产特级黄色大片| 综合毛片免费视频| 欧美在线你懂的| jizzzz日本| 国产精区一区二区| 精品久久免费看| 国产精品久久无码| 欧洲专线二区三区| 最近2019年好看中文字幕视频| 国产在视频线精品视频| 91精品国产乱码久久久久久久| 久久天天躁狠狠躁老女人| 欧美国产日韩综合| 国产一区白浆| 国产欧美一区二区| 亚洲a视频在线观看| 成人h精品动漫一区二区三区| 免费精品视频一区二区三区| 午夜伦全在线观看| 亚洲一区二区三区四区中文字幕 | 日韩成人免费看| 国产精品丝袜一区二区三区| 99国产精品久久久久99打野战| 国产成人精品免费网站| 久久综合福利| 国产在线高清理伦片a| 亚洲午夜激情网站| 久草福利视频在线| 亚洲3区在线| 亚洲人成电影网站色xx| 久艹在线观看视频| 9久re热视频在线精品| 国产精品中文字幕久久久| av资源免费看| 国产清纯白嫩初高生在线观看91| 国产精品88久久久久久妇女 | 国产精品一二三在线| 国产超碰人人模人人爽人人添| 91在线国内视频| 亚洲欧美一二三| 成人性教育av免费网址| 91精品婷婷国产综合久久性色 | 国产精品欧美一区喷水| 成人免费毛片在线观看| 国产精品传媒麻豆hd| 亚洲第一免费播放区| 美女福利视频网| 亚洲永久网站| 成人av资源网| 日本视频在线免费观看| 欧美性xxxxxx| 成人欧美精品一区二区| 成人羞羞视频播放网站| 91av在线不卡| 亚洲第一色视频| 国产精品国产三级国产普通话蜜臀 | 亚洲综合色区另类av| 男人搞女人网站| 三级精品视频| 久久久久中文字幕| 国产巨乳在线观看| 中文一区在线播放| 久草精品在线播放| 日韩有码av| 91国产中文字幕| 欧美特级特黄aaaaaa在线看| 亚洲日本丝袜连裤袜办公室| 亚洲五月天综合| 免费成人av| 欧美一级高清免费| 神马午夜电影一区二区三区在线观看| 一区二区在线观看不卡| 日韩成人精品视频在线观看| 国产永久精品大片wwwapp| 日本精品视频在线| 天堂成人在线| 欧美丝袜一区二区| 亚洲成人日韩在线| 国产精品毛片在线| 麻豆精品传媒视频| 性欧美18xxxhd| 亚洲精品白浆高清久久久久久| 日本熟妇色xxxxx日本免费看| 国产jizzjizz一区二区| 久久久天堂国产精品| 国产日韩欧美中文在线| 欧美成人国产va精品日本一级| 91亚洲欧美激情| 1024成人网色www| 青娱乐国产精品视频| 亚洲女同中文字幕| 99porn视频在线| 黄色在线观看视频网站| 欧美精品一区二区三区四区| 国产一级视频在线播放| www.欧美色图| www.国产区| 日韩精品一区二区三区免费观影| 国产精品视频播放| 国产在线激情视频| 精品久久久久一区| 男人午夜免费视频| 国产视频视频一区| 亚洲午夜激情影院| 亚洲欧美伊人| 久草一区二区| 欧美在线va视频| 日韩视频免费大全中文字幕| 朝桐光av在线一区二区三区| 亚洲成a人v欧美综合天堂| 一区二区三区免费在线观看视频| 日韩精品三区四区| 中文字幕不卡每日更新1区2区| 欧洲大片精品免费永久看nba| 国产69精品久久久久9| 你懂的视频在线免费| 欧美日韩国产综合一区二区三区| 麻豆亚洲av熟女国产一区二| 91麻豆精东视频| 在线能看的av网站| 1024成人| 天堂一区二区三区 | 好吊色欧美一区二区三区视频| 麻豆视频在线观看免费网站黄| 在线观看不卡av| 国产成人av免费看| 色狠狠色狠狠综合| 538任你躁在线精品视频网站| 97se亚洲国产综合自在线| 成年网站免费在线观看| 99热免费精品| 一区二区不卡在线视频 午夜欧美不卡'| 秋霞影院一区| 国产成人精品一区二区三区| 91亚洲天堂| 国产一区二区三区在线| 懂色av一区二区三区四区 | 亚洲AV无码精品国产| 色综合天天综合在线视频| 麻豆网址在线观看| 久久综合九色综合欧美就去吻| 亚洲一二三不卡| 老**午夜毛片一区二区三区| 成人黄色片免费| jvid福利在线一区二区| 精品国产一区二区三区四区vr| 日韩久久99| 国产成人avxxxxx在线看| 羞羞电影在线观看www| 中文在线资源观看视频网站免费不卡| 天天舔天天干天天操| 欧美久久久久久久久久| 国产视频91在线| 一区二区三区在线免费| 亚洲一级理论片| 久久久久久久久久久久久久久99 | 黄在线观看网站| 欧美精品一区二区三区久久久竹菊| 亚洲成色最大综合在线| 免费视频国产一区| 黄色99视频| 成人动漫视频| 97视频中文字幕| 日韩一区中文| 国产精品自在线| 主播大秀视频在线观看一区二区| 欧美在线一级va免费观看| 岛国av免费在线观看| 久久久久久久久久久91| 污污视频在线| 久久91精品国产| 亚洲综合伊人久久大杳蕉| 久久久国产在线视频| 欧美成人hd| 久久精品电影网站| 欧美jizzhd69巨大| 久久久成人av| 黄网页在线观看| 美女av一区二区三区 | 欧美亚洲成人免费| 99riav视频在线观看| 欧美国产激情18| 国产网红在线观看| 欧美日韩国产999| 秋霞在线视频| 久久久亚洲成人| 国产欧洲在线| 国产成人aa精品一区在线播放| 欧美日韩视频网站| 国产精品视频内| 欧美亚洲黄色| 91在线国产电影| 香蕉大人久久国产成人av| 国产一区免费视频| 免费久久久久久久久| 日韩亚洲视频| 五月天激情综合网| 国产乱子伦精品无码专区| 亚洲欧洲日本mm| 久久久久狠狠高潮亚洲精品| 日韩黄色免费电影| 不卡中文字幕在线观看| 国产盗摄一区二区三区| 日韩综合第一页| 久久精品人人做人人爽97| 永久免费观看片现看| 一区二区三区在线高清| 97免费在线观看视频| 欧美综合色免费| 国产同性人妖ts口直男| 日韩成人av网址| 8888四色奇米在线观看| 欧美激情视频一区二区三区不卡| 亚洲天堂av影院| 国产日韩av高清| 国产精品45p| 亚洲人成人77777线观看| 欧美日韩精品免费观看视频完整| 日韩在线视频在线观看| 加勒比av一区二区| 好吊色视频一区二区三区| 国产视频911| 国产一级片免费观看| 欧美吻胸吃奶大尺度电影| www.亚洲天堂.com| 亚洲午夜av电影| 性欧美videos高清hd4k| 国产精品9999| 盗摄牛牛av影视一区二区| 视频一区二区综合| 亚洲区第一页| 免费网站在线观看黄| 久久一二三国产| 黄色一级片中国| 在线看不卡av| 日韩中文字幕免费在线观看| 一区二区三区回区在观看免费视频| 羞羞视频在线观看免费| 国产精品日韩在线观看| 色哟哟精品丝袜一区二区| 日韩视频在线观看视频| 石原莉奈在线亚洲二区| 免费啪视频在线观看| 中文字幕一区二区三区不卡在线 | 国产拍在线视频| 亚洲综合av影视| 大片网站久久| 波多野结衣作品集| 99re6这里只有精品视频在线观看| 日韩成人短视频| 欧美日韩在线播放一区| 四虎成人免费在线| 欧美黄色性视频| 成人综合日日夜夜| 亚洲一区精彩视频| 日韩av一区二区在线影视| 99久久人妻精品免费二区| 一区二区三区四区av| 国产剧情久久久| www日韩中文字幕在线看| 成人影院在线免费观看| 日韩欧美三级一区二区| 久久精品91| 日本丰满少妇裸体自慰| 婷婷中文字幕综合| 国模无码一区二区三区| 欧美人在线视频| 日本免费一区二区三区视频| 成年人免费观看的视频| 美女在线视频一区| 国产aaaaaaaaa| 欧美亚洲国产一区二区三区va | 波多野结衣一区二区三区免费视频| 国产精品h视频| 精品无人码麻豆乱码1区2区 | 91精品中文在线| 欧美韩国日本在线观看| 日本人视频jizz页码69| 日本一区二区三区免费乱视频 | 免费毛片b在线观看| 国产视频一区二区不卡| 亚洲高清电影| 黄色a一级视频| 日本韩国一区二区三区| 二区在线视频| 成人免费淫片aa视频免费| 88国产精品视频一区二区三区| 国产精品久久久久久久av福利| 亚洲人成伊人成综合网小说| 99国产精品久久久久久久成人| 欧美乱大交做爰xxxⅹ性3| 日韩一区网站| 国产精品久久久久9999爆乳| www.日韩av| 国产情侣免费视频| 日韩最新中文字幕电影免费看| 色狠狠一区二区三区| 国产成人生活片| www.欧美日韩国产在线| 欧美成人一区二区三区四区| 中文字幕欧美专区| 欧美高清一级片| 亚洲国产精品无码观看久久| 91视频你懂的| 中文字幕乱伦视频| 欧美精品在线极品| xxxx日韩| 91激情视频在线| 亚洲欧美一区二区三区久本道91 | 久久久久久久久久伊人| 成人一区二区三区在线观看| 韩国av中文字幕| 日韩中文综合网| 9999久久久久| 92看片淫黄大片一级| 国产精品看片你懂得| www.黄色一片| 国产精品久久久久久久av大片| 欧美1区3d| 国产成人精品无码免费看夜聊软件| 精品视频在线视频| 91九色在线看| 中文字幕欧美日韩一区二区| 成人av手机在线观看| 午夜一区二区三区四区| 欧美高清视频在线观看| 国产麻豆精品久久| 日韩精品xxx| 欧美三级中文字幕在线观看| 青青草视频在线免费直播| 欧美亚洲免费高清在线观看| 国产美女精品一区二区三区| 国产综合精品视频| 欧美理论片在线观看| 国产一区二区精品久| 亚洲av综合色区无码另类小说|