色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

基于Linux內(nèi)核源碼的RCU實(shí)現(xiàn)方案

454398 ? 來源: Chinaunix ? 作者:hiyachen ? 2020-09-25 15:10 ? 次閱讀

RCU(Read-Copy Update)是數(shù)據(jù)同步的一種方式,在當(dāng)前的Linux內(nèi)核中發(fā)揮著重要的作用。RCU主要針對(duì)的數(shù)據(jù)對(duì)象是鏈表,目的是提高遍歷讀取數(shù)據(jù)的效率,為了達(dá)到目的使用RCU機(jī)制讀取數(shù)據(jù)的時(shí)候不對(duì)鏈表進(jìn)行耗時(shí)的加鎖操作。這樣在同一時(shí)間可以有多個(gè)線程同時(shí)讀取該鏈表,并且允許一個(gè)線程對(duì)鏈表進(jìn)行修改(修改的時(shí)候,需要加鎖)。RCU適用于需要頻繁的讀取數(shù)據(jù),而相應(yīng)修改數(shù)據(jù)并不多的情景,例如在文件系統(tǒng)中,經(jīng)常需要查找定位目錄,而對(duì)目錄的修改相對(duì)來說并不多,這就是RCU發(fā)揮作用的最佳場(chǎng)景。

Linux內(nèi)核源碼當(dāng)中,關(guān)于RCU的文檔比較齊全,你可以在 /Documentation/RCU/ 目錄下找到這些文件。Paul E. McKenney 是內(nèi)核中RCU源碼的主要實(shí)現(xiàn)者,他也寫了很多RCU方面的文章。他把這些文章和一些關(guān)于RCU的論文的鏈接整理到了一起。http://www2.rdrop.com/users/paulmck/RCU/

在RCU的實(shí)現(xiàn)過程中,我們主要解決以下問題:

1,在讀取過程中,另外一個(gè)線程刪除了一個(gè)節(jié)點(diǎn)。刪除線程可以把這個(gè)節(jié)點(diǎn)從鏈表中移除,但它不能直接銷毀這個(gè)節(jié)點(diǎn),必須等到所有的讀取線程讀取完成以后,才進(jìn)行銷毀操作。RCU中把這個(gè)過程稱為寬限期(Grace period)。

2,在讀取過程中,另外一個(gè)線程插入了一個(gè)新節(jié)點(diǎn),而讀線程讀到了這個(gè)節(jié)點(diǎn),那么需要保證讀到的這個(gè)節(jié)點(diǎn)是完整的。這里涉及到了發(fā)布-訂閱機(jī)制(Publish-Subscribe Mechanism)。

3, 保證讀取鏈表的完整性。新增或者刪除一個(gè)節(jié)點(diǎn),不至于導(dǎo)致遍歷一個(gè)鏈表從中間斷開。但是RCU并不保證一定能讀到新增的節(jié)點(diǎn)或者不讀到要被刪除的節(jié)點(diǎn)。

寬限期

通過例子,方便理解這個(gè)內(nèi)容。以下例子修改于Paul的文章。

[cpp]view plaincopy

structfoo{

inta;

charb;

longc;

};

DEFINE_SPINLOCK(foo_mutex);

structfoo*gbl_foo;

voidfoo_read(void)

{

foo*fp=gbl_foo;

if(fp!=NULL)

dosomething(fp->a,fp->b,fp->c);

}

voidfoo_update(foo*new_fp)

{

spin_lock(&foo_mutex);

foo*old_fp=gbl_foo;

gbl_foo=new_fp;

spin_unlock(&foo_mutex);

kfee(old_fp);

}

如上的程序,是針對(duì)于全局變量gbl_foo的操作。假設(shè)以下場(chǎng)景。有兩個(gè)線程同時(shí)運(yùn)行 foo_ read和foo_update的時(shí)候,當(dāng)foo_ read執(zhí)行完賦值操作后,線程發(fā)生切換;此時(shí)另一個(gè)線程開始執(zhí)行foo_update并執(zhí)行完成。當(dāng)foo_ read運(yùn)行的進(jìn)程切換回來后,運(yùn)行dosomething 的時(shí)候,fp已經(jīng)被刪除,這將對(duì)系統(tǒng)造成危害。為了防止此類事件的發(fā)生,RCU里增加了一個(gè)新的概念叫寬限期(Grace period)。如下圖所示:

圖中每行代表一個(gè)線程,最下面的一行是刪除線程,當(dāng)它執(zhí)行完刪除操作后,線程進(jìn)入了寬限期。寬限期的意義是,在一個(gè)刪除動(dòng)作發(fā)生后,它必須等待所有在寬限期開始前已經(jīng)開始的讀線程結(jié)束,才可以進(jìn)行銷毀操作。這樣做的原因是這些線程有可能讀到了要?jiǎng)h除的元素。圖中的寬限期必須等待1和2結(jié)束;而讀線程5在寬限期開始前已經(jīng)結(jié)束,不需要考慮;而3,4,6也不需要考慮,因?yàn)樵趯捪奁诮Y(jié)束后開始后的線程不可能讀到已刪除的元素。為此RCU機(jī)制提供了相應(yīng)的API來實(shí)現(xiàn)這個(gè)功能。

[cpp]view plaincopy

voidfoo_read(void)

{

rcu_read_lock();

foo*fp=gbl_foo;

if(fp!=NULL)

dosomething(fp->a,fp->b,fp->c);

rcu_read_unlock();

}

voidfoo_update(foo*new_fp)

{

spin_lock(&foo_mutex);

foo*old_fp=gbl_foo;

gbl_foo=new_fp;

spin_unlock(&foo_mutex);

synchronize_rcu();

kfee(old_fp);

}

其中foo_read中增加了rcu_read_lock和rcu_read_unlock,這兩個(gè)函數(shù)用來標(biāo)記一個(gè)RCU讀過程的開始和結(jié)束。其實(shí)作用就是幫助檢測(cè)寬限期是否結(jié)束。foo_update增加了一個(gè)函數(shù)synchronize_rcu(),調(diào)用該函數(shù)意味著一個(gè)寬限期的開始,而直到寬限期結(jié)束,該函數(shù)才會(huì)返回。我們?cè)賹?duì)比著圖看一看,線程1和2,在synchronize_rcu之前可能得到了舊的gbl_foo,也就是foo_update中的old_fp,如果不等它們運(yùn)行結(jié)束,就調(diào)用kfee(old_fp),極有可能造成系統(tǒng)崩潰。而3,4,6在synchronize_rcu之后運(yùn)行,此時(shí)它們已經(jīng)不可能得到old_fp,此次的kfee將不對(duì)它們產(chǎn)生影響。

寬限期是RCU實(shí)現(xiàn)中最復(fù)雜的部分,原因是在提高讀數(shù)據(jù)性能的同時(shí),刪除數(shù)據(jù)的性能也不能太差。

訂閱——發(fā)布機(jī)制

當(dāng)前使用的編譯器大多會(huì)對(duì)代碼做一定程度的優(yōu)化,CPU也會(huì)對(duì)執(zhí)行指令做一些優(yōu)化調(diào)整,目的是提高代碼的執(zhí)行效率,但這樣的優(yōu)化,有時(shí)候會(huì)帶來不期望的結(jié)果。如例:

[cpp]view plaincopy

voidfoo_update(foo*new_fp)

{

spin_lock(&foo_mutex);

foo*old_fp=gbl_foo;

new_fp->a=1;

new_fp->b=‘b’;

new_fp->c=100;

gbl_foo=new_fp;

spin_unlock(&foo_mutex);

synchronize_rcu();

kfee(old_fp);

}

這段代碼中,我們期望的是6,7,8行的代碼在第10行代碼之前執(zhí)行。但優(yōu)化后的代碼并不對(duì)執(zhí)行順序做出保證。在這種情形下,一個(gè)讀線程很可能讀到 new_fp,但new_fp的成員賦值還沒執(zhí)行完成。當(dāng)讀線程執(zhí)行dosomething(fp->a, fp->b , fp->c ) 的 時(shí)候,就有不確定的參數(shù)傳入到dosomething,極有可能造成不期望的結(jié)果,甚至程序崩潰??梢酝ㄟ^優(yōu)化屏障來解決該問題,RCU機(jī)制對(duì)優(yōu)化屏障做了包裝,提供了專用的API來解決該問題。這時(shí)候,第十行不再是直接的指針賦值,而應(yīng)該改為 :

rcu_assign_pointer(gbl_foo,new_fp);

rcu_assign_pointer的實(shí)現(xiàn)比較簡(jiǎn)單,如下:

[cpp]view plaincopy

#definercu_assign_pointer(p,v)

__rcu_assign_pointer((p),(v),__rcu)

#define__rcu_assign_pointer(p,v,space)

do{

smp_wmb();

(p)=(typeof(*v)__forcespace*)(v);

}while(0)

我們可以看到它的實(shí)現(xiàn)只是在賦值之前加了優(yōu)化屏障 smp_wmb來確保代碼的執(zhí)行順序。另外就是宏中用到的__rcu,只是作為編譯過程的檢測(cè)條件來使用的。

在DEC Alpha CPU機(jī)器上還有一種更強(qiáng)悍的優(yōu)化,如下所示:

[cpp]view plaincopy

voidfoo_read(void)

{

rcu_read_lock();

foo*fp=gbl_foo;

if(fp!=NULL)

dosomething(fp->a,fp->b,fp->c);

rcu_read_unlock();

}

第六行的fp->a,fp->b,fp->c會(huì)在第3行還沒執(zhí)行的時(shí)候就預(yù)先判斷運(yùn)行,當(dāng)他和foo_update同時(shí)運(yùn)行的時(shí)候,可能導(dǎo)致傳入dosomething的一部分屬于舊的gbl_foo,而另外的屬于新的。這樣導(dǎo)致運(yùn)行結(jié)果的錯(cuò)誤。為了避免該類問題,RCU還是提供了宏來解決該問題:

[cpp]view plaincopy

#definercu_dereference(p)rcu_dereference_check(p,0)

#definercu_dereference_check(p,c)

__rcu_dereference_check((p),rcu_read_lock_held()||(c),__rcu)

#define__rcu_dereference_check(p,c,space)

({

typeof(*p)*_________p1=(typeof(*p)*__force)ACCESS_ONCE(p);

rcu_lockdep_assert(c,"suspiciousrcu_dereference_check()"

"usage");

rcu_dereference_sparse(p,space);

smp_read_barrier_depends();

((typeof(*p)__force__kernel*)(_________p1));

})

staticinlineintrcu_read_lock_held(void)

{

if(!debug_lockdep_rcu_enabled())

return1;

if(rcu_is_cpu_idle())

return0;

if(!rcu_lockdep_current_cpu_online())

return0;

returnlock_is_held(&rcu_lock_map);

}

這段代碼中加入了調(diào)試信息,去除調(diào)試信息,可以是以下的形式(其實(shí)這也是舊版本中的代碼):

[cpp]view plaincopy

#definercu_dereference(p)({

typeof(p)_________p1=p;

smp_read_barrier_depends();

(_________p1);

})

在賦值后加入優(yōu)化屏障smp_read_barrier_depends()。

我們之前的第四行代碼改為foo *fp = rcu_dereference(gbl_foo);,就可以防止上述問題。

數(shù)據(jù)讀取的完整性

還是通過例子來說明這個(gè)問題:

如圖我們?cè)谠璴ist中加入一個(gè)節(jié)點(diǎn)new到A之前,所要做的第一步是將new的指針指向A節(jié)點(diǎn),第二步才是將Head的指針指向new。這樣做的目的是當(dāng)插入操作完成第一步的時(shí)候,對(duì)于鏈表的讀取并不產(chǎn)生影響,而執(zhí)行完第二步的時(shí)候,讀線程如果讀到new節(jié)點(diǎn),也可以繼續(xù)遍歷鏈表。如果把這個(gè)過程反過來,第一步head指向new,而這時(shí)一個(gè)線程讀到new,由于new的指針指向的是Null,這樣將導(dǎo)致讀線程無法讀取到A,B等后續(xù)節(jié)點(diǎn)。從以上過程中,可以看出RCU并不保證讀線程讀取到new節(jié)點(diǎn)。如果該節(jié)點(diǎn)對(duì)程序產(chǎn)生影響,那么就需要外部調(diào)用做相應(yīng)的調(diào)整。如在文件系統(tǒng)中,通過RCU定位后,如果查找不到相應(yīng)節(jié)點(diǎn),就會(huì)進(jìn)行其它形式的查找,相關(guān)內(nèi)容等分析到文件系統(tǒng)的時(shí)候再進(jìn)行敘述。

我們?cè)倏匆幌聞h除一個(gè)節(jié)點(diǎn)的例子:

如圖我們希望刪除B,這時(shí)候要做的就是將A的指針指向C,保持B的指針,然后刪除程序?qū)⑦M(jìn)入寬限期檢測(cè)。由于B的內(nèi)容并沒有變更,讀到B的線程仍然可以繼續(xù)讀取B的后續(xù)節(jié)點(diǎn)。B不能立即銷毀,它必須等待寬限期結(jié)束后,才能進(jìn)行相應(yīng)銷毀操作。由于A的節(jié)點(diǎn)已經(jīng)指向了C,當(dāng)寬限期開始之后所有的后續(xù)讀操作通過A找到的是C,而B已經(jīng)隱藏了,后續(xù)的讀線程都不會(huì)讀到它。這樣就確保寬限期過后,刪除B并不對(duì)系統(tǒng)造成影響。

小結(jié)

RCU的原理并不復(fù)雜,應(yīng)用也很簡(jiǎn)單。但代碼的實(shí)現(xiàn)確并不是那么容易,難點(diǎn)都集中在了寬限期的檢測(cè)上,后續(xù)分析源代碼的時(shí)候,我們可以看到一些極富技巧的實(shí)現(xiàn)方式。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11310

    瀏覽量

    209601
  • 數(shù)據(jù)同步
    +關(guān)注

    關(guān)注

    0

    文章

    17

    瀏覽量

    8166
  • rcu
    rcu
    +關(guān)注

    關(guān)注

    0

    文章

    21

    瀏覽量

    5455
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    騰訊云內(nèi)核團(tuán)隊(duì)修復(fù)Linux關(guān)鍵Bug

    騰訊云操作系統(tǒng)(Tencent OS)內(nèi)核團(tuán)隊(duì)近日在Linux社區(qū)取得了顯著成果。他們提交的兩項(xiàng)改進(jìn)方案,成功解決了自2021年以來一直困擾眾多一線廠商,并在近期讓多個(gè)Linux頂級(jí)
    的頭像 發(fā)表于 12-31 10:58 ?140次閱讀

    嵌入式學(xué)習(xí)-飛凌嵌入式ElfBoard ELF 1板卡-Linux內(nèi)核移植之內(nèi)核簡(jiǎn)介

    學(xué)到本章節(jié),大家應(yīng)該對(duì)Linux操作系統(tǒng)都有了一定的了解,但可能還不知道我們拿到手的內(nèi)核源碼都經(jīng)歷了什么。linux有一個(gè)龐大的開源社區(qū),每個(gè)人都可以向開源社區(qū)提交代碼。由于
    發(fā)表于 12-16 13:08

    飛凌嵌入式ElfBoard ELF 1板卡-Linux內(nèi)核移植之內(nèi)核簡(jiǎn)介

    學(xué)到本章節(jié),大家應(yīng)該對(duì)Linux操作系統(tǒng)都有了一定的了解,但可能還不知道我們拿到手的內(nèi)核源碼都經(jīng)歷了什么。linux有一個(gè)龐大的開源社區(qū),每個(gè)人都可以向開源社區(qū)提交代碼。由于
    發(fā)表于 12-13 09:03

    deepin社區(qū)亮相第19屆中國(guó)Linux內(nèi)核開發(fā)者大會(huì)

    中國(guó) Linux 內(nèi)核開發(fā)者大會(huì),作為中國(guó) Linux 內(nèi)核領(lǐng)域最具影響力的峰會(huì)之一,一直以來都備受矚目。
    的頭像 發(fā)表于 10-29 16:35 ?515次閱讀

    linux內(nèi)核中通用HID觸摸驅(qū)動(dòng)

    linux內(nèi)核中,為HID觸摸面板實(shí)現(xiàn)了一個(gè)通用的驅(qū)動(dòng)程序,位于/drivers/hid/hid-multitouch.c文件中。hid觸摸驅(qū)動(dòng)是以struct hid_driver實(shí)現(xiàn)
    的頭像 發(fā)表于 10-29 10:55 ?656次閱讀
    <b class='flag-5'>linux</b><b class='flag-5'>內(nèi)核</b>中通用HID觸摸驅(qū)動(dòng)

    詳解linux內(nèi)核的uevent機(jī)制

    linux內(nèi)核中,uevent機(jī)制是一種內(nèi)核和用戶空間通信的機(jī)制,用于通知用戶空間應(yīng)用程序各種硬件更改或其他事件,比如插入或移除硬件設(shè)備(如USB驅(qū)動(dòng)器或網(wǎng)絡(luò)接口)。uevent表示“用戶空間
    的頭像 發(fā)表于 09-29 17:01 ?707次閱讀

    linux驅(qū)動(dòng)程序如何加載進(jìn)內(nèi)核

    Linux系統(tǒng)中,驅(qū)動(dòng)程序是內(nèi)核與硬件設(shè)備之間的橋梁。它們?cè)试S內(nèi)核與硬件設(shè)備進(jìn)行通信,從而實(shí)現(xiàn)對(duì)硬件設(shè)備的控制和管理。 驅(qū)動(dòng)程序的編寫 驅(qū)動(dòng)程序的編寫是
    的頭像 發(fā)表于 08-30 15:02 ?486次閱讀

    Linux內(nèi)核測(cè)試技術(shù)

    。內(nèi)核測(cè)試技術(shù)是實(shí)現(xiàn)這一目標(biāo)的關(guān)鍵手段。本文將詳細(xì)介紹 Linux 內(nèi)核測(cè)試的各種技術(shù),包括單元測(cè)試、集成測(cè)試、功能測(cè)試和性能測(cè)試等,并討論不同測(cè)試方法的優(yōu)缺點(diǎn)及其適用場(chǎng)景。
    的頭像 發(fā)表于 08-13 13:42 ?505次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>測(cè)試技術(shù)

    Linux內(nèi)核中的頁面分配機(jī)制

    Linux內(nèi)核中是如何分配出頁面的,如果我們站在CPU的角度去看這個(gè)問題,CPU能分配出來的頁面是以物理頁面為單位的。也就是我們計(jì)算機(jī)中常講的分頁機(jī)制。本文就看下Linux內(nèi)核是如何管
    的頭像 發(fā)表于 08-07 15:51 ?296次閱讀
    <b class='flag-5'>Linux</b><b class='flag-5'>內(nèi)核</b>中的頁面分配機(jī)制

    ElfBoard技術(shù)貼|在NXP源碼基礎(chǔ)上適配ELF 1開發(fā)板的按鍵功能

    本次源碼適配工作是在NXPi.MX6ULLEVK評(píng)估板的Linux內(nèi)核源碼(特定版本:Linux-imx_4.1.15)基礎(chǔ)上進(jìn)行的。主要目
    的頭像 發(fā)表于 07-10 09:54 ?620次閱讀
    ElfBoard技術(shù)貼|在NXP<b class='flag-5'>源碼</b>基礎(chǔ)上適配ELF 1開發(fā)板的按鍵功能

    歡創(chuàng)播報(bào) 華為宣布鴻蒙內(nèi)核已超越Linux內(nèi)核

    1 華為宣布鴻蒙內(nèi)核已超越Linux內(nèi)核 ? 6月21日,在華為開發(fā)者大會(huì)上, HarmonyOS NEXT(鴻蒙NEXT)——真正獨(dú)立于安卓和iOS的鴻蒙操作系統(tǒng),正式登場(chǎng)。這是HarmonyOS
    的頭像 發(fā)表于 06-27 11:30 ?849次閱讀

    AOSP源碼定制-內(nèi)核驅(qū)動(dòng)編寫

    有時(shí)候?yàn)榱朔治鲆恍さ臋z測(cè),需要在內(nèi)核層面對(duì)讀寫相關(guān)的操作進(jìn)行監(jiān)控,每次去修改對(duì)應(yīng)的內(nèi)核源碼編譯重刷過于耗時(shí)耗力,這里就來嘗試編寫一個(gè)內(nèi)核驅(qū)動(dòng),載入后監(jiān)控讀寫。
    的頭像 發(fā)表于 04-23 11:15 ?1251次閱讀
    AOSP<b class='flag-5'>源碼</b>定制-<b class='flag-5'>內(nèi)核</b>驅(qū)動(dòng)編寫

    使用 PREEMPT_RT 在 Ubuntu 中構(gòu)建實(shí)時(shí) Linux 內(nèi)核

    盟通技術(shù)干貨構(gòu)建實(shí)時(shí)Linux內(nèi)核簡(jiǎn)介盟通技術(shù)干貨Motrotech如果需要在Linux實(shí)現(xiàn)實(shí)時(shí)計(jì)算性能,進(jìn)而有效地將Linux轉(zhuǎn)變?yōu)镽T
    的頭像 發(fā)表于 04-12 08:36 ?2532次閱讀
    使用 PREEMPT_RT 在 Ubuntu 中構(gòu)建實(shí)時(shí) <b class='flag-5'>Linux</b> <b class='flag-5'>內(nèi)核</b>

    Ubuntu 24.04 LTS選用Linux 6.8為默認(rèn)內(nèi)核

    關(guān)于Ubuntu 24.04 LTS使用何種內(nèi)核版本,一直備受關(guān)注。Canonical工程師Andrea Righi昨日宣布,Ubuntu 24.04將默認(rèn)搭載Linux 6.8內(nèi)核。
    的頭像 發(fā)表于 01-29 11:27 ?1127次閱讀

    rk3399移植Linux內(nèi)核

    RK3399是一款由中國(guó)廠商瑞芯微推出的高性能處理器芯片,被廣泛用于嵌入式系統(tǒng)開發(fā)。在進(jìn)行應(yīng)用程序開發(fā)之前,我們需要將Linux內(nèi)核移植到RK3399上,以支持硬件的驅(qū)動(dòng)和功能。本文將詳細(xì)介紹如何將
    的頭像 發(fā)表于 01-08 09:56 ?1156次閱讀
    主站蜘蛛池模板: 乳色吐息未增删樱花ED在线观看 | 欧美猛男gaygayxxgv| 全身无赤裸裸美女网站| 鞋奴的视频VK| 2018三级网站免费观看| 嫩草国产福利视频一区二区| 青青久在线| 性色AV一区二区三区V视界影院| 在线观看国产小视频| 成人毛片免费观看视频大全| 国产亚洲日韩欧美视频| 免费在线看视频| 午夜影视免费| 99国产在线视频有精品视频| 国产剧情麻豆mv| 男女高潮又爽又黄又无遮挡| 无限资源日本2019版| 7m凹凸国产刺激在线视频| 国产精品久久久久久熟妇吹潮软件| 久在线观看福利视频| 婷婷午夜影院| 99视频在线观看视频| 禁漫H天堂免费A漫| 日韩毛片在线视频| 中文字幕久久久| 国产欧美一区二区三区免费| 欧美激情一区二区三区视频| 亚洲欧美精品一中文字幕| 大陆老熟女60岁| 噜噜噜狠狠夜夜躁| 亚洲spank男男实践网站| rio 快播| 狂躁美女BBBBBB视频| 性欧美13处14处破| 别插我B嗯啊视频免费| 狂野欧美性猛XXXX乱大交| 亚洲VA欧美VA天堂V国产综合| 操他射他影院| 免费高清毛片| 伊人久久大香线蕉无码麻豆| 国产精品日韩欧美一区二区三区|