在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

TensorRT構建具有動態形狀的引擎的步驟

星星科技指導員 ? 來源:NVIDIA ? 作者:Ken He ? 2022-05-13 16:40 ? 次閱讀

動態形狀(Dynamic Shapes)是延遲指定部分或全部張量維度直到運行時的能力。動態形狀可以通過 C++Python 接口使用。 以下部分提供了更詳細的信息;但是,這里概述了構建具有動態形狀的引擎的步驟:

1.網絡定義不得具有隱式批次維度。

C++

通過調用創建INetworkDefinition

IBuilder::createNetworkV2(1U <<
        static_cast(NetworkDefinitionCreationFlag::kEXPLICIT_BATCH))

Python

通過調用創建tensorrt.INetworkDefinition

create_network(1 <<
        int(tensorrt.NetworkDefinitionCreationFlag.EXPLICIT_BATCH))

這些調用要求網絡沒有隱式批處理維度。

2.-1作為維度的占位符來指定輸入張量的每個運行時維度。

3.指定一個或多個優化配置文件,為具有運行時維度的輸入指定允許的維度范圍,以及自動調整器將優化的維度。有關詳細信息,請參閱優化配置文件

4.要使用引擎:

  • 從引擎創建執行上下文,與沒有動態形狀的情況相同。
  • 指定步驟 3 中涵蓋輸入維度的優化配置文件之一。
  • 指定執行上下文的輸入維度。設置輸入維度后,您可以獲得TensorRT針對給定輸入維度計算的輸出維度。
  • Enqueue work。

8.1. Specifying Runtime Dimensions

構建網絡時,使用-1表示輸入張量的運行時維度。例如,要創建一個名為foo的 3D 輸入張量,其中最后兩個維度在運行時指定,第一個維度在構建時固定,請發出以下命令。

C++

networkDefinition.addInput("foo", DataType::kFLOAT, Dims3(3, -1, -1))

Python

network_definition.add_input("foo", trt.float32, (3, -1, -1))

在運行時,您需要在選擇優化配置文件后設置輸入維度(請參閱優化配置文件)。設輸入foobindingIndex0,輸入的維度為[3,150,250]。在為前面的示例設置優化配置文件后,您將調用:

C++

context.setBindingDimensions(0, Dims3(3, 150, 250))

Python

context.set_binding_shape(0, (3, 150, 250))

在運行時,向引擎詢問綁定維度會返回用于構建網絡的相同維度,這意味著每個運行時維度都會得到-1。例如:

C++

engine.getBindingDimensions(0) returns a Dims with dimensions {3, -1, -1}

Python

engine.get_binding_shape(0) returns (3, -1, -1)

要獲取特定于每個執行上下文的實際維度,請查詢執行上下文:

C++

context.getBindingDimensions(0) returns a Dims with dimensions {3, 150, 250}.

Python

context.get_binding_shape(0) returns (3, 150, 250).

注意:輸入的setBindingDimensions的返回值僅表明與為該輸入設置的優化配置文件相關的一致性。指定所有輸入綁定維度后,您可以通過查詢網絡輸出綁定的維度來檢查整個網絡在動態輸入形狀方面是否一致。

nvinfer1::Dims out_dim = context->getBindingDimensions(out_index);

if (out_dim.nbDims == -1) {
gLogError << "Invalid network output, this might be caused by inconsistent input shapes." << std::endl;
// abort inference
}

8.2. Optimization Profiles

優化配置文件描述了每個網絡輸入的維度范圍以及自動調諧器將用于優化的維度。使用運行時維度時,您必須在構建時創建至少一個優化配置文件。兩個配置文件可以指定不相交或重疊的范圍。

例如,一個配置文件可能指定最小尺寸[3,100,200],最大尺寸[3,200,300]和優化尺寸[3,150,250]而另一個配置文件可能指定最小,最大和優化尺寸[3,200,100] , [3,300,400] ,和[3,250,250]

要創建優化配置文件,首先構造一個IOptimizationProfile。然后設置最小、優化和最大維度,并將其添加到網絡配置中。優化配置文件定義的形狀必須為網絡定義有效的輸入形狀。以下是前面提到的第一個配置文件對輸入foo的調用:

C++

IOptimizationProfile* profile = builder.createOptimizationProfile();
profile->setDimensions("foo", OptProfileSelector::kMIN, Dims3(3,100,200);
profile->setDimensions("foo", OptProfileSelector::kOPT, Dims3(3,150,250);
profile->setDimensions("foo", OptProfileSelector::kMAX, Dims3(3,200,300);

config->addOptimizationProfile(profile)

Python

profile = builder.create_optimization_profile();
profile.set_shape("foo", (3, 100, 200), (3, 150, 250), (3, 200, 300)) 
config.add_optimization_profile(profile)

在運行時,您需要在設置輸入維度之前設置優化配置文件。配置文件按照添加的順序編號,從0開始。請注意,每個執行上下文必須使用單獨的優化配置文件。 要選擇示例中的第一個優化配置文件,請使用:

C++調用context.setOptimizationProfileAsync(0, stream)

其中stream是在此上下文中用于后續enqueue()或enqueueV2()調用的 CUDA 流。

Python設置context.set_optimization_profile_async(0, stream)

如果關聯的 CUDA 引擎具有動態輸入,則必須使用唯一的配置文件索引至少設置一次優化配置文件,該唯一配置文件索引未被其他未銷毀的執行上下文使用。對于為引擎創建的第一個執行上下文,隱式選擇配置文件 0。

可以調用setOptimizationProfileAsync()在配置文件之間切換。它必須在當前上下文中的任何enqueue()enqueueV2()操作完成后調用。當多個執行上下文同時運行時,允許切換到以前使用但已被具有不同動態輸入維度的另一個執行上下文釋放的配置文件。

setOptimizationProfileAsync()函數替換了現在已棄用的 APIsetOptimizationProfile()版本。使用setOptimizationProfile()在優化配置文件之間切換可能會導致后續enqueue()enqueueV2()操作操作中的 GPU 內存復制操作。要在入隊期間避免這些調用,請改用setOptimizationProfileAsync()API。

在由多個配置文件構建的引擎中,每個配置文件都有單獨的綁定索引。第K個配置文件的輸入/輸出張量的名稱附加了[profile K],其中K以十進制表示。例如,如果INetworkDefinition的名稱為“foo”,并且bindingIndex指的是優化配置文件中索引為3的張量,則engine.getBindingName ( bindingIndex )返回“foo [profile 3]”。

同樣,如果使用ICudaEngine::getBindingIndex(name)獲取第一個配置文件 (K=0) 之外的配置文件 K 的索引,請將“[profile K]”附加到INetworkDefinition中使用的名稱。例如,如果張量在INetworkDefinition中被稱為“foo” ,則engine.getBindingIndex ( “ foo [profile 3] ” )在優化配置文件3中返回張量“foo”的綁定索引。

始終省略K=0的后綴。

8.2.1. Bindings For Multiple Optimization Profiles

考慮一個具有四個輸入、一個輸出、在IBuilderConfig中具有三個優化配置文件的網絡。該引擎有15個綁定,每個優化配置文件有5個,在概念上組織為一個表:

每行都是一個配置文件。表中的數字表示綁定索引。第一個配置文件的綁定索引為 0..4,第二個配置文件為 5..9,第三個配置文件為 10..14。

對于綁定屬于第一個配置文件但指定了另一個配置文件的情況,接口具有“自動更正”功能。在這種情況下,TensorRT 會警告錯誤,然后從同一列中選擇正確的綁定索引。

為了向后半兼容,接口在綁定屬于第一個配置文件但指定了另一個配置文件的情況下具有“自動更正”功能。在這種情況下,TensorRT 會警告錯誤,然后從同一列中選擇正確的綁定索引。

8.3. Layer Extensions For Dynamic Shapes

一些層具有允許指定動態形狀信息的可選輸入,并且有一個新層IShapeLayer用于在運行時訪問張量的形狀。此外,一些層允許計算新的形狀。下一節將討論語義細節和限制。以下是與動態形狀結合使用時可能有用的內容的摘要。

IShapeLayer輸出一個包含輸入張量尺寸的一維張量。例如,如果輸入張量的維度為[2,3,5,7],則輸出張量是包含{2,3,5,7}的四元素一維張量。如果輸入張量是標量,則它的維度為[] ,輸出張量是包含{}的零元素一維張量。

IResizeLayer接受包含所需輸出尺寸的可選第二個輸入。

IShuffleLayer接受包含重塑尺寸的可選第二個輸入。例如,以下網絡將張量Y重塑為與X具有相同的維度:

C++

auto* reshape = networkDefinition.addShuffle(Y);
reshape.setInput(1, networkDefintion.addShape(X)->getOutput(0));

Python

reshape = network_definition.add_shuffle(y)
reshape.set_input(1, network_definition.add_shape(X).get_output(0))

ISliceLayer接受可選的第二、第三和第四個輸入,其中包含開始、大小和步幅。

IConcatenationLayer, IElementWiseLayer, IGatherLayer, IIdentityLayer, and IReduceLayer

可用于對形狀進行計算并創建新的形狀張量。

8.4. Restrictions For Dynamic Shapes

由于層的權重具有固定大小,因此會出現以下層限制:

  • IConvolutionLayerIDeconvolutionLayer要求通道維度是構建時常數。
  • IFullyConnectedLayer要求最后三個維度是構建時常量。
  • Int8要求通道維度是構建時常數。
  • 接受額外形狀輸入的層(IResizeLayerIShuffleLayerISliceLayer)要求額外的形狀輸入與最小和最大優化配置文件的尺寸以及運行時數據輸入的尺寸兼容;否則,它可能導致構建時或運行時錯誤。

必須是構建時常量的值不必是 API 級別的常量。 TensorRT 的形狀分析器通過進行形狀計算的層進行逐個元素的常數傳播。常量傳播發現一個值是構建時常量就足夠了。

8.5. Execution Tensors vs. Shape Tensors

使用動態形狀的引擎采用兩階段執行策略。

  1. 計算所有張量的形狀
  2. 將工作流式傳輸到 GPU。

階段 1 是隱含的,由需求驅動,例如在請求輸出維度時。第 2 階段與之前版本的TensorRT 相同。兩階段執行對動態性施加了一些限制,這些限制對于理解是很重要的。

關鍵限制是:

  • 張量的等級必須在構建時確定。
  • 張量是執行張量、形狀張量或兩者兼而有之。歸類為形狀張量的張量受到限制。

執行張量是傳統的TensorRT張量。形狀張量是與形狀計算相關的張量。它必須是0D1D,類型為Int32FloatBool,并且其形狀必須在構建時可確定。例如,有一個IShapeLayer,其輸出是一維張量,其中包含輸入張量的維度。輸出是一個形狀張量。IShuffleLayer接受一個可選的第二個輸入,可以指定重塑尺寸。第二個輸入必須是一個形狀張量。

有些層在它們處理的張量類型方面是“多態的”。例如,IElementWiseLayer可以將兩個INT32執行張量相加或將兩個INT32形狀張量相加。張量的類型取決于其最終用途。如果總和用于重塑另一個張量,那么它就是一個“形狀張量”。

8.5.1. Formal Inference Rules

TensorRT 用于對張量進行分類的形式推理規則基于類型推理代數。令E表示執行張量, S表示形狀張量。

IActivationLayer具有:

IActivationLayer: E → E

因為它將執行張量作為輸入,將執行張量作為輸出。IElementWiseLayer在這方面是多態的,有兩個特點:

IElementWiseLayer: S × S → S, E × E → E

為簡潔起見,讓我們采用約定t是表示任一類張量的變量,并且特征中的所有t都指同一類張量。然后,前面的兩個特征可以寫成一個單一的多態特征:

IElementWiseLayer: t × t → t

雙輸入IShuffleLayer有一個形狀張量作為第二個輸入,并且相對于第一個輸入是多態的:

IShuffleLayer (two inputs): t × S → t

IConstantLayer沒有輸入,但可以產生任何一種張量,所以它的特征是:

IConstantLayer: → t

IShapeLayer的特征允許所有四種可能的組合E→E 、 E→S 、 S→E和S→S ,因此可以用兩個自變量編寫:

IShapeLayer: t1 → t2

這是完整的規則集,它也可以作為可以使用哪些層來操縱形狀張量的參考:

IAssertionLayer: S → 
IConcatenationLayer: t × t × ...→ t
IIfConditionalInputLayer: t → t
IIfConditionalOutputLayer: t → t
IConstantLayer: → t
IActivationLayer: t → t
IElementWiseLayer: t × t → t
IFillLayer: S → t
IFillLayer: S × E × E → E 
IGatherLayer: t × t → t
IIdentityLayer: t → t
IReduceLayer: t → t
IResizeLayer (one input): E → E
IResizeLayer (two inputs): E × S → E
ISelectLayer: t × t × t → t
IShapeLayer: t1 → t2
IShuffleLayer (one input): t → t
IShuffleLayer (two inputs): t × S → t
ISliceLayer (one input): t → t
ISliceLayer (two inputs): t × S → t
ISliceLayer (three inputs): t × S × S → t
ISliceLayer (four inputs): t × S × S × S → t
IUnaryLayer: t → t
all other layers: E × ... → E × ...

因為輸出可以是多個后續層的輸入,所以推斷的“類型”不是唯一的。例如,一個IConstantLayer可能會饋入一個需要執行張量的用途和另一個需要形狀張量的用途。IConstantLayer的輸出被歸類為兩者,可以在兩階段執行的階段 1 和階段 2 中使用。

在構建時知道形狀張量的等級的要求限制了ISliceLayer可用于操縱形狀張量的方式。具體來說,如果指定結果大小的第三個參數不是構建時常數,則生成的形狀張量的長度在構建時將不再已知,從而打破形狀張量對構建時形狀的限制.更糟糕的是,它可能被用來重塑另一個張量,打破了在構建時必須知道張量等級的限制。

可以通過方法ITensor::isShapeTensor()ITensor::isExecutionTensor ()方法檢查 TensorRT 的推理,它為形狀張量返回true,它為執行張量返回true。在調用這些方法之前先構建整個網絡,因為它們的答案可能會根據添加的張量用途而改變。

例如,如果一個部分構建的網絡將兩個張量T1T2相加來創建張量T3,并且還不需要任何形狀張量,則isShapeTensor()對所有三個張量都返回false。將IShuffleLayer的第二個輸入設置為T3會導致所有三個張量成為形狀張量,因為IShuffleLayer要求其第二個可選輸入是形狀張量,如果IElementWiseLayer的輸出是形狀張量,那么它的輸入也是形狀張量。

8.6. Shape Tensor I/O (Advanced)

有時需要使用形狀張量作為網絡 I/O 張量。例如,考慮一個僅由IshuffleLayer組成的網絡。 TensorRT 推斷第二個輸入是一個形狀張量。ITensor::isShapeTensor為它返回 true。因為它是一個輸入形狀張量,所以 TensorRT 需要兩件事:

  • 在構建時:形狀張量的優化配置文件值。
  • 在運行時:形狀張量的值。

輸入形狀張量的形狀在構建時始終是已知的。這是需要描述的值,因為它們可用于指定執行張量的維度。

可以使用IOptimizationProfile::setShapeValues設置優化配置文件值。類似于必須為具有運行時維度的執行張量提供最小、最大和優化維度的方式,必須在構建時為形狀張量提供最小、最大和優化值。

對應的運行時方法是IExecutionContext::setInputShapeBinding,它在運行時設置形狀張量的值。

因為“執行張量”與“形狀張量”的推斷是基于最終用途,所以 TensorRT無法推斷網絡輸出是否為形狀張量。您必須通過INetworkDefinition::markOutputForShapes方法告訴它。

除了讓您輸出形狀信息以進行調試外,此功能對于編寫引擎也很有用。例如,考慮構建三個引擎,每個引擎用于子網絡A、B、C,其中從A 到 B 或 B 到 C的連接可能涉及形狀張量。逆序構建網絡:C、B、A。構建網絡 C 后,可以使用ITensor::isShapeTensor判斷輸入是否為形狀張量,并使用INetworkDefinition::markOutputForShapes標記網絡中對應的輸出張量B.然后檢查B的哪些輸入是形狀張量,并在網絡A中標記對應的輸出張量。

網絡邊界處的形狀張量必須具有Int32類型。它們不能具有FloatBool類型。Bool的一種解決方法是使用Int32作為 I/O 張量,帶有01,并且:

  • 通過ElementWiseOperation::kGREATER轉換為Bool ,即x > 0
  • 通過ISelectLayerBool轉換,即y ? 1:0

8.7. INT8 Calibration With Dynamic Shapes

要為具有動態形狀的網絡運行 INT8 校準,必須設置校準優化配置文件。使用配置文件的kOPT值執行校準。校準輸入數據大小必須與此配置文件匹配。

要創建校準優化配置文件,首先,構造一個IOptimizationProfile,其方式與創建一般優化配置文件的方式相同。然后將配置文件設置為配置:

C++

config->setCalibrationProfile(profile)

Python

config.set_calibration_profile(profile)

校準配置文件必須有效或為nullptrkMINkMAX值被kOPT覆蓋。要檢查當前校準配置文件,請使用IBuilderConfig::getCalibrationProfile。 此方法返回指向當前校準配置文件的指針,如果未設置校準配置文件,則返回nullptr。為具有動態形狀的網絡運行校準時,getBatchSize()校準器方法必須返回1 。

關于作者

Ken He 是 NVIDIA 企業級開發者社區經理 & 高級講師,擁有多年的 GPU 和人工智能開發經驗。自 2017 年加入 NVIDIA 開發者社區以來,完成過上百場培訓,幫助上萬個開發者了解人工智能和 GPU 編程開發。在計算機視覺,高性能計算領域完成過多個獨立項目。并且,在機器人無人機領域,有過豐富的研發經驗。對于圖像識別,目標的檢測與跟蹤完成過多種解決方案。曾經參與 GPU 版氣象模式GRAPES,是其主要研發者。

審核編輯:郭婷

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 引擎
    +關注

    關注

    1

    文章

    361

    瀏覽量

    22613
  • C++
    C++
    +關注

    關注

    22

    文章

    2114

    瀏覽量

    73785
收藏 人收藏

    評論

    相關推薦

    容器引擎是什么意思

    容器引擎是一種虛擬化技術,它利用操作系統的內核來實現對應用程序的隔離和打包,使得應用程序可以在不同的環境中運行而無需修改代碼。主機推薦小編為您整理發布容器引擎是什么意思,以下是關于容器引擎的詳細解釋。
    的頭像 發表于 01-09 09:49 ?82次閱讀

    常見的容器云服務引擎有哪些?

    常見的容器云服務引擎有哪些?云服務引擎涵蓋數據庫、數據存儲、數據處理、數據分析、容器云、機器學習及數據集成等多個領域,提供一站式解決方案。云服務引擎是云計算領域的重要組成部分,它們提供了各種服務來幫助用戶
    的頭像 發表于 01-07 09:49 ?65次閱讀

    在NVIDIA TensorRT-LLM中啟用ReDrafter的一些變化

    Recurrent Drafting (簡稱 ReDrafter) 是蘋果公司為大語言模型 (LLM) 推理開發并開源的一種新型推測解碼技術,該技術現在可與 NVIDIA TensorRT-LLM 一起使用。
    的頭像 發表于 12-25 17:31 ?221次閱讀
    在NVIDIA <b class='flag-5'>TensorRT</b>-LLM中啟用ReDrafter的一些變化

    解鎖NVIDIA TensorRT-LLM的卓越性能

    NVIDIA TensorRT-LLM 是一個專為優化大語言模型 (LLM) 推理而設計的庫。它提供了多種先進的優化技術,包括自定義 Attention Kernel、Inflight
    的頭像 發表于 12-17 17:47 ?247次閱讀

    NVIDIA TensorRT-LLM Roadmap現已在GitHub上公開發布

    感謝眾多用戶及合作伙伴一直以來對NVIDIA TensorRT-LLM的支持。TensorRT-LLM 的 Roadmap 現已在 GitHub 上公開發布!
    的頭像 發表于 11-28 10:43 ?326次閱讀
    NVIDIA <b class='flag-5'>TensorRT</b>-LLM Roadmap現已在GitHub上公開發布

    TensorRT-LLM低精度推理優化

    本文將分享 TensorRT-LLM 中低精度量化內容,并從精度和速度角度對比 FP8 與 INT8。首先介紹性能,包括速度和精度。其次,介紹量化工具 NVIDIA TensorRT Model
    的頭像 發表于 11-19 14:29 ?386次閱讀
    <b class='flag-5'>TensorRT</b>-LLM低精度推理優化

    靜態與動態載荷下具有粘彈性密封劑光伏組件的力學特性分析

    對光伏組件和建筑集成光伏系統均需要進行機械性能評估,這一評估是確保這些系統長期功能性和優化商業產品的關鍵步驟。通過機械加載、非均勻機械加載和動態機械加載等方式進行性能測試,以驗證光伏組件在外部機械
    的頭像 發表于 11-19 01:03 ?341次閱讀
    靜態與<b class='flag-5'>動態</b>載荷下<b class='flag-5'>具有</b>粘彈性密封劑光伏組件的力學特性分析

    使用SSR構建React應用的步驟

    使用SSR(Server-Side Rendering,服務器端渲染)構建React應用的步驟通常包括以下幾個階段: 一、項目初始化與配置 創建React項目 : 可以使用Create React
    的頭像 發表于 11-18 11:30 ?375次閱讀

    容器云服務引擎是什么意思?

    容器云服務引擎是什么意思?容器云服務引擎是一種基于云原生架構的容器編排工具,能夠幫助用戶快速構建、部署和管理容器化應用。它支持容器化應用的全生命周期管理,包括部署、管理和擴展,旨在簡化云原生應用的操作過程。
    的頭像 發表于 10-19 17:08 ?204次閱讀

    如何構建Linux根文件系統

    構建Linux根文件系統是一個涉及多個步驟和概念的過程,它對于Linux系統的啟動和運行至關重要。
    的頭像 發表于 10-05 16:47 ?332次閱讀

    容器云服務引擎是什么?如何使用

    容器云服務引擎(CloudContainerEngine,簡稱CCE),是一個企業級的Kubernetes集群托管服務,提供高度可擴展、高性能的云原生應用部署和管理方案。容器云服務引擎一種基于云原生
    的頭像 發表于 09-30 10:17 ?220次閱讀

    魔搭社區借助NVIDIA TensorRT-LLM提升LLM推理效率

    “魔搭社區是中國最具影響力的模型開源社區,致力給開發者提供模型即服務的體驗。魔搭社區利用NVIDIA TensorRT-LLM,大大提高了大語言模型的推理性能,方便了模型應用部署,提高了大模型產業應用效率,更大規模地釋放大模型的應用價值。”
    的頭像 發表于 08-23 15:48 ?491次閱讀

    如何使用Cygwin在Win64中構建環境?

    如何使用Cygwin在Win64中構建環境? 我已經下載了cross_tool、cygwin_x86-84.exe和 sdk, 那么我應該采取什么步驟構建一個好的編譯環境呢?
    發表于 07-10 06:59

    交換芯片的構建方式

    交換芯片的構建方式是一個高度復雜且精細的過程,它涉及多個關鍵步驟和考量因素。下面將詳細闡述交換芯片的構建方式。
    的頭像 發表于 03-22 16:22 ?495次閱讀

    谷歌搜索引擎優化的各個方面和步驟

    谷歌搜索引擎是最受歡迎和廣泛使用的搜索引擎之一,為了使你的網站在谷歌上更好地排名并提高曝光度,你可以采取一些谷歌搜索引擎優化的步驟。 使用關鍵字研究工具,如Google AdWords
    的頭像 發表于 01-25 10:29 ?941次閱讀
    主站蜘蛛池模板: 狠狠狠色丁香婷婷综合激情| 国产特级毛片aaaaaa毛片| 婷婷综合网站| 久久精品高清| 久久综合狠狠综合久久综合88| 色偷偷91久久综合噜噜噜噜| 午夜精品久久久久蜜桃| 嘿嘿嘿视频在线观看网站| 高清一区二区三区四区五区| 亚洲一级免费视频| 欧美视频免费一区二区三区| 五月激情网站| 操香蕉| 5555kkkk香蕉在线观看| 玖玖草在线观看| 阿v视频在线观看免费播放| h网站在线播放| xxx日本69hd| 欧美呜巴又大粗又长| 亚洲ol| 欧美视频精品在线| 国产在线视频资源| 久久久免费| 久久国产伦三级理电影| 俺来也婷婷| 666精品国产精品亚洲| 操的好爽| 天天操人人爱| 免费视频精品| 四虎h789fcom| 西西人体44renti大胆亚洲| 日本级毛片免费观看| 老师办公室高h文小说| 亚洲成a人v在线观看| 欧美成人午夜精品一区二区| 成人国产日本亚洲精品| 女同久久| 操日韩美女| 精品一区二区三区视频| 日韩理论电影2021第1页| 成年黄网站免费大全毛片|