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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

【Android 修炼手册】Gradle 篇 -- Gradle 源码分析

發布時間:2023/12/15 36 豆豆
生活随笔 收集整理的這篇文章主要介紹了 【Android 修炼手册】Gradle 篇 -- Gradle 源码分析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

預備知識

  • 理解 gradle 的基本開發
  • 了解 gradle task 和 plugin 使用及開發
  • 了解 android gradle plugin 的使用
  • 看完本文可以達到什么程度

  • 了解 gradle 的實現原理
  • 閱讀前準備工作

  • clone EasyGradle 項目
  • 下載 Gradle 源碼 作為參考
  • 讀代碼的姿勢

  • 調用鏈路,方便讀代碼時對照
  • 集中于整體框架,一些細節不做追究
  • 目錄

    本文主要從下面幾個部分進行分析

  • Gradle 的啟動
  • loadSettings
  • configureBuild
  • constructTaskGraph
  • runTasks
  • finishBuild
  • gradle 腳本如何編譯和執
  • 插件調用流程
  • 一、Gradle 的啟動

    1.1 整體實現圖

    1.2 具體分析

    我們執行一個構建任務的時候,都是執行 ./gradlew assembleDebug 這樣的命令,其中的 gradlew 腳本就是整個 gradle 構建的入口,我們先從這里看起。
    前面的代碼基本上就是判斷環境,設置變量的,直接看最后一行:

    exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@" 復制代碼

    最后執行的命令基本上如下:

    exec $JAVA_HOME/bin/java -classpath $APP_HOME/gradle/wrapper/gradle-wrapper.jar org.gradle.wrapper.GradleWrapperMain 復制代碼

    基本上可以看到,就是執行了 gradle/wrapper/gradle-wrapper.jar 里的 org.gradle.wrapper.GradleWrapperMain,這樣我們就知道了,gradle 的入口類是 org.gradle.wrapper.GradleWrapperMain,也就知道代碼該從何開始看了。
    先看 GradleWrapperMain 的 main 函數:

    // GradleWrapperMain public static void main(String[] args) throws Exception {// ...WrapperExecutor wrapperExecutor = WrapperExecutor.forWrapperPropertiesFile(propertiesFile);wrapperExecutor.execute(args,new Install(logger, new Download(logger, "gradlew", wrapperVersion()), new PathAssembler(gradleUserHome)),new BootstrapMainStarter()); } 復制代碼

    重要的類有兩個 org.gradle.wrapper.WrapperExecutor 和 org.gradle.wrapper.BootstrapMainStarter。我們繼續跟進 WrapperExecutor.execute 里看一下:

    // WrapperExecutor.execute public void execute(String[] args, Install install, BootstrapMainStarter bootstrapMainStarter) throws Exception {File gradleHome = install.createDist(config);bootstrapMainStarter.start(args, gradleHome); } 復制代碼

    這里就做了兩件事:

  • 下載 gradle wrapper 需要的依賴以及源碼。其中的 gradle wrapper 版本就是我們在 gradle/wrapper/gradle-wrapper.properties 里配置的 distributionUrl,下載位置就是在 gradle-wrapper.properties 里配置的 distributionPath 和 zipStorePath。zipStorePath 是下載的壓縮包位置,distributionPath 是解壓后的位置,一般默認的位置就是 HOME/.gradle/wrapper/dists/,在這里就可以找到 gradle wrapper 的內容了。
    如果創建過多個項目的話,我們在 HOME/.gradle/wrapper/dists/ 里可以看到不同版本的 gradle wrapper,這也說明了我們之前最開始說的,為什么要使用 gradle wrapper 而不是直接在電腦里安裝 gradle,就是因為 gradle wrapper 會根據不同的項目下載不同版本的內容,項目彼此之間互不影響。
  • 執行 gradle 構建流程。這里就是順著 BootstrapMainStarter.start() 往下執行了,中間過程就不看了,比較曲折,對理解整體流程也沒什么太大的幫助。最終會運行到 DefaultGradleLauncher.executeTasks(),然后再往下的流程就非常清晰了。
  • // DefaultGradleLauncher public GradleInternal executeTasks() {doBuildStages(Stage.Build);return gradle; }private void doBuildStages(Stage upTo) {// ...loadSettings();configureBuild();constructTaskGraph();runTasks();finishBuild(); } 復制代碼

    基本上構建過程就是分五步走,下面分別看這五個流程。

    二、loadSettings

    2.1 整體實現圖

    2.2 具體分析

    loadSettings 主要是加載 settings.gradle 文件,然后創建對應的 project。

    // DefaultGradleLauncher.loadSettings private void loadSettings() {if (stage == null) {buildListener.buildStarted(gradle);buildOperationExecutor.run(new LoadBuild());stage = Stage.Load;} } 復制代碼

    整體構建流程:

    2.2.1 調用 BuildListener.buildStarted() 回調接口

    通知構建開始。這個就是我們之前在 Gradle 基本使用 里說的生命周期回調。

    2.2.2 執行 init 腳本

    調用鏈路

    LoadBuild.run -> InitScriptHandler.executeScripts 復制代碼

    之前在 Gradle 基本使用 里說過 init.gradle 的作用,會在每個項目 build 之前被調用,做一些初始化的操作,就是在這里被調用的。

    2.2.3 查找 settings.gradle 位置

    調用鏈路

    LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> DefaultSettingsLoader.findSettings -> DefaultSettingsFinder.find -> BuildLayoutFactory.getLayoutFor 復制代碼

    實現分析
    在 getLayoutFor 里,查找 settings.gradle 文件邏輯如下:

  • 如果參數里通過 -c xxx.gradle 指定了 settings.gradle 文件的位置,那么直接使用指定的 settings.gradle 文件
  • 如果沒有指定 settings 文件,就在當前目錄下查找
  • 如果當前目錄沒有,會一直往上級目錄查找,以及同級的 maseter/ 目錄下查找
  • 如果都沒有找到,還是默認在當前目錄下
  • // BuildLayoutFactory public BuildLayout getLayoutFor(BuildLayoutConfiguration configuration) {if (configuration.isUseEmptySettings()) {return new BuildLayout(configuration.getCurrentDir(), configuration.getCurrentDir(), null);}File explicitSettingsFile = configuration.getSettingsFile();if (explicitSettingsFile != null) {if (!explicitSettingsFile.isFile()) {throw new MissingResourceException(explicitSettingsFile.toURI(), String.format("Could not read settings file '%s' as it does not exist.", explicitSettingsFile.getAbsolutePath()));}return new BuildLayout(configuration.getCurrentDir(), configuration.getCurrentDir(), explicitSettingsFile);}File currentDir = configuration.getCurrentDir();boolean searchUpwards = configuration.isSearchUpwards();return getLayoutFor(currentDir, searchUpwards ? null : currentDir.getParentFile()); } 復制代碼
    2.2.4 編譯 buildSrc 文件夾下的內容,buildSrc 可以看作插件類似的功能

    調用鏈路

    LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> BuildSourceBuilder.buildAndCreateClassLoader 復制代碼

    在上一步找到 settings.gradle 文件以后,會以 settings.gradle 所在的同級目錄下,查找 buildSrc 目錄,并進行編譯,這樣可以保證在構建 settings.gradle 的時候可以引用到 buildSrc 目錄里的內容。

    2.2.5 解析 gradle.properites

    調用鏈路

    LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> NotifyingSettingsProcessor.process -> PropertiesLoadingSettingsProcessor.process -> DefaultGradlePropertiesLoader.loadProperties 復制代碼

    實現分析
    這一步會讀取 gradle.properties 文件里的配置,系統配置,環境變量,以及命令行傳入的配置并存儲。

    // DefaultGradlePropertiesLoader void loadProperties(File settingsDir, StartParameter startParameter, Map<String, String> systemProperties, Map<String, String> envProperties) {defaultProperties.clear();overrideProperties.clear();addGradleProperties(defaultProperties, new File(settingsDir, Project.GRADLE_PROPERTIES));addGradleProperties(overrideProperties, new File(startParameter.getGradleUserHomeDir(), Project.GRADLE_PROPERTIES));setSystemProperties(startParameter.getSystemPropertiesArgs());overrideProperties.putAll(getEnvProjectProperties(envProperties));overrideProperties.putAll(getSystemProjectProperties(systemProperties));overrideProperties.putAll(startParameter.getProjectProperties()); } 復制代碼
    2.2.6 解析 settings.gradle

    調用鏈路

    LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> NotifyingSettingsProcessor.process -> PropertiesLoadingSettingsProcessor.process -> ScriptEvaluatingSettingsProcessor.process -> ScriptEvaluatingSettingsProcessor.applySettingsScript -> BuildOperationScriptPlugin.apply 復制代碼

    實現分析
    在 ScriptEvaluatingSettingsProcessor 里,先創建了 SettingsInternal 實例,以及 ScriptSource 實例,代表 settings.gradle 文件在內存中的映射,之后就調用 BuildOperationScriptPlugin.apply 去執行 settings.gradle 文件了。
    關于 BuildOperationScriptPlugin.apply,我們后面細說,因為在解析 build.gradle 文件的時候也會用到這個方法。
    下面是對應的代碼:

    // ScriptEvaluatingSettingsProcessor public SettingsInternal process(GradleInternal gradle,SettingsLocation settingsLocation,ClassLoaderScope buildRootClassLoaderScope,StartParameter startParameter) {Timer settingsProcessingClock = Timers.startTimer();Map<String, String> properties = propertiesLoader.mergeProperties(Collections.<String, String>emptyMap());SettingsInternal settings = settingsFactory.createSettings(gradle, settingsLocation.getSettingsDir(),settingsLocation.getSettingsScriptSource(), properties, startParameter, buildRootClassLoaderScope);applySettingsScript(settingsLocation, settings);LOGGER.debug("Timing: Processing settings took: {}", settingsProcessingClock.getElapsed());return settings; }private void applySettingsScript(SettingsLocation settingsLocation, final SettingsInternal settings) {ScriptSource settingsScriptSource = settingsLocation.getSettingsScriptSource();ClassLoaderScope settingsClassLoaderScope = settings.getClassLoaderScope();ScriptHandler scriptHandler = scriptHandlerFactory.create(settingsScriptSource, settingsClassLoaderScope);ScriptPlugin configurer = configurerFactory.create(settingsScriptSource, scriptHandler, settingsClassLoaderScope, settings.getRootClassLoaderScope(), true);configurer.apply(settings); } 復制代碼
    2.2.7 創建 project 以及 subproject

    調用鏈路

    LoadBuild.run -> NotifyingSettingsLoader.findAndLoadSettings -> CompositeBuildSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findAndLoadSettings -> DefaultSettingsLoader.findSettingsAndLoadIfAppropriate -> NotifyingSettingsProcessor.process -> ProjectPropertySettingBuildLoader.load -> InstantiatingBuildLoader.load 復制代碼

    實現分析
    在解析了 settings.gradle 文件以后,就可以知道項目里有哪些 project,就可以創建 project 實例了。

    // InstantiatingBuildLoader // 這里傳入的參數對應的是:rootProjectDescriptor: SettingsInternal.getRootProject() defaultProject: SettingsInternal.getDefaultProject() buildRootClassLoaderScope:SettingsInternal.getRootClassLoaderScope() public void load(ProjectDescriptor rootProjectDescriptor, ProjectDescriptor defaultProject, GradleInternal gradle, ClassLoaderScope buildRootClassLoaderScope) {createProjects(rootProjectDescriptor, gradle, buildRootClassLoaderScope);attachDefaultProject(defaultProject, gradle); }private void attachDefaultProject(ProjectDescriptor defaultProject, GradleInternal gradle) {gradle.setDefaultProject(gradle.getRootProject().getProjectRegistry().getProject(defaultProject.getPath())); }private void createProjects(ProjectDescriptor rootProjectDescriptor, GradleInternal gradle, ClassLoaderScope buildRootClassLoaderScope) {// 創建主項目實例// ProjectInternal 繼承自 Project,最終返回的 rootProject 是 DefaultProject 類型ProjectInternal rootProject = projectFactory.createProject(rootProjectDescriptor, null, gradle, buildRootClassLoaderScope.createChild("root-project"), buildRootClassLoaderScope);gradle.setRootProject(rootProject);addProjects(rootProject, rootProjectDescriptor, gradle, buildRootClassLoaderScope); }private void addProjects(ProjectInternal parent, ProjectDescriptor parentProjectDescriptor, GradleInternal gradle, ClassLoaderScope buildRootClassLoaderScope) {// 創建子項目實例for (ProjectDescriptor childProjectDescriptor : parentProjectDescriptor.getChildren()) {ProjectInternal childProject = projectFactory.createProject(childProjectDescriptor, parent, gradle, parent.getClassLoaderScope().createChild("project-" + childProjectDescriptor.getName()), buildRootClassLoaderScope);addProjects(childProject, childProjectDescriptor, gradle, buildRootClassLoaderScope);} }// ProjectFactory public DefaultProject createProject(ProjectDescriptor projectDescriptor, ProjectInternal parent, GradleInternal gradle, ClassLoaderScope selfClassLoaderScope, ClassLoaderScope baseClassLoaderScope) {// 獲取 project 對應的 build.gradle File buildFile = projectDescriptor.getBuildFile();ScriptSource source = UriScriptSource.file("build file", buildFile);// 創建 project 實例DefaultProject project = instantiator.newInstance(DefaultProject.class,projectDescriptor.getName(),parent,projectDescriptor.getProjectDir(),source,gradle,gradle.getServiceRegistryFactory(),selfClassLoaderScope,baseClassLoaderScope);// 設置 project 的層級關系if (parent != null) {parent.addChildProject(project);}// 注冊 projectprojectRegistry.addProject(project);return project; } 復制代碼

    這里根據 settings.gradle 的配置,創建項目實例。創建子項目的時候,如果父項目不為空,就將自己設置成父項目的子項目,這樣就可以通過 project.getChildProjects 獲取項目的子項目了。
    我們在寫 gradle 腳本的時候,經常會用到的 project 屬性,就是在這個時候創建出來了。

    到此為止,就解析了 settings.gradle 文件然后創建了項目實例。

    三、configureBuild

    3.1 整體實現圖

    3.2 具體分析

    我們之前有說到,gradle 構建過程分為配置階段和運行階段,配置階段主要是執行腳本的內容,運行階段是執行 task 的內容,這里就是配置階段的流程。要注意,之前說的配置和運行階段,是從整體來看的兩個階段,從源碼來理解,就是這篇文章介紹的幾個階段,要更細化一點。
    配置階段執行的內容比較簡單,就是把 gradle 腳本編譯成 class 文件,然后運行(gradle 是采用 groovy 語言編寫的,groovy 是一門 jvm 語言,所以必須要編譯成 class 才能運行)。

    // DefaultGradleLauncher private void configureBuild() {if (stage == Stage.Load) {buildOperationExecutor.run(new ConfigureBuild());stage = Stage.Configure;} } 復制代碼

    在配置項目的時候,如果指定了 configure-on-demand 參數,只會配置主項目以及執行 task 需要的項目,默認沒有指定,會配置所有的項目,這里只看默認情況。

    3.2.1 配置主項目及其子項目的主要鏈路

    調用鏈路

    ConfigureBuild.run -> DefaultBuildConfigurer.configure -> TaskPathProjectEvaluator.configureHierarchy -> TaskPathProjectEvaluator.configure -> DefaultProject.evaluate -> LifecycleProjectEvaluator.evaluate -> LifecycleProjectEvaluator.doConfigure -> ConfigureActionsProjectEvaluator.evaluate 復制代碼

    實現分析

    // TaskPathProjectEvaluator public void configureHierarchy(ProjectInternal project) {configure(project);for (Project sub : project.getSubprojects()) {configure((ProjectInternal) sub);} } 復制代碼

    最終執行到了 LifecycleProjectEvaluator.doConfigure

    3.2.2 回調 BuildListener.beforeEvaluate 接口

    在這里回調 beforeEvaluate 接口,通知配置將要開始。我們也就知道了這個回調執行的階段。

    3.2.3 設置默認的 task 和 插件

    調用鏈路

    ConfigureBuild.run -> DefaultBuildConfigurer.configure -> TaskPathProjectEvaluator.configureHierarchy -> TaskPathProjectEvaluator.configure -> DefaultProject.evaluate -> LifecycleProjectEvaluator.evaluate -> LifecycleProjectEvaluator.doConfigure -> ConfigureActionsProjectEvaluator.evaluate -> PluginsProjectConfigureActions.execute 復制代碼

    實現分析
    在 PluginsProjectConfigureActions 里,會給 project 添加兩個 task:init 和 wrapper,然后添加幫助插件:org.gradle.help-tasks。

    3.2.4 編譯腳本并執行

    調用鏈路

    ConfigureBuild.run -> DefaultBuildConfigurer.configure -> TaskPathProjectEvaluator.configureHierarchy -> TaskPathProjectEvaluator.configure -> DefaultProject.evaluate -> LifecycleProjectEvaluator.evaluate -> LifecycleProjectEvaluator.doConfigure -> ConfigureActionsProjectEvaluator.evaluate -> BuildScriptProcessor.execute -> BuildOperationScriptPlugin.apply 復制代碼

    實現分析
    這里調用的還是 BuildOperationScriptPlugin.apply 去編譯和執行 build.gradle 腳本,和前面解析 settings.gradle 是一樣的,這里我們先知道這個就是編譯 build.gradle 為 class。
    文件并且執行,然后先往后看流程,后面再詳細說腳本是如何編譯和執行的。

    3.2.5 回調 BuildListener.afterEvaluate
    3.2.6 回調 BuildListener.projectsEvaluated

    四、constructTaskGraph

    4.1 整體實現圖

    4.2 具體分析

    這一步是構建 task 依賴圖

    // DefaultGradleLauncher private void constructTaskGraph() {if (stage == Stage.Configure) {buildOperationExecutor.run(new CalculateTaskGraph());stage = Stage.TaskGraph;} } 復制代碼
    4.2.1 處理需要排除的 task

    調用鏈路

    CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> ExcludedTaskFilteringBuildConfigurationAction.configure 復制代碼

    實現分析

    // ExcludedTaskFilteringBuildConfigurationAction public void configure(BuildExecutionContext context) {GradleInternal gradle = context.getGradle();Set<String> excludedTaskNames = gradle.getStartParameter().getExcludedTaskNames();if (!excludedTaskNames.isEmpty()) {final Set<Spec<Task>> filters = new HashSet<Spec<Task>>();for (String taskName : excludedTaskNames) {filters.add(taskSelector.getFilter(taskName));}gradle.getTaskGraph().useFilter(Specs.intersect(filters));}context.proceed(); } 復制代碼

    這一步是用來處理需要排除的 task,也就是在命令行通過 -x or --exclude-task 指定的 task,這里主要是給 TaskGraph 設置了 filter,以便在后面計算依賴的時候排除相應的 task。

    4.2.2 添加默認的 task

    調用鏈路

    CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> DefaultTasksBuildExecutionAction.configure 復制代碼

    實現分析
    這里會檢查命令行里是否有傳入 Task 名稱進來,如果指定了要執行的 task,那么什么都不做。
    如果沒有指定,就看 project 是否有默認的 task,默認的 task 可以通過 defaultTasks 在 build.gradle 里進行指定。
    如果也默認 task 也沒有,那么就把要指定的 task 設置成 help task,也就是輸出 gradle 的幫助內容。

    4.2.3 計算 task 依賴圖

    調用鏈路

    CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> TaskNameResolvingBuildConfigurationAction.configure 復制代碼

    實現分析

  • 根據命令行的 taskname 篩選 task。如果我們的 task 指定了 project,也就是類似這樣的 :app:assembleDebug,那么就直接選中了 task,如果沒有指定具體 project,那么會把所有 project 下符合 taskname 的 task 都篩選出來。
  • CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> TaskNameResolvingBuildConfigurationAction.configure -> CommandLineTaskParser.parseTasks 復制代碼
  • 把 task 添加到 taskGraph 中,這里會處理 task 的依賴關系,包括 dependson finalizedby mustrunafter shouldrunafter,然后把信息都保存在 org.gradle.execution.taskgraph.TaskInfo 里。
  • CalculateTaskGraph.run -> DefaultBuildConfigurationActionExecuter.select -> TaskNameResolvingBuildConfigurationAction.configure -> DefaultTaskGraphExecuter.addTasks 復制代碼
    4.2.4 生成 task graph

    調用鏈路

    CalculateTaskGraph.run -> TaskGraphExecuter.populate -> DefaultTaskExecutionPlan.determineExecutionPlan 復制代碼

    實現分析
    根據上一步計算的 task 及其依賴,生成 task 圖

    五、runTasks

    5.1 整體實現圖

    5.2 具體分析

    task 圖生成以后,就開始執行 task

    5.2.1 處理 dry run

    調用鏈路

    DefaultBuildExecuter.execute -> DryRunBuildExecutionAction.execute 復制代碼

    實現分析
    如果在命令行里指定了 --dry-run,在這里就會攔截 task 的執行,直接輸出 task 的名稱以及執行的先后關系。

    5.2.2 創建線程,執行 task

    調用鏈路

    DefaultBuildExecuter.execute -> SelectedTaskExecutionAction.execute -> DefaultTaskPlanExecutor.process 復制代碼

    實現分析
    創建 TaskExecutorWorker 去執行 task,默認是 8 個線程。

    // DefaultTaskPlanExecutor public void process(TaskExecutionPlan taskExecutionPlan, Action<? super TaskInternal> taskWorker) {ManagedExecutor executor = executorFactory.create("Task worker for '" + taskExecutionPlan.getDisplayName() + "'");try {WorkerLease parentWorkerLease = workerLeaseService.getCurrentWorkerLease();// 開線程startAdditionalWorkers(taskExecutionPlan, taskWorker, executor, parentWorkerLease); taskWorker(taskExecutionPlan, taskWorker, parentWorkerLease).run();taskExecutionPlan.awaitCompletion();} finally {executor.stop();} } 復制代碼
    5.2.3 task 執行前處理

    調用鏈路

    DefaultBuildExecuter.execute -> SelectedTaskExecutionAction.execute -> DefaultTaskPlanExecutor.process -> TaskExecutorWorker.run -> DefaultTaskExecutionPlan.executeWithTask -> DefaultTaskExecutionPlan.selectNextTask -> DefaultTaskExecutionPlan.processTask -> EventFiringTaskWorker.execute -> DefaultBuildOperationExecutor.run 復制代碼

    實現分析
    到這里就正式開始 task 的執行過程了。有幾個步驟:

  • 回調 TaskExecutionListener.beforeExecute
  • 鏈式執行一些列對 Task 的處理,具體的處理如下:
  • CatchExceptionTaskExecuter.execute // 加了 try catch,防止執行過程中異常 ExecuteAtMostOnceTaskExecuter.execute // 判斷 task 是否執行過 SkipOnlyIfTaskExecuter.execute // 判斷 task 的 onlyif 條件是否滿足執行 SkipTaskWithNoActionsExecuter.execute // 跳過沒有 action 的 task,沒有 action 說明 task 不需要執行 ResolveTaskArtifactStateTaskExecuter.execute // 設置 artifact 狀態 SkipEmptySourceFilesTaskExecuter.execute // 跳過設置了 source file 但是 source file 為空的 task,source file 為空說明 task 沒有需要處理的資源 ValidatingTaskExecuter.execute() // 確認 task 是否可以執行 ResolveTaskOutputCachingStateExecuter.execute // 處理 task output 緩存 SkipUpToDateTaskExecuter.execute // 跳過 update-to-date 的 task ExecuteActionsTaskExecuter.execute // 真正執行 task 復制代碼
    5.2.4 task 執行

    調用鏈路

    DefaultBuildExecuter.execute -> SelectedTaskExecutionAction.execute -> DefaultTaskPlanExecutor.process -> TaskExecutorWorker.run -> DefaultTaskExecutionPlan.executeWithTask -> DefaultTaskExecutionPlan.selectNextTask -> DefaultTaskExecutionPlan.processTask -> EventFiringTaskWorker.execute -> DefaultBuildOperationExecutor.run -> ExecuteActionsTaskExecuter.execute 復制代碼

    實現分析
    經過前面一系列處理,這里開始真正執行 task 了。

  • 回調 TaskActionListener.beforeActions
  • 回調 OutputsGenerationListener.beforeTaskOutputsGenerated
  • 取出 task 中的 Actions 全部執行
  • // ExecuteActionsTaskExecuter private GradleException executeActions(TaskInternal task, TaskStateInternal state, TaskExecutionContext context) {final List<ContextAwareTaskAction> actions = new ArrayList<ContextAwareTaskAction>(task.getTaskActions());int actionNumber = 1;for (ContextAwareTaskAction action : actions) {// ...executeAction("Execute task action " + actionNumber + "/" + actions.size() + " for " + task.getPath(), task, action, context);// ...actionNumber++;}return null; } 復制代碼

    這里可以看到,Task 的本質,其實就是執行其中的 Actions。舉個例子來說,我們一般自定義 Task 的時候,經常用下面的寫法:

    task {doLast {// task 具體任務} } 復制代碼

    這里的 doLast 就相當于給 Task 添加了一個 Action。
    看一下 AbstractTask 的 doLast 方法

    // AbstractTask public Task doLast(final Action<? super Task> action) {// ...taskMutator.mutate("Task.doLast(Action)", new Runnable() {public void run() {getTaskActions().add(wrap(action));}});return this; }private ContextAwareTaskAction wrap(final Action<? super Task> action) {if (action instanceof ContextAwareTaskAction) {return (ContextAwareTaskAction) action;}return new TaskActionWrapper(action); } 復制代碼

    可以看到,我們傳入的閉包,最終是包裝成 TaskActionWrapper 添加到 task 的 actions 中的。

  • 回調 TaskActionListener.afterActions
  • 回調 TaskExecutionListener.afterExecute
  • 六、finishBuild

    6.1 整體實現圖

    6.2 具體分析

    private void finishBuild(BuildResult result) {if (stage == Stage.Finished) {return;}buildListener.buildFinished(result);if (!isNestedBuild()) {gradle.getServices().get(IncludedBuildControllers.class).stopTaskExecution();}stage = Stage.Finished; } 復制代碼

    這里邏輯不多,回調了 BuildListener.buildFinished 接口

    通過上面幾個步驟,我們基本上看到了 gradle 的執行流程,簡單來說,步驟如下:

  • 解析 settings.gradle 并執行,生成 Project 實例
  • 解析 build.gradle 并執行
  • 生成 task 依賴圖
  • 執行 task
  • 七、gradle 腳本如何編譯和執行

    在前面介紹 loadSettings 和 configureBuild 階段的時候,我們提到了 BuildOperationScriptPlugin.apply 這個方法,只是簡單帶過,是用來編譯 gradle 腳本并執行的,這里來具體分析一下。

    7.1 編譯腳本

    調用鏈路

    BuildOperationScriptPlugin.apply -> DefaultScriptPluginFactory.ScriptPluginImpl.apply -> DefaultScriptCompilerFactory.ScriptCompilerImpl.compile -> BuildScopeInMemoryCachingScriptClassCompiler.compile -> CrossBuildInMemoryCachingScriptClassCache.getOrCompile -> FileCacheBackedScriptClassCompiler.compile 復制代碼

    實現分析
    這里編譯過程分為兩部分,首先編譯腳本的 buildscript {} 部分,忽略其他部分,然后再編譯腳本的其他部分并執行。所以 buildscript {} 里的內容會先于其他內容執行

  • 會先檢查緩存,如果有緩存的話,直接使用,沒有緩存再進行編譯

  • 最終會調用到 CompileToCrossBuildCacheAction.execute -> DefaultScriptCompilationHandler.compileToDir -> DefaultScriptCompilationHandler.compileScript 去執行真正的編譯操作
    腳本緩存路徑: /Users/zy/.gradle/caches/4.1/scripts-remapped/build_a3v29m9cbrge95ug6eejz9wuw/31f5shvfkfunwn5ullupyy7xt/cp_proj4dada6424967ba8dfea75e81c8880f7f/classes
    目錄下的 class 如下:

  • 具體編譯方法是通過 RemappingScriptSource.getResource().getText() 獲取到腳本內容,然后通過 GroovyClassLoader.parseClass 編譯的。
    我們以 app/build.gradle 為例,看一下最終生成的腳本是什么樣子的。
    build.gradle 腳本內容

  • apply plugin: 'com.android.application' apply plugin: 'myplugin'android {compileSdkVersion 26defaultConfig {applicationId "com.zy.easygradle"minSdkVersion 19targetSdkVersion 26versionCode 1versionName "1.0"}buildTypes {release {minifyEnabled falseproguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'}}compileOptions {sourceCompatibility 1.8targetCompatibility 1.8}flavorDimensions "size", "color"productFlavors {big {dimension "size"}small {dimension "size"}blue {dimension "color"}red {dimension "color"}} }dependencies { // implementation gradleApi()implementation fileTree(dir: 'libs', include: ['*.jar'])implementation 'com.android.support:appcompat-v7:26.1.0'implementation 'com.android.support.constraint:constraint-layout:1.1.3'implementation project(':module1') }gradle.addBuildListener(new BuildListener() {@Overridevoid buildStarted(Gradle gradle) {// println('構建開始')}@Overridevoid settingsEvaluated(Settings settings) {// println('settings 文件解析完成')}@Overridevoid projectsLoaded(Gradle gradle) {// println('項目加載完成')}@Overridevoid projectsEvaluated(Gradle gradle) {// println('項目解析完成')}@Overridevoid buildFinished(BuildResult result) {// println('構建完成')} })gradle.addProjectEvaluationListener(new ProjectEvaluationListener() {@Overridevoid beforeEvaluate(Project project) {// println("${project.name} 項目配置之前調用")}@Overridevoid afterEvaluate(Project project, ProjectState state) {// println("${project.name} 項目配置之后調用")} })gradle.taskGraph.whenReady {// println("task 圖構建完成") } gradle.taskGraph.beforeTask {// println("task 執行完成") } gradle.taskGraph.afterTask {// println("task 執行完成") }task task1 {doLast {println('task2')} }task task2 {doLast {println('task2')} } task1.finalizedBy(task2) 復制代碼

    編譯后 class 內容

    package defpackage;import groovy.lang.MetaClass; import java.lang.ref.SoftReference; import org.codehaus.groovy.reflection.ClassInfo; import org.codehaus.groovy.runtime.GStringImpl; import org.codehaus.groovy.runtime.ScriptBytecodeAdapter; import org.codehaus.groovy.runtime.callsite.CallSite; import org.codehaus.groovy.runtime.callsite.CallSiteArray; import org.codehaus.groovy.runtime.typehandling.ShortTypeHandling; import org.gradle.api.internal.project.ProjectScript; import org.gradle.internal.scripts.ScriptOrigin;/* compiled from: /Users/zy/workspace/note/blog/android-training/gradle/EasyGradle/app/build.gradle */ public class build_ak168fqfikdepd6py4yef8tgs extends ProjectScript implements ScriptOrigin {private static /* synthetic */ SoftReference $callSiteArray = null;private static /* synthetic */ ClassInfo $staticClassInfo = null;public static transient /* synthetic */ boolean __$stMC = false;private static final /* synthetic */ String __originalClassName = "_BuildScript_";private static final /* synthetic */ String __signature = "988274f32891a2a3d3b8d16074617c05";private static /* synthetic */ CallSiteArray $createCallSiteArray() {String[] strArr = new String[22];build_ak168fqfikdepd6py4yef8tgs.$createCallSiteArray_1(strArr);return new CallSiteArray(build_ak168fqfikdepd6py4yef8tgs.class, strArr);}private static /* synthetic */ void $createCallSiteArray_1(String[] strArr) {strArr[0] = "apply";strArr[1] = "apply";strArr[2] = "android";strArr[3] = "dependencies";strArr[4] = "addBuildListener";strArr[5] = "gradle";strArr[6] = "addProjectEvaluationListener";strArr[7] = "gradle";strArr[8] = "whenReady";strArr[9] = "taskGraph";strArr[10] = "gradle";strArr[11] = "beforeTask";strArr[12] = "taskGraph";strArr[13] = "gradle";strArr[14] = "afterTask";strArr[15] = "taskGraph";strArr[16] = "gradle";strArr[17] = "task";strArr[18] = "task";strArr[19] = "finalizedBy";strArr[20] = "task1";strArr[21] = "task2";}/* JADX WARNING: inconsistent code. *//* Code decompiled incorrectly, please refer to instructions dump. */private static /* synthetic */ org.codehaus.groovy.runtime.callsite.CallSite[] $getCallSiteArray() {/*r0 = $callSiteArray;if (r0 == 0) goto L_0x000e;L_0x0004:r0 = $callSiteArray;r0 = r0.get();r0 = (org.codehaus.groovy.runtime.callsite.CallSiteArray) r0;if (r0 != 0) goto L_0x0019;L_0x000e:r0 = defpackage.build_ak168fqfikdepd6py4yef8tgs.$createCallSiteArray();r1 = new java.lang.ref.SoftReference;r1.<init>(r0);$callSiteArray = r1;L_0x0019:r0 = r0.array;return r0;*/throw new UnsupportedOperationException("Method not decompiled: build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray():org.codehaus.groovy.runtime.callsite.CallSite[]");}public build_ak168fqfikdepd6py4yef8tgs() {build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray();}protected /* synthetic */ MetaClass $getStaticMetaClass() {if (getClass() != build_ak168fqfikdepd6py4yef8tgs.class) {return ScriptBytecodeAdapter.initMetaClass(this);}ClassInfo classInfo = $staticClassInfo;if (classInfo == null) {classInfo = ClassInfo.getClassInfo(getClass());$staticClassInfo = classInfo;}return classInfo.getMetaClass();}public String getContentHash() {return __signature;}public String getOriginalClassName() {return __originalClassName;}public Object run() {CallSite[] $getCallSiteArray = build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray();$getCallSiteArray[0].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "com.android.application"}));$getCallSiteArray[1].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "myplugin"}));$getCallSiteArray[2].callCurrent(this, new _run_closure1(this, this));$getCallSiteArray[3].callCurrent(this, new _run_closure2(this, this));$getCallSiteArray[4].call($getCallSiteArray[5].callGroovyObjectGetProperty(this), new 1(this));$getCallSiteArray[6].call($getCallSiteArray[7].callGroovyObjectGetProperty(this), new 2(this));$getCallSiteArray[8].call($getCallSiteArray[9].callGetProperty($getCallSiteArray[10].callGroovyObjectGetProperty(this)), new _run_closure3(this, this));$getCallSiteArray[11].call($getCallSiteArray[12].callGetProperty($getCallSiteArray[13].callGroovyObjectGetProperty(this)), new _run_closure4(this, this));$getCallSiteArray[14].call($getCallSiteArray[15].callGetProperty($getCallSiteArray[16].callGroovyObjectGetProperty(this)), new _run_closure5(this, this));$getCallSiteArray[17].callCurrent(this, "task1", new _run_closure6(this, this));$getCallSiteArray[18].callCurrent(this, "task2", new _run_closure7(this, this));return $getCallSiteArray[19].call($getCallSiteArray[20].callGroovyObjectGetProperty(this), $getCallSiteArray[21].callGroovyObjectGetProperty(this));}public /* synthetic */ Object this$dist$get$7(String name) {build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray();return ScriptBytecodeAdapter.getGroovyObjectProperty(build_ak168fqfikdepd6py4yef8tgs.class, this, ShortTypeHandling.castToString(new GStringImpl(new Object[]{name}, new String[]{"", ""})));}public /* synthetic */ Object this$dist$invoke$7(String name, Object args) {build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray();return ScriptBytecodeAdapter.invokeMethodOnCurrentN(build_ak168fqfikdepd6py4yef8tgs.class, this, ShortTypeHandling.castToString(new GStringImpl(new Object[]{name}, new String[]{"", ""})), ScriptBytecodeAdapter.despreadList(new Object[0], new Object[]{args}, new int[]{0}));}public /* synthetic */ void this$dist$set$7(String name, Object value) {build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray();ScriptBytecodeAdapter.setGroovyObjectProperty(value, build_ak168fqfikdepd6py4yef8tgs.class, this, ShortTypeHandling.castToString(new GStringImpl(new Object[]{name}, new String[]{"", ""})));} } 復制代碼

    可以看到,腳本類繼承自 ProjectScript,實現了 run 方法。
    run 方法里做了些什么呢,先看第一行,

    CallSite[] $getCallSiteArray = build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray(); 復制代碼

    獲取到 callsiteArray,這個就是 createCallSiteArray_1() 方法中賦值的,可以看到,此處的 callsiteArray,都是腳本中的 dsl,其實也就是調用的方法名。 獲取到 callsiteArray 以后,執行 $getCallSiteArray[0].callCurrent() 類似的方法,這個就是在調用方法。調用的方法對應的腳本代碼在下面加了注釋。

    public Object run() {CallSite[] $getCallSiteArray = build_ak168fqfikdepd6py4yef8tgs.$getCallSiteArray();// apply plugin "com.android.application" 依賴插件$getCallSiteArray[0].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "com.android.application"}));// apply plugin myplugin$getCallSiteArray[1].callCurrent(this, ScriptBytecodeAdapter.createMap(new Object[]{"plugin", "myplugin"}));// android {}$getCallSiteArray[2].callCurrent(this, new _run_closure1(this, this));// dependencies {} $getCallSiteArray[3].callCurrent(this, new _run_closure2(this, this));// task {}$getCallSiteArray[17].callCurrent(this, "task1", new _run_closure6(this, this));// ...return $getCallSiteArray[19].call($getCallSiteArray[20].callGroovyObjectGetProperty(this), $getCallSiteArray[21].callGroovyObjectGetProperty(this)); } 復制代碼

    上面看到,task1 對應的是 _run_closure6 這個類,我們看看這個類的內容。

    /* compiled from: /Users/zy/workspace/note/blog/android-training/gradle/EasyGradle/app/build.gradle */ public class build_ak168fqfikdepd6py4yef8tgs$_run_closure6 extends Closure implements GeneratedClosure, ScriptOrigin {private static final /* synthetic */ String __originalClassName = "_BuildScript_$_run_closure6";private static /* synthetic */ CallSiteArray $createCallSiteArray() {String[] strArr = new String[1];strArr[0] = "doLast";return new CallSiteArray(build_ak168fqfikdepd6py4yef8tgs$_run_closure6.class, strArr);}public build_ak168fqfikdepd6py4yef8tgs$_run_closure6(Object _outerInstance, Object _thisObject) {build_ak168fqfikdepd6py4yef8tgs$_run_closure6.$getCallSiteArray();super(_outerInstance, _thisObject);}public Object doCall() {build_ak168fqfikdepd6py4yef8tgs$_run_closure6.$getCallSiteArray();return doCall(null);}public Object doCall(Object it) {return build_ak168fqfikdepd6py4yef8tgs$_run_closure6.$getCallSiteArray()[0].callCurrent(this, new _closure17(this, getThisObject()));} } 復制代碼

    省略了一些內容,可以看到,這個閉包的類繼承了 Closure,然后實現了 doCall 方法,在 doCall 方法里,調用了 doLast 方法,傳入了 _closure17 實例。這個就是腳本中的 task { doLast {} } 對應的實現。
    我們再看看 _closure17 的實現。

    /* compiled from: /Users/zy/workspace/note/blog/android-training/gradle/EasyGradle/app/build.gradle */ public class build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17 extends Closure implements GeneratedClosure, ScriptOrigin {private static /* synthetic */ SoftReference $callSiteArray = null;private static /* synthetic */ ClassInfo $staticClassInfo = null;public static transient /* synthetic */ boolean __$stMC = false;private static final /* synthetic */ String __originalClassName = "_BuildScript_$_run_closure6$_closure17";private static final /* synthetic */ String __signature = "ab46bccc923a8e0a93329f7333d732c8";private static /* synthetic */ CallSiteArray $createCallSiteArray() {String[] strArr = new String[1];strArr[0] = "println";return new CallSiteArray(build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17.class, strArr);}public Object doCall() {build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17.$getCallSiteArray();return doCall(null);}public Object doCall(Object it) {return build_ak168fqfikdepd6py4yef8tgs$_run_closure6$_closure17.$getCallSiteArray()[0].callCurrent(this, "task2");} } 復制代碼

    同樣也是繼承了 Closure,在 doCall 方法里調用了 println,這正是我們在 task 的里執行的任務,也就是前面提到的 task 的 actions。

    這里我們再理順一下,每一個 build.gradle 腳本,對應一個繼承了 ProjectScript 的類,每一個閉包,對應了一個繼承自 Closure 的類

    7.2 調用腳本 run 方法

    接著就是執行腳本類的 run 方法,也就是我們在上面分析的 run 方法。
    其中強調的一點是,run 方法里對 task 的創建,僅僅是執行了 task.doCall,這也就是為什么配置階段不會執行 task 任務,但會執行 task 閉包里的內容。

    task task1 {// 配置階段會執行println('configure')doLast {// 運行階段執行println('run')} } 復制代碼

    八、插件調用流程

    之前在 Gradle的基本使用 里講到過自定義插件,使用的時候是通過 apply plugin 'xxx' 來使用的,具體的調用鏈路如下:

    apply: "xxx" -> Script.run -> ProjectScript.apply -> DefaultObjectConfigurationAction.run -> DefaultObjectConfigurationAction.applyType(pluginId) -> DefaultPluginManager.apply -> DefaultPluginManager.AddPluginBuildOperation.run -> AddPluginBuildOperation.addPlugin -> RuleBasedPluginTarget.applyImpreative -> ImperativeOnlyPluginTarget.applyImperative -> Plugin.apply 復制代碼

    最后的 Plugin.apply 就調用到插件里實現的 apply() 函數了

    九、總結

    整體結構圖

  • gradle 運行流程
  • loadSettings configureBuild constructTaskGraph runTasks finishBuild 復制代碼
  • Task 的本質,就是一系列的 Actions
  • 腳本編譯流程 獲取腳本內容 -> 編譯成 class 文件,繼承自 ProjectScript -> 執行 ProjectScript.run 方法
  • 腳本的 buildscript 在腳本其他內容前執行
  • 【Android 修煉手冊】系列內容 每周更新 歡迎關注下面賬號,獲取更新:
    微信搜索公眾號: ZYLAB
    Github
    知乎
    掘金

    總結

    以上是生活随笔為你收集整理的【Android 修炼手册】Gradle 篇 -- Gradle 源码分析的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    亚洲码国产日韩欧美高潮在线播放 | www.夜夜操.com | 成人免费视频在线观看 | 久久久久久蜜桃一区二区 | 国产成人久久久久 | 在线视频观看你懂的 | 在线91视频 | 一级片免费观看视频 | 久草在线视频新 | 99热999| 久久国产精品久久精品国产演员表 | 成人h电影 | 欧美一级视频一区 | www.伊人色.com | 久久美女精品 | 丁香五月网久久综合 | 少妇搡bbbb搡bbb搡aa | 夜色.com | 天天干天天干天天射 | 国产精品18毛片一区二区 | 天天操夜夜逼 | 中文字幕在线高清 | 中文字幕久久精品亚洲乱码 | 免费成人黄色片 | 91精品久久久久久综合乱菊 | 国产99久久九九精品免费 | 国产午夜精品一区 | 天天碰天天操视频 | 黄色激情网址 | 亚洲成人国产精品 | 91视视频在线直接观看在线看网页在线看 | 国内三级在线观看 | 天天操天天操天天 | 亚洲免费a| 在线成人av| 中文字幕黄网 | 青青河边草手机免费 | 国产91在线观 | 99久久这里有精品 | 在线色视频小说 | 天天做夜夜做 | 久久图 | 日韩视频免费在线观看 | 亚洲精品在线免费观看视频 | 国产精品va视频 | 久久久麻豆视频 | www.97视频| 一区二区三区在线视频111 | 亚洲国产三级在线观看 | 日韩欧美视频二区 | 网站在线观看你们懂的 | 日韩av资源在线观看 | 97视频网址 | 亚洲乱码在线观看 | 亚洲视频综合 | 亚州精品天堂中文字幕 | 中文亚洲欧美日韩 | www操操操 | 高潮毛片无遮挡高清免费 | 欧洲黄色片 | 天无日天天操天天干 | 婷婷福利影院 | 丁香一区二区 | 操操日 | 96av麻豆蜜桃一区二区 | 九九九在线观看视频 | 美女视频黄网站 | 97超碰伊人| 欧美午夜理伦三级在线观看 | 最近免费观看的电影完整版 | 国产在线精品区 | 久久久久免费视频 | 色综合国产| 久久在线影院 | 日韩精品中文字幕久久臀 | 日韩欧美在线观看一区二区三区 | 精品美女在线视频 | 99se视频在线观看 | 国产高清在线一区 | 午夜视频在线观看一区二区 | 国产视频资源 | 91九色最新 | 91麻豆精品国产91久久久更新时间 | 91午夜精品 | 亚洲男男gaygay无套同网址 | 91久久国产综合精品女同国语 | 亚洲一区二区三区91 | 欧美在线1 | 一区二区av | 国产在线a| 国产小视频在线免费观看视频 | 视频三区在线 | 色欧美88888久久久久久影院 | 99久久精品久久久久久清纯 | 日韩在线小视频 | 一区二区视频在线观看免费 | 国产99久 | 99麻豆视频 | 最近高清中文字幕 | 亚洲免费资源 | 国内外激情视频 | 国外成人在线视频网站 | 996久久国产精品线观看 | 欧美日韩国内在线 | 亚洲人片在线观看 | 亚av在线 | 久久婷婷综合激情 | 91人人澡人人爽 | 欧日韩在线视频 | 亚洲精品视频网址 | 久久久99精品免费观看乱色 | 亚洲欧洲av| 日韩网页 | 国产精品欧美精品 | 久久电影中文字幕视频 | 天天操综合网站 | 五月婷在线观看 | 亚洲精品国偷自产在线99热 | av超碰在线 | 国产高清视频在线观看 | 色婷婷 亚洲 | 91女神的呻吟细腰翘臀美女 | 狠狠躁夜夜躁人人爽超碰97香蕉 | 丝袜+亚洲+另类+欧美+变态 | 日本久久精 | 午夜精品婷婷 | 99久久精品国产一区二区成人 | 日本公乱妇视频 | 99精品欧美一区二区 | 久久精品视频免费 | 亚洲欧美成aⅴ人在线观看 四虎在线观看 | 在线观看mv的中文字幕网站 | 国产精品成人一区二区三区 | 欧美高清成人 | 亚洲在线不卡 | 五月天亚洲婷婷 | 一区在线电影 | 久久区二区| 免费观看mv大片高清 | 色黄久久久久久 | 国产婷婷色 | 人人爽人人香蕉 | 黄a在线 | 激情www| 免费av 在线 | 最新av中文字幕 | 免费观看版 | 九九九九色 | 欧美久久久一区二区三区 | 欧美精品一区二区在线观看 | 在线观看视频黄色 | 麻豆视频91 | 亚洲黄色区 | 超碰在线官网 | 有码中文字幕在线观看 | 欧美精品一区二区蜜臀亚洲 | 五月天激情视频在线观看 | 超碰免费久久 | 久久人人爽人人爽人人片av软件 | 在线观看亚洲国产 | av.com在线 | 中文字幕av全部资源www中文字幕在线观看 | 国产精品乱看 | 亚州精品在线视频 | 久久久久久久99精品免费观看 | 一个色综合网站 | 91视频首页 | 亚洲成人一二三 | 久久久午夜电影 | 91欧美日韩国产 | 色网站免费在线观看 | 亚洲精品久久久久999中文字幕 | 日韩在线网址 | 免费黄色在线播放 | 91在线视频免费观看 | 欧美日韩不卡在线 | 91精品久久久久久综合五月天 | 亚洲综合在线五月 | 国产电影黄色av | 美女视频永久黄网站免费观看国产 | av3级在线| 国产一二三在线视频 | 成人免费观看在线视频 | 国产专区一 | 亚洲 成人 一区 | 亚洲综合狠狠干 | 网站免费黄 | 国产999久久久 | 久久久久久免费视频 | 日韩中文字幕亚洲一区二区va在线 | 婷婷激情5月天 | 黄色免费网站下载 | 国产美女视频网站 | 黄色网在线免费观看 | 超碰在线人人爱 | 欧美亚洲久久 | 日韩欧美在线播放 | 99精品一区| 国产99一区二区 | 国产蜜臀av | 亚洲精品美女免费 | 欧美午夜精品久久久久 | 在线国产欧美 | 日韩欧美国产成人 | 久久综合影视 | 亚洲男男gⅴgay双龙 | 日韩视频a | 久久久精品免费观看 | 一区视频在线 | 亚洲最大av在线播放 | 在线观看精品一区 | 中国一级片免费看 | 亚洲国产精品电影 | 一区二区三区日韩在线观看 | 精品国产一区二区在线 | 欧美日韩国产一区二区在线观看 | 99精品福利 | 六月丁香在线观看 | 在线看一区| 国产一区二区三精品久久久无广告 | 欧美超碰在线 | 久久久五月天 | 欧美日韩高清国产 | 国产视频一区二区在线 | 国产精品久久久久久久7电影 | 中文字幕专区高清在线观看 | 国产女人免费看a级丨片 | 97国产在线视频 | 婷婷色在线播放 | www.精选视频.com | 成片人卡1卡2卡3手机免费看 | 69精品在线观看 | 国产xx视频 | www.天天综合| 亚洲精品免费观看 | 国产精品久久久久久久久免费看 | 国产亚洲精品久久久久久 | 精品国产一区二区三区久久 | 91在线免费视频 | 中文字幕免费高清 | 一级黄色片在线免费看 | 超碰在线亚洲 | 青青草久草在线 | 看v片| 婷婷在线综合 | 久热色超碰 | 国产精品免费久久久久影院仙踪林 | 91香蕉视频污在线 | 色五月激情五月 | 亚洲va韩国va欧美va精四季 | 日本久久电影网 | 92精品国产成人观看免费 | 婷婷丁香狠狠爱 | 欧美一级专区免费大片 | 国产精品高清在线观看 | 日韩在线视频在线观看 | 天天色天天色天天色 | 88av网站 | 激情av网址| 欧美日韩国产一区 | 中文十次啦 | 国内精品久久久久国产 | 日韩免费专区 | 亚洲 欧美 成人 | 久久久首页| 色婷婷在线观看视频 | 最新av在线免费观看 | 欧美在线视频一区二区三区 | 亚洲精品99久久久久久 | 久久一区二区三区超碰国产精品 | 色偷偷男人的天堂av | 婷婷丁香激情 | 久久综合久久综合久久综合 | 成人资源站 | 三级黄色网络 | 成人小视频免费在线观看 | 中文字幕国产精品一区二区 | 中文字幕一区二区三区乱码不卡 | 亚洲国产精品视频 | 亚洲成人精品在线 | www国产亚洲精品久久麻豆 | 天天天天综合 | 伊人激情网 | 最近免费观看的电影完整版 | 一区二区三区动漫 | 中文字幕国产精品一区二区 | 97视频资源 | 亚洲午夜久久久久久久久 | 99精品视频在线观看免费 | 日韩高清无线码2023 | 久久久久99精品成人片三人毛片 | 九九视频精品在线 | 国产999精品久久久久久绿帽 | 91看片淫黄大片在线播放 | 91精品视频免费看 | 中文字幕a在线 | 亚洲aaa毛片 | 日韩免费电影一区二区 | 国产精品久久久久久69 | 日本公乱妇视频 | 成人黄色毛片视频 | 日批视频在线播放 | 国产成人精品久久二区二区 | 六月丁香综合网 | 欧洲亚洲精品 | 国产日韩欧美在线观看 | 欧美国产视频在线 | 91亚洲综合| 亚洲欧美日韩精品一区二区 | 免费看片日韩 | 99视频 | 91精品国产99久久久久 | 色婷婷综合五月 | 久久伊人操| 欧美精品久久久久久久久久久 | 国产精品一区二区三区免费看 | 特级毛片爽www免费版 | 久久久久久亚洲精品 | 亚洲免费永久精品国产 | 97精品国自产拍在线观看 | 久久久午夜视频 | 亚洲欧美国产精品va在线观看 | 日韩r级在线 | 久久久国产精品电影 | 日日夜夜中文字幕 | 国产精品美女久久久久久久久久久 | 久久综合狠狠综合久久狠狠色综合 | 久久久影视 | 婷婷久月| 婷婷在线免费视频 | 夜夜摸夜夜爽 | 亚洲精品在线一区二区三区 | 亚洲视频 中文字幕 | 亚洲精品动漫久久久久 | 久久99久久99精品免观看软件 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 成人av电影免费在线播放 | 天天躁日日躁狠狠躁av麻豆 | 日批视频在线观看免费 | 国产日韩精品一区二区三区在线 | 国产精品一区二区在线观看 | 国内精品视频在线 | a视频在线播放 | 精品国产成人 | 在线观看一区二区视频 | 欧美一级黄色视屏 | 中文字幕在线播放日韩 | 午夜免费福利片 | 国产黄在线免费观看 | 国产午夜精品久久久久久久久久 | 九九久久成人 | 偷拍视频一区 | 国产激情电影综合在线看 | 亚洲欧美日韩不卡 | 国产亚洲欧美精品久久久久久 | 黄色视屏av | 91精品在线免费视频 | 国产成人久 | 欧美巨大荫蒂茸毛毛人妖 | 免费碰碰| 欧美亚洲一级片 | 黄色软件在线看 | 日韩成人av在线 | 日韩黄色在线电影 | 五月导航| 99婷婷 | 狠狠色丁香婷婷综合久小说久 | 成人午夜电影免费在线观看 | 欧美精品在线观看免费 | 国产成人精品免高潮在线观看 | 中文一区在线 | 久久超碰网 | 久久精品艹 | 视频一区亚洲 | 探花视频免费观看高清视频 | 91麻豆视频 | 丁香一区二区 | 国产最新在线视频 | 婷婷丁香社区 | 麻豆国产网站 | 999国内精品永久免费视频 | 少妇av片| 日韩一级成人av | 狂野欧美激情性xxxx | 成人三级av | 天堂av在线网站 | 日本黄色免费网站 | 国产成人精品一区二区三区在线 | 亚洲最大的av网站 | 91看片看淫黄大片 | 日日夜夜天天久久 | 亚洲一二区视频 | 亚洲一区二区精品在线 | 97超碰国产在线 | 国产成人精品久久亚洲高清不卡 | 久久久国产一区二区三区 | 一区二区影院 | 人人射| 最新动作电影 | 国产在线精品一区 | 久草在线费播放视频 | 美女搞黄国产视频网站 | 久久久久一区二区三区 | av免费网站 | 91成人在线观看喷潮 | 国产一级大片在线观看 | 婷婷电影网| 久久伊人热 | 男女拍拍免费视频 | 色欧美日韩 | 日韩免费在线观看视频 | 日韩一二三 | 丁香电影小说免费视频观看 | 亚洲成av人片在线观看香蕉 | 美女网站视频免费都是黄 | 激情五月六月婷婷 | 九九热在线精品 | 免费色视频网址 | 粉嫩av一区二区三区四区在线观看 | 久久免费视频2 | 狠狠色丁香婷婷综合欧美 | 操碰av | aaa亚洲精品一二三区 | 日韩欧美高清 | 天天干com| 中字幕视频在线永久在线观看免费 | 香蕉久久国产 | 在线成人短视频 | 综合网五月天 | 午夜色大片在线观看 | 亚洲免费精品一区二区 | 五月情婷婷 | 亚洲综合色视频在线观看 | 婷色| 91免费在线视频 | 高清av免费观看 | 天天综合网在线 | 国产伦理一区二区三区 | 亚洲va天堂va欧美ⅴa在线 | 日韩.com | 999成人网 | 亚洲精品综合一区二区 | 波多野结衣一区三区 | 91亚洲精品久久久中文字幕 | 欧美色图狠狠干 | 91麻豆精品国产午夜天堂 | 欧美日韩国产伦理 | 成人黄色av免费在线观看 | 久久成年视频 | 成人a v视频 | 黄色免费网站 | 成人av地址 | 成人试看120秒 | 狠狠色噜噜狠狠狠狠2021天天 | 亚洲成人精品在线观看 | 国产99精品 | 欧美性爽爽| 开心丁香婷婷深爱五月 | 久久久久亚洲精品成人网小说 | 97日日碰人人模人人澡分享吧 | 久久久激情网 | 成人av影院在线观看 | 国产精品自在欧美一区 | www.com在线观看| 欧美福利网站 | 亚洲成人午夜av | 五月天综合激情网 | 久久久精品国产一区二区三区 | 天天色官网 | 美女视频黄频大全免费 | 中文字幕亚洲在线观看 | 久久 地址 | 成人黄色电影免费观看 | 亚洲爱av | 国产成在线观看免费视频 | 国产成人福利片 | 精品1区2区3区 | 国产999精品久久久久久 | 午夜精品久久一牛影视 | 日韩一区在线播放 | 久久久久国产精品免费免费搜索 | 中国精品少妇 | 色欧美88888久久久久久影院 | 天天干天天射天天爽 | 国内精品久久久 | 黄色毛片在线 | 国产a级精品 | 国产精品精品久久久久久 | 日本一区二区三区免费观看 | 精品国产成人在线影院 | 激情欧美日韩一区二区 | 国产日产精品一区二区三区四区 | 国产区精品 | 高清免费在线视频 | 国产成人综 | 国产精品美女免费 | 国产成人一区在线 | 亚洲成人黄色 | 黄色小视频在线观看免费 | 国产亚洲情侣一区二区无 | 久久一区二 | 日韩免费视频观看 | 国产午夜精品福利视频 | 丁香婷婷在线 | 亚洲午夜不卡 | 奇米777777| 操操日日 | 在线国产视频 | 欧美中文字幕久久 | 久久综合久久综合这里只有精品 | 亚洲永久精品在线观看 | 久久久久国产精品免费免费搜索 | 欧美另类视频 | 美女视频免费一区二区 | 精品国产伦一区二区三区免费 | 香蕉视频网站在线观看 | 国产三级久久久 | 天堂av在线中文在线 | 久久av免费 | 97在线播放视频 | 免费看污网站 | 国内精品久久久久久久影视麻豆 | 欧美另类xxx | 国产资源站 | 色婷婷综合久久久中文字幕 | jizz18欧美18| 国产精品久久久久毛片大屁完整版 | 看av在线| 在线免费观看黄色 | 欧美日韩亚洲在线 | 亚洲爱爱视频 | 婷婷久久一区 | 美女网站免费福利视频 | 国产精品你懂的在线观看 | 毛片a级片 | 91亚洲成人 | 亚洲欧美国产日韩在线观看 | 精品国产一区二区三区久久久 | 四虎最新域名 | 国产高清日韩 | 天堂在线免费视频 | 国产福利一区二区三区在线观看 | 成年人电影免费在线观看 | 色婷婷综合五月 | 综合色在线 | 国产成人av网 | 国产成人精品久久久 | 久久久久久久久久久免费 | 偷拍久久久| 国产色拍拍拍拍在线精品 | 国产一级在线看 | 精品一区二区免费在线观看 | av线上看 | 午夜精品区 | 成人亚洲欧美 | 天天爱天天爽 | 亚洲综合视频在线 | 91精品一区在线观看 | 在线亚洲欧美视频 | 日韩欧美视频免费在线观看 | 日日夜夜狠狠干 | 久久免费国产电影 | 色婷婷亚洲婷婷 | 日韩影视在线 | 人人爽人人插 | 国产亚洲观看 | 亚洲综合视频在线观看 | 九色激情网| 美女av免费 | 91毛片在线| 国产黄在线播放 | 久久国产手机看片 | 激情五月婷婷综合 | 亚洲一级黄色大片 | www.狠狠色| 久久精品一区二区三区国产主播 | 欧美成人69av | 伊人亚洲综合 | 黄色电影小说 | 久艹在线免费观看 | 日本黄色大片免费看 | 国产中文字幕久久 | 国产在线一卡 | 婷婷精品进入 | 国产精品国产三级国产aⅴ9色 | 人人插人人做 | 四川bbb搡bbb爽爽视频 | 99久久99久久精品国产片 | 91最新中文字幕 | 国产精品自拍在线 | 在线导航福利 | 91精品无人成人www | 可以免费看av| 久久这里精品视频 | 干狠狠| 色综合激情网 | 亚洲经典在线 | 五月天亚洲综合小说网 | 国产美腿白丝袜足在线av | av在线网站免费观看 | 中文字幕在线成人 | 欧美日韩国产一区二区三区在线观看 | 成人午夜电影在线观看 | 久久久视频在线 | 久久新视频 | 人人艹视频 | 天天爽夜夜爽人人爽曰av | 在线观看黄色的网站 | 精品国产欧美一区二区三区不卡 | 五月天婷亚洲天综合网精品偷 | 四虎在线视频免费观看 | 在线免费观看黄色小说 | 国产在线精品二区 | 麻豆视频免费 | 99久久一区 | 国产高清久久久 | 日本不卡123 | 久久亚洲影视 | 亚洲精品高清视频在线观看 | 91视频免费 | 欧美日本啪啪无遮挡网站 | 亚洲h在线播放在线观看h | 丁香五婷 | 婷婷草| 久久久久久久久久久精 | 亚洲成av人片在线观看www | 日韩av二区 | 久久看片网站 | 精品无人国产偷自产在线 | 成年人免费在线观看网站 | 涩涩网站在线观看 | 国产成人黄色 | 久久久午夜电影 | 久久精品麻豆 | 久久久国产一区 | 伊人www22综合色 | 在线亚洲人成电影网站色www | 久久精品99精品国产香蕉 | 999久久a精品合区久久久 | 超碰999 | 91精彩在线视频 | 欧美日韩免费一区二区三区 | 黄视频网站大全 | 婷婷色社区 | 在线91网| 欧美色综合久久 | 国产精品永久免费 | 欧美一二三区在线观看 | 国产精品一区在线 | 香蕉97视频观看在线观看 | 国产中文字幕久久 | 免费成人在线视频网站 | 日韩激情小视频 | 婷婷激情站| 亚洲日本在线视频观看 | 麻豆视频免费在线播放 | 五月综合网站 | 天天操操操操操 | 久草在线免费看视频 | 国产专区精品 | 精品国产一区二区三区蜜臀 | 99热99re6国产在线播放 | 成人免费观看网站 | 一二区av | 亚洲精品午夜视频 | 麻豆视频一区二区 | 国产成人中文字幕 | 国产一区二区三区 在线 | 激情五月婷婷综合 | 狠狠狠狠狠狠干 | 91精品在线免费视频 | 天堂视频中文在线 | 五月开心色 | 91污视频在线观看 | 97看片网| 日本精品视频网站 | 亚洲精品国产免费 | 国产高清无线码2021 | 婷婷综合国产 | 国产丝袜制服在线 | 欧美精彩视频在线观看 | 国产精品a久久 | 国产精品久久久久影院 | 在线亚洲精品 | 天天操夜夜拍 | 亚洲综合精品在线 | 免费看片网址 | 伊人婷婷在线 | 国产激情久久久 | 国产综合在线视频 | 免费在线看成人av | 色综合久久久久久久久五月 | 激情五月看片 | 色狠狠狠| 4p变态网欧美系列 | 精品乱码一区二区三四区 | 国产精品18久久久久vr手机版特色 | 亚洲国内精品视频 | 亚洲妇女av| 中文字幕激情 | 成人av一区二区三区 | 日韩欧美网站 | 国产精品久久久久久久久久99 | 日韩中文在线视频 | 国产原厂视频在线观看 | 国产精品麻豆99久久久久久 | 久久开心激情 | 亚州欧美精品 | 国产亚洲成人精品 | 亚洲 中文 欧美 日韩vr 在线 | 久久情网 | 午夜国产福利在线 | 免费a级大片 | 免费观看一级一片 | 久久涩涩网站 | 国产人免费人成免费视频 | 国内精品久久久久影院优 | 国产精品美 | 国产精品成人品 | 91亚洲精品久久久久图片蜜桃 | 色搞搞 | 亚洲国产精品999 | 久久久久久国产精品亚洲78 | 日本中文字幕网址 | 337p日本大胆噜噜噜噜 | 亚洲精品中文字幕在线观看 | 久一网站| 欧美综合在线视频 | 18女毛片 | 成人午夜精品久久久久久久3d | 国产精品久久久久久模特 | 国产精品女同一区二区三区久久夜 | 国产无套精品久久久久久 | 国产一级二级在线 | 正在播放一区 | 成 人 黄 色 免费播放 | 久久天天躁狠狠躁夜夜不卡公司 | 免费看污网站 | 国产精品欧美一区二区 | 麻豆播放 | 久久综合色8888 | 国产午夜精品福利视频 | 国产精品1区2区在线观看 | 国产精品视频999 | 久草9视频 | 亚洲一区二区三区在线看 | 蜜臀精品久久久久久蜜臀 | 97人人射 | 国产成人av | 黄色a一级片 | 五月综合激情网 | 中文字幕精品www乱入免费视频 | 日本久热 | 天天操天天射天天 | 亚洲2019精品 | 日本性xxxxx| 五月婷婷另类国产 | 91丨porny丨九色 | 国产精品视频地址 | 欧美精品黑人性xxxx | 夜夜躁日日躁狠狠久久88av | 亚洲精品综合久久 | 国产永久免费观看 | 日韩欧美高清不卡 | 91人人爽人人爽人人精88v | 激情综合久久 | 久久香蕉国产 | 在线播放视频一区 | 丁香一区二区 | 婷婷色站| 成年人免费看片 | 天天做天天射 | 五月婷网站| 又黄又爽又湿又无遮挡的在线视频 | 五月婷婷国产 | 日韩精品在线播放 | 狠狠操导航 | 免费高清在线视频一区· | 色wwwww| 国产中文欧美日韩在线 | 天堂av色婷婷一区二区三区 | 日韩精品影视 | 日韩欧美视频一区二区 | 久久国产精品99精国产 | 久久激情片 | 亚洲精品欧洲精品 | 日韩一级电影在线 | 91最新网址 | 亚洲小视频在线观看 | 免费福利视频网站 | 蜜臀av夜夜澡人人爽人人桃色 | 狠狠色噜噜狠狠狠合久 | 黄色a视频免费 | 亚洲综合狠狠干 | 色视频国产直接看 | 91高清完整版在线观看 | 午夜精选视频 | 自拍超碰在线 | 在线看中文字幕 | 国产精品麻豆视频 | 国产精品女主播一区二区三区 | 在线之家免费在线观看电影 | 久久国产电影 | 国产精品久久久亚洲 | 成人免费视频观看 | 久草观看| 中文字幕亚洲欧美 | 91看片淫黄大片一级在线观看 | 日韩精品一区二区三区免费观看 | 国产精品 美女 | 国产短视频在线播放 | 亚洲精品玖玖玖av在线看 | 欧美激情综合五月色丁香小说 | 亚洲激情六月 | 在线免费黄网站 | 丁香五香天综合情 | 日韩性xxxx | 中文在线免费视频 | 国产私拍在线 | 亚洲3级 | 色婷婷色| 亚洲黄色片一级 | 国产伦精品一区二区三区免费 | 99免费观看视频 | 中文国产成人精品久久一 | 国产精品午夜av | 一级性视频 | 色六月婷婷| 黄色三级免费片 | www久草| 久久精品屋 | 国产精品欧美久久 | 婷婷深爱| 成人91av | 视频三区在线 | 日韩欧美视频一区二区三区 | 成人在线免费观看视视频 | av在线播放国产 | 日韩高清一区 | 狠狠色丁香九九婷婷综合五月 | 天天色天天草天天射 | 在线免费色视频 | 国产一级片网站 | 色视频网页 | 成人羞羞视频在线观看免费 | 亚洲成av人片一区二区梦乃 | 久久手机看片 | 久久成年视频 | 精品在线视频一区二区三区 | 五月天堂网 | 国产在线观看 | 亚洲精品国产日韩 | 成年在线观看 | 在线 高清 中文字幕 | 亚洲精品国产自产拍在线观看 | 国产精品欧美久久久久天天影视 | 999国产| 亚洲一级片在线观看 | 日本黄色免费看 | 亚洲精品在线观看免费 | 久久久久免费视频 | 午夜精品久久久久久99热明星 | 国产高清精品在线 | 日韩精品中文字幕久久臀 | 91桃色免费观看 | 亚洲精品久久久久999中文字幕 | 欧美日韩视频在线观看免费 | 成人一级在线 | 丝袜av网站| 日本久久91 | 91在线视频播放 | 91丨九色丨国产在线 | 欧美日韩国产在线 | 久久精品国产一区二区三 | 久久黄色网页 | 成年人国产视频 | 色婷婷久久久 | 在线观看一级视频 | 一区二区三区在线播放 | 精品国产乱码一区二区三区在线 | 久久精品日产第一区二区三区乱码 | 玖玖视频精品 | 又黄又爽的免费高潮视频 | 国产成人一级 | 免费看成人a | 激情综合站 | 国产精品99久久久久 | 免费a网 | 精品国偷自产国产一区 | 日韩性xxx| 天天综合天天综合 | 99免费在线观看 | 亚洲第一成网站 | 欧洲精品亚洲精品 | 久久免费片| 99久久精品免费看国产麻豆 | 国产另类xxxxhd高清 | 超碰成人网| 天天激情综合网 | 美女中文字幕 | 久久精品视频播放 | 国产中文字幕大全 | 日韩精品在线视频 | 国产成人精品一区二区三区网站观看 | 天天看天天操 | 欧日韩在线视频 | 狠狠干天天操 | 久久综合免费视频 | 碰超在线| 久久一精品 | 丁香六月色 | 久久久久女人精品毛片九一 | 黄色aaaaa| 久久精品男人的天堂 | 午夜影院一级 | 成人免费毛片aaaaaa片 | 最新日韩在线观看视频 | av在线免费观看不卡 | 亚洲人成影院在线 | 亚洲网站在线看 | 亚洲精品免费在线视频 | 碰超在线观看 | 精品视频免费观看 | 国产视频一区二区三区在线 | 日本性高潮视频 | 日韩电影一区二区在线 | 区一区二区三在线观看 | 人人澡人人添人人爽一区二区 | 一区二区精品在线视频 | www.av在线播放 | 欧美日韩精品免费观看视频 | 狠狠色免费 | 国产精品久久久久久久久久久久午夜 | 久久深夜 | 中文字幕资源在线观看 | 国产黄色电影 | av短片在线观看 | 久草网站在线 | 久久网站最新地址 | 黄色国产成人 | 免费看成年人 | 91麻豆精品国产自产在线 | 国产一级片在线播放 | 九草视频在线观看 | 精品字幕 | 日韩成人免费观看 | 黄色影院在线免费观看 | 超碰在线98 | 亚洲欧美日韩在线一区二区 | 91亚洲精品久久久 | 久久久www成人免费精品张筱雨 | 美女福利视频一区二区 | 中中文字幕av在线 | 久久久麻豆精品一区二区 | 三级黄色在线观看 | 九九热免费视频在线观看 | 狠狠狠色丁香婷婷综合激情 | 欧美日韩首页 | 夜夜视频资源 | 成人av影视 | 丁香综合 | 国产不卡视频在线播放 | 国产一级片免费播放 | 插插插色综合 | 超碰在线天天 | 欧美日韩中文字幕在线视频 | 天天透天天插 | 久久99久久99精品免观看软件 | 亚洲精品综合在线观看 | 午夜精品视频福利 | 91色影院 | 亚洲精品一区二区三区高潮 | 日本久久综合视频 | 99久久精品一区二区成人 | 六月丁香伊人 | 丁香网五月天 | 2023亚洲精品国偷拍自产在线 | 91精品国自产在线偷拍蜜桃 | 91成人小视频 | 国产人成在线观看 | 精品久久久久久国产偷窥 | 亚洲精品国产精品国自产观看 | 日韩欧美一区二区不卡 | 97av视频在线观看 | 亚洲情婷婷 | 国产精品欧美久久久久无广告 | 狠狠躁夜夜躁人人爽视频 | 天天综合成人网 | 激情久久伊人 | 婷婷夜夜 | 午夜精品久久一牛影视 | 午夜久久久久久久久久久 | 制服丝袜在线91 | 欧美日韩一区二区在线观看 | 黄色av电影网 | 久草剧场| 91超在线 | 中文字幕精品一区二区精品 |