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

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 运维知识 > Android >内容正文

Android

在 Mac 上为 Android 编译 WebRTC

發(fā)布時(shí)間:2024/4/11 Android 75 豆豆
生活随笔 收集整理的這篇文章主要介紹了 在 Mac 上为 Android 编译 WebRTC 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

在 Mac 上為 Android 編譯 WebRTC 的基本流程和在任意平臺(tái)上編譯任何其它目標(biāo)平臺(tái)的 WebRTC 大體一致,但在 Mac 上為 Android 編譯 WebRTC 不是 WebRTC 官方正式支持的 WebRTC 的構(gòu)建方式,因而需要針對(duì)這種構(gòu)建方式,對(duì) WebRTC 構(gòu)建系統(tǒng)中的部分配置和工具做一些適當(dāng)?shù)恼{(diào)整。本文將描述在 Mac 上為 Android 編譯 WebRTC 的流程,流程中各個(gè)步驟將會(huì)遇到的問(wèn)題,問(wèn)題發(fā)生原因的簡(jiǎn)單解釋,以及問(wèn)題的解決方法。

在 Mac 上為 Android 編譯 WebRTC 的過(guò)程如下。

1. 創(chuàng)建 args.gn 文件

為 Android 版 WebRTC 的構(gòu)建創(chuàng)建 args.gn 文件做一些全局的構(gòu)建配置開關(guān),如:

is_component_build = false rtc_build_tools = false rtc_include_tests = false treat_warnings_as_errors = false use_rtti = false use_dummy_lastchange = true is_clang = true target_cpu = "arm64" is_debug = true symbol_level = 2 clang_base_path = "/Users/lisi/Projects/opensource/OpenRTCClient/build_system/llvm-build/mac/mac/Release+Asserts" android_sdk_build_tools_version = "30.0.1" target_os = "android" android_ndk_root = "/Users/lisi/Library/Android/sdk/ndk/23.1.7779620" android_sdk_root = "/Users/lisi/Library/Android/sdk" clang_use_chrome_plugins = false lint_android_sdk_root = "/Users/lisi/Library/Android/sdk" use_custom_libcxx = false use_sysroot = false enable_libaom = false enable_libaom_decoder = false rtc_use_h264 = true rtc_enable_protobuf = true rtc_include_ilbc = false rtc_libvpx_build_vp9 = true ffmpeg_branding = "Chrome"

2. gn gen 生成構(gòu)建配置文件

通過(guò) gn gen 根據(jù)創(chuàng)建的 args.gn 文件,各個(gè) BUILD.gn 和 *.gni 構(gòu)建配置文件,生成 ninja 構(gòu)建所需的配置文件。假設(shè) args.gn 文件位于 build/android/arm64/debug 目錄下,執(zhí)行如下命令:

OpenRTCClient % gn gen build/android/arm64/debug

這里就報(bào)了錯(cuò),報(bào)錯(cuò)信息如下:

OpenRTCClient % gn gen build/android/arm64/debug ERROR Unresolved dependencies. //examples/androidapp/third_party/autobanh:autobanh_java__header(//build/toolchain/android:android_clang_arm64)needs //third_party/ijar:ijar(//build/toolchain/mac:clang_x64) //third_party/accessibility_test_framework:accessibility_test_framework_java__header(//build/toolchain/android:android_clang_arm64)needs //third_party/ijar:ijar(//build/toolchain/mac:clang_x64) //third_party/android_deps:android_arch_core_common_java__header(//build/toolchain/android:android_clang_arm64)needs //third_party/ijar:ijar(//build/toolchain/mac:clang_x64) //third_party/android_deps:android_arch_core_runtime_java__classes__header(//build/toolchain/android:android_clang_arm64)needs //third_party/ijar:ijar(//build/toolchain/mac:clang_x64) //third_party/android_deps:android_arch_lifecycle_common_java8_java__header(//build/toolchain/android:android_clang_arm64)needs //third_party/ijar:ijar(//build/toolchain/mac:clang_x64) //third_party/android_deps:android_arch_lifecycle_common_java__header(//build/toolchain/android:android_clang_arm64)needs //third_party/ijar:ijar(//build/toolchain/mac:clang_x64) . . . . . .

報(bào)錯(cuò)信息提示說(shuō),找不到被廣泛依賴的一個(gè)構(gòu)建目標(biāo) //third_party/ijar:ijar。

通過(guò)這段報(bào)錯(cuò)信息,我們不難了解到,這個(gè)構(gòu)建目標(biāo)應(yīng)該是定義在 webrtc/third_party/ijar/BUILD.gn 文件中的。我們打開這個(gè)文件,來(lái)看下這個(gè)構(gòu)建目標(biāo)的定義:

if (is_linux || is_chromeos) {config("ijar_compiler_flags") {if (is_clang) {cflags = ["-Wno-shadow","-Wno-unused-but-set-variable",]}}executable("ijar") {sources = ["classfile.cc","common.h","ijar.cc","mapped_file.h","mapped_file_unix.cc","platform_utils.cc","platform_utils.h","zip.cc","zip.h","zlib_client.cc","zlib_client.h",]deps = [ "//third_party/zlib" ]configs += [ ":ijar_compiler_flags" ]# Always build release since this is a build tool.if (is_debug) {configs -= [ "//build/config:debug" ]configs += [ "//build/config:release" ]}} }

通過(guò)該文件,可以看到,只有在 Linux 和 ChromeOS 上,才會(huì)定義構(gòu)建目標(biāo) //third_party/ijar:ijar。我們修改上面的 if 判斷里的條件,同樣為 Mac 定義這個(gè)構(gòu)建目標(biāo)。具體來(lái)說(shuō),是將如下的條件判斷:

if (is_linux || is_chromeos) {

修改為:

if (is_linux || is_chromeos || is_mac) {

webrtc/third_party/ijar/BUILD.gn 這個(gè)文件中的其它內(nèi)容保持不變。

這樣我們可以成功執(zhí)行 gn gen 了:

OpenRTCClient % gn gen build/android/arm64/debug Done. Made 5530 targets from 318 files in 2743ms

3. 執(zhí)行 ninja 命令構(gòu)建 WebRTC

(1). -ffile-compilation-dir=. 導(dǎo)致的編譯錯(cuò)誤

執(zhí)行 ninja 命令構(gòu)建 WebRTC:

OpenRTCClient % ninja -C build/android/arm64/debug

還沒(méi)編譯幾行代碼,就報(bào)了個(gè)編譯錯(cuò)誤:

OpenRTCClient % ninja -C build/android/arm64/debug ninja: Entering directory `/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug' [1/5234] CC obj/base/third_party/libevent/libevent/event_tagging.o FAILED: obj/base/third_party/libevent/libevent/event_tagging.o ../../../../../../../Library/Android/sdk/ndk/23.1.7779620/toolchains/llvm/prebuilt/darwin-x86_64/bin/clang -MMD -MF obj/base/third_party/libevent/libevent/event_tagging.o.d -DHAVE_CONFIG_H -D_GNU_SOURCE -DANDROID -DHAVE_SYS_UIO_H -DANDROID_NDK_VERSION_ROLL=r22_1 -D_DEBUG -DDYNAMIC_ANNOTATIONS_ENABLED=1 -I../../../../webrtc/base/third_party/libevent/android -I../../../../webrtc -Igen -fno-delete-null-pointer-checks -fno-ident -fno-strict-aliasing --param=ssp-buffer-size=4 -fstack-protector -funwind-tables -fPIC -fcolor-diagnostics -fmerge-all-constants -fcrash-diagnostics-dir=../../../../webrtc/tools/clang/crashreports -mllvm -instcombine-lower-dbg-declare=0 -ffp-contract=off -ffunction-sections -fno-short-enums --target=aarch64-linux-android21 -mno-outline-atomics -mno-outline -Wno-builtin-macro-redefined -D__DATE__= -D__TIME__= -D__TIMESTAMP__= -ffile-compilation-dir=. -no-canonical-prefixes -Oz -fdata-sections -ffunction-sections -fno-unique-section-names -fno-omit-frame-pointer -g2 -gdwarf-aranges -ggnu-pubnames -Xclang -fuse-ctor-homing -fvisibility=hidden -Wheader-hygiene -Wstring-conversion -Wtautological-overlap-compare -Wall -Wno-unused-variable -Wno-c++11-narrowing -Wno-unused-but-set-variable -Wno-misleading-indentation -Wno-missing-field-initializers -Wno-unused-parameter -Wloop-analysis -Wno-unneeded-internal-declaration -Wenum-compare-conditional -Wno-psabi -Wno-ignored-pragma-optimize -Wmax-tokens -std=c11 --sysroot=../../../../../../../Library/Android/sdk/ndk/23.1.7779620/toolchains/llvm/prebuilt/darwin-x86_64/sysroot -c ../../../../webrtc/base/third_party/libevent/event_tagging.c -o obj/base/third_party/libevent/libevent/event_tagging.o clang: error: unknown argument: '-ffile-compilation-dir=.' [2/5234] CC obj/base/third_party/libevent/libevent/signal.o . . . . . .

這段報(bào)錯(cuò)信息是說(shuō),clang 無(wú)法識(shí)別一個(gè)編譯選項(xiàng) -ffile-compilation-dir=.。在 stack overflow 上有個(gè)帖子在討論這個(gè)問(wèn)題,Can’t build Webrtc with Bitcode enabled。出現(xiàn)這個(gè)問(wèn)題,是因?yàn)榫幾g WebRTC 用的 clang 版本一般都比較新,WebRTC 的構(gòu)建系統(tǒng)為編譯設(shè)置了較新的 clang 版本才支持的一個(gè)編譯選項(xiàng),但本地編譯時(shí)使用的 clang 版本不夠新,于是出現(xiàn)了編譯工具無(wú)法識(shí)別編譯選項(xiàng)的問(wèn)題。

這個(gè)編譯選項(xiàng)是在 webrtc/build/config/compiler/BUILD.gn 文件中加的:

if (is_clang && strip_absolute_paths_from_debug_symbols) {# If debug option is given, clang includes $cwd in debug info by default.# For such build, this flag generates reproducible obj files even we use# different build directory like "out/feature_a" and "out/feature_b" if# we build same files with same compile flag.# Other paths are already given in relative, no need to normalize them.if (is_nacl) {# TODO(https://crbug.com/1231236): Use -ffile-compilation-dir= here.cflags += ["-Xclang","-fdebug-compilation-dir","-Xclang",".",]} else {# -ffile-compilation-dir is an alias for both -fdebug-compilation-dir=# and -fcoverage-compilation-dir=.cflags += [ "-ffile-compilation-dir=." ]}if (!is_win) {# We don't use clang -cc1as on Windows (yet? https://crbug.com/762167)asmflags = [ "-Wa,-fdebug-compilation-dir,." ]}

這里我們將這個(gè)編譯選項(xiàng)替換掉來(lái)解決這個(gè)問(wèn)題。具體來(lái)說(shuō),是將如下這行代碼:

cflags += [ "-ffile-compilation-dir=." ]

替換為:

cflags += [ "-fdebug-compilation-dir" ]

修改完成后,重新執(zhí)行 gn gen 和 ninja 命令。

(2). jni_generator.py 執(zhí)行失敗導(dǎo)致的錯(cuò)誤

隨后遇到如下編譯報(bào)錯(cuò):

[1109/5146] ACTION //sdk/android:generated_external_classes_jni(//build/toolchain/android:android_clang_arm64) FAILED: gen/sdk/android/generated_external_classes_jni/Integer_jni.h gen/sdk/android/generated_external_classes_jni/Double_jni.h gen/sdk/android/generated_external_classes_jni/Long_jni.h gen/sdk/android/generated_external_classes_jni/Iterable_jni.h gen/sdk/android/generated_external_classes_jni/Iterator_jni.h gen/sdk/android/generated_external_classes_jni/Boolean_jni.h gen/sdk/android/generated_external_classes_jni/BigInteger_jni.h gen/sdk/android/generated_external_classes_jni/Map_jni.h gen/sdk/android/generated_external_classes_jni/LinkedHashMap_jni.h gen/sdk/android/generated_external_classes_jni/ArrayList_jni.h gen/sdk/android/generated_external_classes_jni/Enum_jni.h python3 ../../../../webrtc/base/android/jni_generator/jni_generator.py --ptr_type=long --includes ../../../../../../../../webrtc/sdk/android/src/jni/jni_generator_helper.h --jar_file ../../../../../../../Library/Android/sdk/platforms/android-31/android.jar --output_file gen/sdk/android/generated_external_classes_jni/Integer_jni.h --output_file gen/sdk/android/generated_external_classes_jni/Double_jni.h --output_file gen/sdk/android/generated_external_classes_jni/Long_jni.h --output_file gen/sdk/android/generated_external_classes_jni/Iterable_jni.h --output_file gen/sdk/android/generated_external_classes_jni/Iterator_jni.h --output_file gen/sdk/android/generated_external_classes_jni/Boolean_jni.h --output_file gen/sdk/android/generated_external_classes_jni/BigInteger_jni.h --output_file gen/sdk/android/generated_external_classes_jni/Map_jni.h --output_file gen/sdk/android/generated_external_classes_jni/LinkedHashMap_jni.h --output_file gen/sdk/android/generated_external_classes_jni/ArrayList_jni.h --output_file gen/sdk/android/generated_external_classes_jni/Enum_jni.h --input_file=java/lang/Integer.class --input_file=java/lang/Double.class --input_file=java/lang/Long.class --input_file=java/lang/Iterable.class --input_file=java/util/Iterator.class --input_file=java/lang/Boolean.class --input_file=java/math/BigInteger.class --input_file=java/util/Map.class --input_file=java/util/LinkedHashMap.class --input_file=java/util/ArrayList.class --input_file=java/lang/Enum.class Traceback (most recent call last):File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/base/android/jni_generator/jni_generator.py", line 1630, in <module>sys.exit(main())File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/base/android/jni_generator/jni_generator.py", line 1624, in mainGenerateJNIHeader(java_path, header_path, args)File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/base/android/jni_generator/jni_generator.py", line 1485, in GenerateJNIHeaderjni_from_javap = JNIFromJavaP.CreateFromClass(input_file, options)File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/base/android/jni_generator/jni_generator.py", line 847, in CreateFromClassjni_from_javap = JNIFromJavaP(stdout.split('\n'), options)File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/base/android/jni_generator/jni_generator.py", line 767, in __init__self.fully_qualified_class = self.fully_qualified_class.replace('.', '/') AttributeError: 'JNIFromJavaP' object has no attribute 'fully_qualified_class' [1110/5146] CXX obj/third_party/abseil-cpp/absl/base/spinlock_wait/spinlock_wait.o warning: unknown warning option '-Wno-unused-but-set-variable'; did you mean '-Wno-unused-const-variable'? [-Wunknown-warning-option]

這次是構(gòu)建系統(tǒng)在執(zhí)行腳本文件 webrtc/base/android/jni_generator/jni_generator.py 生成 JNI 頭文件時(shí)出錯(cuò)。根據(jù)出錯(cuò)的調(diào)用堆棧信息,撈出來(lái)相關(guān)的主要的代碼如下:

class JNIFromJavaP(object):"""Uses 'javap' to parse a .class file and generate the JNI header file."""def __init__(self, contents, options):self.contents = contentsself.namespace = options.namespacefor line in contents:class_name = re.match('.*?(public).*?(class|interface) (?P<class_name>\S+?)( |\Z)', line)if class_name:self.fully_qualified_class = class_name.group('class_name')breakself.fully_qualified_class = self.fully_qualified_class.replace('.', '/')# Java 7's javap includes type parameters in output, like HashSet<T>. Strip# away the <...> and use the raw class name that Java 6 would've given us. . . . . . .@staticmethoddef CreateFromClass(class_file, options):class_name = os.path.splitext(os.path.basename(class_file))[0]javap_path = os.path.abspath(options.javap)p = subprocess.Popen(args=[javap_path, '-c', '-verbose', '-s', class_name],cwd=os.path.dirname(class_file),stdout=subprocess.PIPE,stderr=subprocess.PIPE,universal_newlines=True)stdout, _ = p.communicate()jni_from_javap = JNIFromJavaP(stdout.split('\n'), options)return jni_from_javap . . . . . . def GenerateJNIHeader(input_file, output_file, options):try:if os.path.splitext(input_file)[1] == '.class':jni_from_javap = JNIFromJavaP.CreateFromClass(input_file, options)content = jni_from_javap.GetContent()else:jni_from_java_source = JNIFromJavaSource.CreateFromFile(input_file, options)content = jni_from_java_source.GetContent()except ParseError as e:print(e)sys.exit(1)if output_file:with build_utils.AtomicOutput(output_file, mode='w') as f:f.write(content)else:print(content)

相關(guān)代碼的大致意思是,在 GenerateJNIHeader() 函數(shù)中調(diào)用 JNIFromJavaP.CreateFromClass(input_file, options) 方法;在 JNIFromJavaP.CreateFromClass(input_file, options) 方法中執(zhí)行 javap 命令,得到命令執(zhí)行的標(biāo)準(zhǔn)輸出的內(nèi)容,并以此內(nèi)容為參數(shù)創(chuàng)建 JNIFromJavaP,在類 JNIFromJavaP 構(gòu)造函數(shù)中出錯(cuò)。

我們打印 JNIFromJavaP 構(gòu)造函數(shù)接收的 contents 參數(shù),可以看到這個(gè)參數(shù)是空字符串,顯然不符合預(yù)期。我們提取 JNIFromJavaP.CreateFromClass(input_file, options) 方法中執(zhí)行的 javap 命令,這個(gè)命令大概像這樣:

/Library/Java/JavaVirtualMachines/jdk-11.0.13.jdk/Contents/Home/bin/javap -c -verbose -s Integer

我們?cè)诮K端命令行中執(zhí)行這個(gè)命令,這個(gè)命令執(zhí)行會(huì)報(bào)錯(cuò):

OpenRTCClient % /Library/Java/JavaVirtualMachines/jdk-11.0.13.jdk/Contents/Home/bin/javap -c -verbose -s Integer 錯(cuò)誤: 找不到類: Integer

來(lái)看一下給 javap 傳的兩個(gè)參數(shù)是干什么的:

OpenRTCClient % javap --help 用法: javap <options> <classes> 其中, 可能的選項(xiàng)包括:-? -h --help -help 輸出此幫助消息-version 版本信息-v -verbose 輸出附加信息-l 輸出行號(hào)和本地變量表-public 僅顯示公共類和成員-protected 顯示受保護(hù)的/公共類和成員-package 顯示程序包/受保護(hù)的/公共類和成員 (默認(rèn))-p -private 顯示所有類和成員-c 對(duì)代碼進(jìn)行反匯編-s 輸出內(nèi)部類型簽名-sysinfo 顯示正在處理的類的系統(tǒng)信息 (路徑, 大小, 日期, MD5 散列)-constants 顯示最終常量--module <模塊>, -m <模塊> 指定包含要反匯編的類的模塊--module-path <路徑> 指定查找應(yīng)用程序模塊的位置--system <jdk> 指定查找系統(tǒng)模塊的位置--class-path <路徑> 指定查找用戶類文件的位置-classpath <路徑> 指定查找用戶類文件的位置-cp <路徑> 指定查找用戶類文件的位置-bootclasspath <路徑> 覆蓋引導(dǎo)類文件的位置GNU 樣式的選項(xiàng)可使用 = (而非空白) 來(lái)分隔選項(xiàng)名稱 及其值。每個(gè)類可由其文件名, URL 或其 全限定類名指定。示例:path/to/MyClass.classjar:file:///path/to/MyJar.jar!/mypkg/MyClass.classjava.lang.Object

-c 參數(shù)用于對(duì)代碼進(jìn)行反匯編,-s 參數(shù)用于輸出內(nèi)部類型簽名。不過(guò)這里出錯(cuò)的關(guān)鍵不是參數(shù),而是指定類的方式。javap 命令接受的指定類的方式如上面的 help 輸出中最下面的說(shuō)明。Mac 平臺(tái) javap 命令接受的指定類的方式與 Linux 平臺(tái)的 javap 命令接受的不太一樣。

WebRTC 的構(gòu)建系統(tǒng)會(huì)通過(guò) webrtc/base/android/jni_generator/jni_generator.py 為幾個(gè)標(biāo)準(zhǔn)庫(kù)的類生成 JNI 頭文件,這些類的類文件打包在 rt.jar 文件中。我們從 rt.jar 文件中提取所有的類文件,并對(duì)類文件執(zhí)行 javap 命令:

opensource % mkdir jar_classes opensource % cd jar_classes jar_classes % jar xf /Library/Java/JavaVirtualMachines/jdk1.8.0_311.jdk/Contents/Home/jre/lib/rt.jar jar_classes % cd java/lang lang % javap -c -s Integer 錯(cuò)誤: 找不到類: Integer lang % javap -c -s Integer.class Compiled from "Integer.java" public final class java.lang.Integer extends java.lang.Number implements java.lang.Comparable<java.lang.Integer> {public static final int MIN_VALUE;descriptor: Ipublic static final int MAX_VALUE;

在類文件所在的目錄,我們僅通過(guò)傳入類名類指定類時(shí),javap 執(zhí)行失敗。而通過(guò) 類名.class 的形式指定類時(shí),則可以執(zhí)行成功。在 Linux 平臺(tái)上,如果位于類文件所在的目錄,僅通過(guò)傳入類名來(lái)指定類,而沒(méi)有 .class 后綴名,javap 也可以執(zhí)行成功。

對(duì)于上面例子中的 Integer,在 Mac 平臺(tái)上執(zhí)行 javap 時(shí),正確的參數(shù)格式有多種,可以是 java.lang.Integer,可以是類文件的完整文件路徑,如 /var/folders/1x/5vtfxt410vbf9xhk8vvfz8l40000gn/T/tmp2gbxloem/java/lang/Integer.class,也可以是類文件的文件名 Integer.class,但不能是 Integer。

由此,也就可以得到我們對(duì)這個(gè)問(wèn)題的解決方案了。

方案一,通過(guò)類文件的完整文件路徑來(lái)給 javap 指定類,修改 JNIFromJavaP.CreateFromClass(input_file, options) 方法的實(shí)現(xiàn)為:

@staticmethoddef CreateFromClass(class_file, options):if sys.platform == "darwin":class_name = class_fileelse:class_name = os.path.splitext(os.path.basename(class_file))[0]javap_path = os.path.abspath(options.javap)p = subprocess.Popen(args=[javap_path, '-c', '-verbose', '-s', class_name],cwd=os.path.dirname(class_file),stdout=subprocess.PIPE,stderr=subprocess.PIPE,universal_newlines=True)stdout, _ = p.communicate()jni_from_javap = JNIFromJavaP(stdout.split('\n'), options)return jni_from_javap

方案二,由于創(chuàng)建進(jìn)程執(zhí)行命令時(shí),設(shè)置了工作目錄,我們還可以通過(guò)類文件的文件名來(lái)給 javap 指定類,這修改 JNIFromJavaP.CreateFromClass(input_file, options) 方法的實(shí)現(xiàn)為:

@staticmethoddef CreateFromClass(class_file, options):if sys.platform == "darwin":class_name = os.path.basename(class_file)else:class_name = os.path.splitext(os.path.basename(class_file))[0]javap_path = os.path.abspath(options.javap)p = subprocess.Popen(args=[javap_path, '-c', '-verbose', '-s', class_name],cwd=os.path.dirname(class_file),stdout=subprocess.PIPE,stderr=subprocess.PIPE,universal_newlines=True)stdout, _ = p.communicate()jni_from_javap = JNIFromJavaP(stdout.split('\n'), options)return jni_from_javap

方案三,通過(guò)類路徑來(lái)給 javap 指定類,這修改 JNIFromJavaP.CreateFromClass(input_file, options) 方法的實(shí)現(xiàn)為:

@staticmethoddef CreateFromClass(class_file, options):if sys.platform == "darwin":path_items = class_file.split("/")i = len(path_items) - 2while i >= 0:class_name = path_items[i] + "." + class_nameif path_items[i] == "java":breaki = i -1else:class_name = os.path.splitext(os.path.basename(class_file))[0]javap_path = os.path.abspath(options.javap)p = subprocess.Popen(args=[javap_path, '-c', '-verbose', '-s', class_name],cwd=os.path.dirname(class_file),stdout=subprocess.PIPE,stderr=subprocess.PIPE,universal_newlines=True)stdout, _ = p.communicate()jni_from_javap = JNIFromJavaP(stdout.split('\n'), options)return jni_from_javap

方案三的這種實(shí)現(xiàn),對(duì)于要生成 JNI 的類有一定的限制。

相對(duì)來(lái)說(shuō),綜合考慮,方案二無(wú)疑是最好的。

(3). Android SDK 工具缺失或版本問(wèn)題導(dǎo)致的編譯錯(cuò)誤

再次執(zhí)行 ninja,遇到如下編譯報(bào)錯(cuò):

OpenRTCClient % mv ~/Library/Android/sdk/cmdline-tools/latest ~/Library/Android/sdk/cmdline-tools/latest_bak OpenRTCClient % webrtc_build build android arm64 debug /Users/lisi/Projects/opensource/OpenRTCClient/build_system/tools/bin/mac/ninja -C /Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug ninja: Entering directory `/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug' [355/4574] ACTION //examples:AppRTCMobile_test_apk__test_apk__merge_manifests(//build/toolchain/android:android_clang_arm64) FAILED: gen/examples/AppRTCMobile_test_apk__test_apk_manifest/AndroidManifest.xml python3 ../../../../webrtc/build/android/gyp/merge_manifest.py --depfile gen/examples/AppRTCMobile_test_apk__test_apk__merge_manifests.d --android-sdk-cmdline-tools ../../../../../../../Library/Android/sdk/cmdline-tools/latest --root-manifest ../../../../webrtc/examples/androidtests/AndroidManifest.xml --output gen/examples/AppRTCMobile_test_apk__test_apk_manifest/AndroidManifest.xml --extras @FileArg\(gen/examples/AppRTCMobile_test_apk__test_apk.build_config.json:extra_android_manifests\) --min-sdk-version=21 --target-sdk-version=21 Traceback (most recent call last):File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/merge_manifest.py", line 149, in <module>main(sys.argv[1:])File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/merge_manifest.py", line 129, in mainbuild_utils.CheckOutput(File "/Users/lisi/Projects/opensource/OpenRTCClient/webrtc/build/android/gyp/util/build_utils.py", line 276, in CheckOutputraise CalledProcessError(cwd, args, stdout + stderr) util.build_utils.CalledProcessError: Command failed: ( cd /Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug; /Library/Java/JavaVirtualMachines/jdk-11.0.13.jdk/Contents/Home/bin/java -Xmx1G -noverify -cp ../../../../../../../Library/Android/sdk/cmdline-tools/latest/lib/build-system/manifest-merger.jar:../../../../../../../Library/Android/sdk/cmdline-tools/latest/lib/common/common.jar:../../../../../../../Library/Android/sdk/cmdline-tools/latest/lib/sdk-common/sdk-common.jar:../../../../../../../Library/Android/sdk/cmdline-tools/latest/lib/sdklib/sdklib.jar:../../../../../../../Library/Android/sdk/cmdline-tools/latest/lib/external/com/google/guava/guava/30.1-jre/guava-30.1-jre.jar:../../../../../../../Library/Android/sdk/cmdline-tools/latest/lib/external/kotlin-plugin-ij/Kotlin/kotlinc/lib/kotlin-stdlib.jar:../../../../../../../Library/Android/sdk/cmdline-tools/latest/lib/external/com/google/code/gson/gson/2.8.6/gson-2.8.6.jar com.android.manifmerger.Merger --out /Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/gen/examples/AppRTCMobile_test_apk__test_apk_manifest/tmpvg3qg_k3AndroidManifest.xml --property MIN_SDK_VERSION=21 --property TARGET_SDK_VERSION=21 --libs gen/third_party/android_sdk/android_test_base_java/AndroidManifest.xml:gen/third_party/android_sdk/android_test_runner_java/AndroidManifest.xml --main /var/folders/1x/5vtfxt410vbf9xhk8vvfz8l40000gn/T/AndroidManifest.xmlwu610lv1 --property PACKAGE=org.appspot.apprtc.test --remove-tools-declarations ) 錯(cuò)誤: 找不到或無(wú)法加載主類 com.android.manifmerger.Merger 原因: java.lang.ClassNotFoundException: com.android.manifmerger.Merger

這個(gè)報(bào)錯(cuò)是因?yàn)?#xff0c;編譯 Android 需要生成一些測(cè)試 APK 文件,此時(shí)需要執(zhí)行 Android SDK 中的一些 Java 工具,這里提示找不到的主類 com.android.manifmerger.Merger,其定義在 sdk/cmdline-tools/latest/lib/build-system/manifest-merger.jar 這個(gè) jar 文件中,但在我們的 SDK 中,沒(méi)有 sdk/cmdline-tools/latest/ 這個(gè)目錄。

不難看出 sdk/cmdline-tools/latest/ 是要引用最新版本的 cmdline-tools,但我們的 SDK 包中,只安裝了 3.0 版。

由此,可以有很多種方案來(lái)解決這個(gè)問(wèn)題。

方案一,創(chuàng)建一個(gè)符號(hào)鏈接 sdk/cmdline-tools/latest 指向 sdk/cmdline-tools/3.0,如:

tmp % ln -s ~/Library/Android/sdk/cmdline-tools/3.0 ~/Library/Android/sdk/cmdline-tools/lastest

方案二,安裝最新版的 cmdline-tools,既可以通過(guò) Android Studio 來(lái)完成,也可以通過(guò) Android SDK 提供的命令行工具來(lái)做,如:

~/Library/Android/sdk/tools/bin/sdkmanager --install "cmdline-tools;latest"

如果發(fā)現(xiàn)安裝了最新版的 cmdline-tools 還是找不到主類 com.android.manifmerger.Merger,則可以看下 cmdline-tools 中是否有 ~/Library/Android/sdk/cmdline-tools/latest/lib/build-system/manifest-merger.jar 這個(gè) jar 文件:

OpenRTCClient % ls -l ~/Library/Android/sdk/cmdline-tools/latest/lib/build-system/ total 424 drwxr-xr-x 3 lisi staff 96 4 11 14:21 builder-model drwxr-xr-x 3 lisi staff 96 4 11 14:21 builder-test-api -rwxr-xr-x 1 lisi staff 213592 4 11 14:21 tools.manifest-merger.jar

如果是這種情況,可以拷貝 tools.manifest-merger.jar 或者為它創(chuàng)建符號(hào)鏈接,如:

OpenRTCClient % cp ~/Library/Android/sdk/cmdline-tools/latest/lib/build-system/tools.manifest-merger.jar ~/Library/Android/sdk/cmdline-tools/latest/lib/build-system/manifest-merger.jar

在上面的報(bào)錯(cuò)信息中,我們可以看到引用了多個(gè) jar 文件,對(duì)于其中的每一個(gè) jar 文件,我們都需要以類似這樣的方式確保它們的存在。

方案三,修改 WebRTC 的構(gòu)建配置。
這里的報(bào)錯(cuò),是在執(zhí)行 webrtc/build/config/android/internal_rules.gni 中定義的 template("merge_manifests") 時(shí)報(bào)出的,cmdline-tools 的路徑是在這里定義的,如:

template("merge_manifests") {action_with_pydeps(target_name) {forward_variables_from(invoker, TESTONLY_AND_VISIBILITY + [ "deps" ])script = "//build/android/gyp/merge_manifest.py"depfile = "$target_gen_dir/$target_name.d"inputs = [invoker.build_config,invoker.input_manifest,]outputs = [ invoker.output_manifest ]_rebased_build_config = rebase_path(invoker.build_config, root_build_dir)args = ["--depfile",rebase_path(depfile, root_build_dir),"--android-sdk-cmdline-tools",rebase_path("${android_sdk_root}/cmdline-tools/latest",root_build_dir),"--root-manifest",rebase_path(invoker.input_manifest, root_build_dir),"--output",rebase_path(invoker.output_manifest, root_build_dir),"--extras","@FileArg($_rebased_build_config:extra_android_manifests)","--min-sdk-version=${invoker.min_sdk_version}","--target-sdk-version=${invoker.target_sdk_version}",]

可以將這里的 cmdline-tools 的路徑修改為我們本地已經(jīng)安裝的版本的路徑。

(4). Clang 缺少 Android NDK 包的庫(kù)或工具導(dǎo)致編譯失敗

再次執(zhí)行 ninja,遇到如下編譯報(bào)錯(cuò):

OpenRTCClient % webrtc_build build android arm64 debug /Users/lisi/Projects/opensource/OpenRTCClient/build_system/tools/bin/mac/ninja -C /Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug ninja: Entering directory `/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug' [14/1093] LINK ./stun_prober FAILED: stun_prober exe.unstripped/stun_prober python3 "../../../../webrtc/build/toolchain/gcc_link_wrapper.py" --output="./stun_prober" --strip="../../../../build_system/llvm-build/mac/mac/Release+Asserts/bin/llvm-strip" --unstripped-file="./exe.unstripped/stun_prober" -- ../../../../build_system/llvm-build/mac/mac/Release+Asserts/bin/clang++ -fuse-ld=lld -Wl,--fatal-warnings -Wl,--build-id -fPIC -Wl,-z,noexecstack -Wl,-z,relro -Wl,-z,now -Wl,-z,max-page-size=4096 -Wl,--color-diagnostics -Wl,--no-rosegment -Wl,--no-call-graph-profile-sort -Wl,--exclude-libs=libvpx_assembly_arm.a --unwindlib=none --target=aarch64-linux-android21 -Wl,-mllvm,-enable-machine-outliner=never -no-canonical-prefixes -Wl,--gc-sections -Wl,--gdb-index --sysroot=../../../../../../../Library/Android/sdk/ndk/23.1.7779620/toolchains/llvm/prebuilt/darwin-x86_64/sysroot -Wl,--warn-shared-textrel -Wl,-z,defs -Wl,--as-needed -pie -Bdynamic -Wl,-z,nocopyreloc -Wl,--dynamic-linker,/system/bin/linker64 -o "./exe.unstripped/stun_prober" -Wl,--start-group @"./stun_prober.rsp" -Wl,--end-group -ldl -lm -llog -lGLESv2 -lOpenSLES ld.lld: error: cannot open ../../../../build_system/llvm-build/mac/mac/Release+Asserts/lib/clang/14.0.0/lib/linux/libclang_rt.builtins-aarch64-android.a: No such file or directory ld.lld: error: cannot open ../../../../build_system/llvm-build/mac/mac/Release+Asserts/lib/clang/14.0.0/lib/linux/libclang_rt.builtins-aarch64-android.a: No such file or directory clang++: error: linker command failed with exit code 1 (use -v to see invocation)

這是由于 webrtc 中的 LLVM 工具鏈中缺少編譯鏈接 Android 二進(jìn)制文件所需的一些庫(kù)文件。這可以通過(guò)將 NDK 中的庫(kù)文件拷貝的 LLVM 工具鏈的對(duì)應(yīng)目錄下來(lái)解決,如:

android-ndk-r23b-darwin % cp -r toolchains/llvm/prebuilt/darwin-x86_64//lib64/clang/12.0.8/lib/linux ~/Projects/opensource/OpenRTCClient/build_system/llvm-build/mac/mac/Release+Asserts/lib/clang/14.0.0/lib/

再次執(zhí)行 ninja,遇到如下編譯報(bào)錯(cuò):

OpenRTCClient % webrtc_build build android arm64 debug /Users/lisi/Projects/opensource/OpenRTCClient/build_system/tools/bin/mac/ninja -C /Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug ninja: Entering directory `/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug' [33/880] LINK ./stun_prober FAILED: stun_prober exe.unstripped/stun_prober python3 "../../../../webrtc/build/toolchain/gcc_link_wrapper.py" --output="./stun_prober" --strip="../../../../build_system/llvm-build/mac/mac/Release+Asserts/bin/llvm-strip" --unstripped-file="./exe.unstripped/stun_prober" -- ../../../../build_system/llvm-build/mac/mac/Release+Asserts/bin/clang++ -fuse-ld=lld -Wl,--fatal-warnings -Wl,--build-id -fPIC -Wl,-z,noexecstack -Wl,-z,relro -Wl,-z,now -Wl,-z,max-page-size=4096 -Wl,--color-diagnostics -Wl,--no-rosegment -Wl,--no-call-graph-profile-sort -Wl,--exclude-libs=libvpx_assembly_arm.a --unwindlib=none --target=aarch64-linux-android21 -Wl,-mllvm,-enable-machine-outliner=never -no-canonical-prefixes -Wl,--gc-sections -Wl,--gdb-index --sysroot=../../../../../../../Library/Android/sdk/ndk/23.1.7779620/toolchains/llvm/prebuilt/darwin-x86_64/sysroot -Wl,--warn-shared-textrel -Wl,-z,defs -Wl,--as-needed -pie -Bdynamic -Wl,-z,nocopyreloc -Wl,--dynamic-linker,/system/bin/linker64 -o "./exe.unstripped/stun_prober" -Wl,--start-group @"./stun_prober.rsp" -Wl,--end-group -ldl -lm -llog -lGLESv2 -lOpenSLES Traceback (most recent call last):File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/toolchain/gcc_link_wrapper.py", line 91, in <module>sys.exit(main())File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/toolchain/gcc_link_wrapper.py", line 78, in mainresult = subprocess.call(File "/usr/local/Cellar/python@3.9/3.9.9/Frameworks/Python.framework/Versions/3.9/lib/python3.9/subprocess.py", line 349, in callwith Popen(*popenargs, **kwargs) as p:File "/usr/local/Cellar/python@3.9/3.9.9/Frameworks/Python.framework/Versions/3.9/lib/python3.9/subprocess.py", line 951, in __init__self._execute_child(args, executable, preexec_fn, close_fds,File "/usr/local/Cellar/python@3.9/3.9.9/Frameworks/Python.framework/Versions/3.9/lib/python3.9/subprocess.py", line 1821, in _execute_childraise child_exception_type(errno_num, err_msg, err_filename) FileNotFoundError: [Errno 2] No such file or directory: '../../../../build_system/llvm-build/mac/mac/Release+Asserts/bin/llvm-strip'

這個(gè)報(bào)錯(cuò)是說(shuō),在鏈接的時(shí)候,找不到 LLVM 工具鏈中的 llvm-strip。

同樣,我們將 NDK 下的工具拷貝過(guò)去,同時(shí)也要將依賴的庫(kù)拷貝過(guò)去:

android-ndk-r23b-darwin % cp toolchains/llvm/prebuilt/darwin-x86_64/bin/llvm-strip /Users/lisi/Projects/opensource/OpenRTCClient/build_system/llvm-build/mac/mac/Release+Asserts/bin/ android-ndk-r23b-darwin % cp -r toolchains/llvm/prebuilt/darwin-x86_64/lib64 /Users/lisi/Projects/opensource/OpenRTCClient/build_system/llvm-build/mac/mac/Release+Asserts/

要做同樣處理的還有 llvm-nm 工具:

android-ndk-r23b-darwin % cp toolchains/llvm/prebuilt/darwin-x86_64/bin/llvm-nm /Users/lisi/Projects/opensource/OpenRTCClient/build_system/llvm-build/mac/mac/Release+Asserts/bin/

(5). Unix 域 Socket 地址錯(cuò)誤導(dǎo)致的編譯失敗

再次執(zhí)行 ninja,遇到如下編譯報(bào)錯(cuò):

OpenRTCClient % webrtc_build build android arm64 debug /Users/lisi/Projects/opensource/OpenRTCClient/build_system/tools/bin/mac/ninja -C /Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug ninja: Entering directory `/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug' [7/744] ACTION //sdk/android:base_java__errorprone(//build/toolchain/android:android_clang_arm64) FAILED: obj/sdk/android/base_java__errorprone.errorprone.stamp python3 ../../../../webrtc/build/android/gyp/compile_java.py --depfile=gen/sdk/android/base_java__errorprone.d --generated-dir=gen/sdk/android/base_java/generated_java --jar-path=obj/sdk/android/base_java__errorprone.errorprone.stamp --java-srcjars=\[\"gen/sdk/android/base_java.generated.srcjar\"\] --target-name //sdk/android:base_java__errorprone --header-jar obj/sdk/android/base_java.turbine.jar --classpath=\[\"obj/sdk/android/base_java.turbine.jar\"\] --classpath=@FileArg\(gen/sdk/android/base_java.build_config.json:deps_info:javac_full_interface_classpath\) --java-version=1.8 --bootclasspath=@FileArg\(gen/sdk/android/base_java.build_config.json:android:sdk_interface_jars\) --chromium-code=1 --jar-info-exclude-globs=\[\"\*/R.class\",\ \"\*/R\\\$\*.class\",\ \"\*/Manifest.class\",\ \"\*/Manifest\\\$\*.class\",\ \"\*/GEN_JNI.class\"\] --processorpath=@FileArg\(gen/tools/android/errorprone_plugin/errorprone_plugin.build_config.json:deps_info:host_classpath\) --enable-errorprone @gen/sdk/android/base_java.sources --javac-arg=-Werror Traceback (most recent call last):File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/compile_java.py", line 800, in <module>sys.exit(main(sys.argv[1:]))File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/compile_java.py", line 693, in mainand server_utils.MaybeRunCommand(name=options.target_name,File "/Users/lisi/Projects/opensource/OpenRTCClient/webrtc/build/android/gyp/util/server_utils.py", line 40, in MaybeRunCommandraise eFile "/Users/lisi/Projects/opensource/OpenRTCClient/webrtc/build/android/gyp/util/server_utils.py", line 27, in MaybeRunCommandsock.connect(SOCKET_ADDRESS) FileNotFoundError: [Errno 2] No such file or directory

在 webrtc/build/android/gyp/util/server_utils.py 文件中,我們可以看到 SOCKET_ADDRESS 的定義:

SOCKET_ADDRESS = '\0chromium_build_server_socket'

最前面的 ‘\0’ 導(dǎo)致在 Mac 上創(chuàng)建 unix 域 socket 失敗,并設(shè)置它為臨時(shí)目錄下的一個(gè)路徑,如:

SOCKET_ADDRESS = '/tmp/chromium_build_server_socket'

同時(shí),還有把 Unix 域 socket 的 server 端跑起來(lái),如:

OpenRTCClient % python3 webrtc/build/android/fast_local_dev_server.py

(6). 編譯資源失敗

再次執(zhí)行 ninja,遇到如下編譯報(bào)錯(cuò):

[20/692] ACTION //examples:AppRTCMobile__compile_resources(//build/toolchain/android:android_clang_arm64) FAILED: gen/examples/AppRTCMobile__compile_resources.srcjar obj/examples/AppRTCMobile.ap_ obj/examples/AppRTCMobile.ap_.info gen/examples/AppRTCMobile__compile_resources_R.txt obj/examples/AppRTCMobile/AppRTCMobile.resources.proguard.txt gen/examples/AppRTCMobile__compile_resources.resource_ids python3 ../../../../webrtc/build/android/gyp/compile_resources.py --include-resources=@FileArg\(gen/examples/AppRTCMobile.build_config.json:android:sdk_jars\) --aapt2-path ../../../../webrtc/third_party/android_build_tools/aapt2/aapt2 --dependencies-res-zips=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:dependency_zips\) --extra-res-packages=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:extra_package_names\) --extra-main-r-text-files=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:extra_main_r_text_files\) --min-sdk-version=21 --target-sdk-version=29 --webp-cache-dir=obj/android-webp-cache --android-manifest gen/examples/AppRTCMobile_manifest/AndroidManifest.xml --srcjar-out gen/examples/AppRTCMobile__compile_resources.srcjar --version-code 1 --version-name Developer\ Build --arsc-path obj/examples/AppRTCMobile.ap_ --info-path obj/examples/AppRTCMobile.ap_.info --debuggable --r-text-out gen/examples/AppRTCMobile__compile_resources_R.txt --dependencies-res-zip-overlays=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:dependency_zips\) --proguard-file obj/examples/AppRTCMobile/AppRTCMobile.resources.proguard.txt --emit-ids-out=gen/examples/AppRTCMobile__compile_resources.resource_ids --depfile gen/examples/AppRTCMobile__compile_resources.d E 3273 2236 Subprocess raised an exception: Traceback (most recent call last):File "/Users/lisi/Projects/opensource/OpenRTCClient/webrtc/build/android/gyp/util/parallel.py", line 75, in __call__return self._func(*_fork_params[index], **_fork_kwargs) TypeError: 'NoneType' object is not subscriptable

這里我們通過(guò)把資源編譯的并發(fā)編譯給關(guān)掉來(lái)解決這個(gè)問(wèn)題。具體來(lái)說(shuō),可以修改 webrtc/build/android/gyp/util/parallel.py 這個(gè)文件,將文件中的 DISABLE_ASYNC 設(shè)置為 True,或者導(dǎo)出環(huán)境變量 export DISABLE_ASYNC=1。

(7). aapt 可執(zhí)行文件格式錯(cuò)誤導(dǎo)致編譯失敗

再次執(zhí)行 ninja,遇到如下編譯報(bào)錯(cuò):

[36/579] ACTION //examples:AppRTCMobile__compile_resources(//build/toolchain/android:android_clang_arm64) FAILED: gen/examples/AppRTCMobile__compile_resources.srcjar obj/examples/AppRTCMobile.ap_ obj/examples/AppRTCMobile.ap_.info gen/examples/AppRTCMobile__compile_resources_R.txt obj/examples/AppRTCMobile/AppRTCMobile.resources.proguard.txt gen/examples/AppRTCMobile__compile_resources.resource_ids python3 ../../../../webrtc/build/android/gyp/compile_resources.py --include-resources=@FileArg\(gen/examples/AppRTCMobile.build_config.json:android:sdk_jars\) --aapt2-path ../../../../webrtc/third_party/android_build_tools/aapt2/aapt2 --dependencies-res-zips=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:dependency_zips\) --extra-res-packages=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:extra_package_names\) --extra-main-r-text-files=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:extra_main_r_text_files\) --min-sdk-version=21 --target-sdk-version=29 --webp-cache-dir=obj/android-webp-cache --android-manifest gen/examples/AppRTCMobile_manifest/AndroidManifest.xml --srcjar-out gen/examples/AppRTCMobile__compile_resources.srcjar --version-code 1 --version-name Developer\ Build --arsc-path obj/examples/AppRTCMobile.ap_ --info-path obj/examples/AppRTCMobile.ap_.info --debuggable --r-text-out gen/examples/AppRTCMobile__compile_resources_R.txt --dependencies-res-zip-overlays=@FileArg\(gen/examples/AppRTCMobile.build_config.json:deps_info:dependency_zips\) --proguard-file obj/examples/AppRTCMobile/AppRTCMobile.resources.proguard.txt --emit-ids-out=gen/examples/AppRTCMobile__compile_resources.resource_ids --depfile gen/examples/AppRTCMobile__compile_resources.d WARNING:root:Running in synchronous mode. Traceback (most recent call last):File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/compile_resources.py", line 1039, in <module>main(sys.argv[1:])File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/compile_resources.py", line 956, in mainmanifest_package_name = _PackageApk(options, build)File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/compile_resources.py", line 759, in _PackageApkpartials = _CompileDeps(options.aapt2_path, dep_subdirs,File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/compile_resources.py", line 619, in _CompileDepspartials = lisi(File "/Users/lisi/Projects/opensource/OpenRTCClient/webrtc/build/android/gyp/util/parallel.py", line 207, in BulkForkAndCallyield func(*args, **kwargs)File "/Users/lisi/Projects/opensource/OpenRTCClient/build/android/arm64/debug/../../../../webrtc/build/android/gyp/compile_resources.py", line 583, in _CompileSingleDepbuild_utils.CheckOutput(File "/Users/lisi/Projects/opensource/OpenRTCClient/webrtc/build/android/gyp/util/build_utils.py", line 260, in CheckOutputchild = subprocess.Popen(args,File "/usr/local/Cellar/python@3.9/3.9.9/Frameworks/Python.framework/Versions/3.9/lib/python3.9/subprocess.py", line 951, in __init__self._execute_child(args, executable, preexec_fn, close_fds,File "/usr/local/Cellar/python@3.9/3.9.9/Frameworks/Python.framework/Versions/3.9/lib/python3.9/subprocess.py", line 1821, in _execute_childraise child_exception_type(errno_num, err_msg, err_filename) OSError: [Errno 8] Exec format error: '../../../../webrtc/third_party/android_build_tools/aapt2/aapt2'

這是由于 webrtc/third_party/android_build_tools/aapt2/aapt2 文件實(shí)際上為 Linux 平臺(tái)的可執(zhí)行文件。

編譯資源的 template 在 webrtc/build/config/android/internal_rules.gni 中定義:

template("compile_resources") {forward_variables_from(invoker, TESTONLY_AND_VISIBILITY)_deps = [invoker.android_sdk_dep,invoker.build_config_dep,]if (defined(invoker.android_manifest_dep)) {_deps += [ invoker.android_manifest_dep ]}foreach(_dep, invoker.deps) {_target_label = get_label_info(_dep, "label_no_toolchain")if (filter_exclude([ _target_label ], _java_library_patterns) == [] &&filter_exclude([ _target_label ], _java_resource_patterns) != []) {# Depend on the java libraries' transitive __assetres target instead._deps += [ "${_target_label}__assetres" ]} else {_deps += [ _dep ]}}if (defined(invoker.arsc_output)) {_arsc_output = invoker.arsc_output}_final_srcjar_path = "${target_gen_dir}/${target_name}.srcjar"_script = "//build/android/gyp/compile_resources.py"_inputs = [invoker.build_config,android_sdk_tools_bundle_aapt2,]_rebased_build_config = rebase_path(invoker.build_config, root_build_dir)_args = ["--include-resources=@FileArg($_rebased_build_config:android:sdk_jars)","--aapt2-path",rebase_path(android_sdk_tools_bundle_aapt2, root_build_dir),"--dependencies-res-zips=@FileArg($_rebased_build_config:deps_info:dependency_zips)","--extra-res-packages=@FileArg($_rebased_build_config:deps_info:extra_package_names)","--extra-main-r-text-files=@FileArg($_rebased_build_config:deps_info:extra_main_r_text_files)","--min-sdk-version=${invoker.min_sdk_version}","--target-sdk-version=${invoker.target_sdk_version}","--webp-cache-dir=obj/android-webp-cache",]

其中會(huì)調(diào)用腳本文件 webrtc/build/android/gyp/compile_resources.py 并傳入 aapt2 的路徑作為其中的一個(gè)參數(shù)。可以看到這里引用的 aapt2 的路徑為 android_sdk_tools_bundle_aapt2。android_sdk_tools_bundle_aapt2 是一個(gè)編譯配置項(xiàng),其定義在 webrtc/build/config/android/config.gni 中,這個(gè)配置項(xiàng)有默認(rèn)值:

android_sdk_tools_bundle_aapt2_dir ="//third_party/android_build_tools/aapt2". . . . . .android_sdk_tools_bundle_aapt2 = "${android_sdk_tools_bundle_aapt2_dir}/aapt2"

我們可以通過(guò)在 args.gn 中定制 android_sdk_tools_bundle_aapt2_dir 這個(gè)配置項(xiàng)來(lái)解決這個(gè)問(wèn)題。具體來(lái)說(shuō),需要將這個(gè)配置項(xiàng)設(shè)置為 Android SDK 中我們選擇的 build-tools 的路徑,如下:

android_sdk_tools_bundle_aapt2_dir = "/Users/lisi/Library/Android/sdk/build-tools/30.0.1"

再次再次執(zhí)行 ninja,可以正確編譯出各個(gè)需要的文件。

筆者在 GitHub 上建了一個(gè)倉(cāng)庫(kù),放了 WebRTC 完整的代碼以及構(gòu)建系統(tǒng),包括 WebRTC 的源碼及其依賴的第三方庫(kù),編譯所需要的各種工具等,其中主要針對(duì) WebRTC 的 m98 版,搭建了整個(gè)的構(gòu)建環(huán)境,地址為 OpenRTCClient,有需要的朋友可以參考一些。

Done.

總結(jié)

以上是生活随笔為你收集整理的在 Mac 上为 Android 编译 WebRTC的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

在线观看香蕉视频 | 成人精品一区二区三区电影免费 | 久久爱资源网 | 久精品视频 | 91亚洲欧美激情 | 国产精品久久久久永久免费看 | 四虎影视成人精品国库在线观看 | 中文字幕第一页在线播放 | 久久久午夜剧场 | 国产小视频免费观看 | 亚洲麻豆精品 | 男女激情片在线观看 | 国产免费成人av | 黄色网www | av中文字幕日韩 | 欧美日韩亚洲第一页 | 日韩av不卡在线观看 | 国产精品色婷婷 | 欧美日韩免费在线视频 | 成人一级片免费看 | 精品免费一区二区三区 | 日本精油按摩3 | www178ccom视频在线 | 久青草影院 | 欧美在线视频一区二区三区 | 欧美一区在线看 | 国产在线va | 精壮的侍卫呻吟h | 9色在线视频 | 欧美日韩国产综合网 | 国产精品久久一区二区三区不卡 | 午夜91在线 | 国产一区麻豆 | 欧美另类激情 | 97超碰总站| 亚洲专区视频在线观看 | 丁香午夜 | av免费观看网址 | 在线观看视频 | 99精品福利视频 | 中文字幕网站 | 精品国产诱惑 | aⅴ视频在线 | 国产一区麻豆 | 激情av资源网 | 亚洲人av免费网站 | 国产毛片久久 | 日韩欧美视频免费在线观看 | 成人午夜影院 | 中文字幕电影高清在线观看 | 成人免费一级 | 激情网站 | 黄色片毛片 | 中文字幕在线有码 | 色九色 | 亚洲女欲精品久久久久久久18 | 99精品网站 | 黄色的片子 | 色婷婷中文| 国产福利中文字幕 | 五月天婷婷视频 | 国产精品综合久久久久 | 天天色天天干天天色 | 久久99久久99精品免费看小说 | 婷婷久久网 | 日本高清中文字幕有码在线 | 欧美日韩免费看 | 又黄又色又爽 | 国产拍揄自揄精品视频麻豆 | 久久久成人精品 | 国产色爽 | 999精品 | 欧日韩在线视频 | 又黄又爽又湿又无遮挡的在线视频 | 国产在线久久久 | 久久在草 | 超碰av在线免费观看 | 91大神精品视频 | 色综合久久久久综合 | 西西444www高清大胆 | 国产日韩精品在线 | 99精品欧美一区二区三区 | 久久中文网 | 在线观看91| 国产美女无遮挡永久免费 | 激情网站 | 国产成人精品一区二区三区福利 | 黄色大全免费观看 | 波多野结衣电影一区二区 | 综合伊人久久 | 国产美女精品人人做人人爽 | 国产精品久久久久久久午夜片 | 2000xxx影视| 99久久这里有精品 | 天天操 夜夜操 | 狠狠干网 | 免费亚洲婷婷 | 91av原创| 欧美在线视频第一页 | 亚洲色图 校园春色 | 少妇自拍av | 国产成人久久精品亚洲 | 日韩二级毛片 | 亚洲色图 校园春色 | 久久免费看视频 | 亚洲美女精品视频 | 在线a视频 | 久久人人添人人爽添人人88v | 亚洲免费在线看 | 久久影视一区 | 麻豆传媒一区二区 | 午夜精品一区二区三区免费 | 久久久www成人免费毛片麻豆 | 99精品黄色片免费大全 | 日本黄色大片免费看 | www.亚洲精品视频 | 国产啊v在线观看 | 99色免费视频 | 在线观看中文字幕亚洲 | 欧美精品一区二区三区四区在线 | 97品白浆高清久久久久久 | 国产尤物在线视频 | 成人免费观看大片 | 五月天婷婷免费视频 | www视频免费在线观看 | 天天操天天插 | 日韩美av在线 | 国产成人333kkk | 日韩在线欧美在线 | 99九九视频| 国内久久视频 | 麻豆免费观看视频 | 天天干 天天摸 天天操 | 美女视频免费一区二区 | www成人精品 | 国产 日韩 在线 亚洲 字幕 中文 | 日韩高清三区 | 亚洲一区二区精品视频 | 久久一区二 | 色狠狠狠| 黄色软件在线观看 | 色视频成人在线观看免 | 国产精品欧美久久久久三级 | 国产91大片 | 久久国产精品二国产精品中国洋人 | 黄色av电影免费观看 | 精品久久免费 | 在线免费高清 | 99精品在线视频播放 | 天天操综合网 | 欧美精品国产综合久久 | 国内综合精品午夜久久资源 | 亚洲伦理一区 | 成人欧美一区二区三区黑人麻豆 | 青青草华人在线视频 | 在线视频你懂 | 国产aa精品 | 国产美女久久久 | 在线播放 日韩专区 | 在线亚洲人成电影网站色www | a视频在线播放 | 国产不卡视频在线播放 | 国产精品久久久区三区天天噜 | 91成年视频 | 欧美日韩视频在线观看一区二区 | 成人国产在线 | 中文字幕第一页在线vr | 美国三级黄色大片 | 精品国产一区二区三区久久 | 天堂va在线高清一区 | 日韩综合一区二区 | 国产精品人成电影在线观看 | 国产黄影院色大全免费 | 免费久久久久久 | 毛片随便看 | 91在线视频播放 | 美女视频免费精品 | 国产精品美女999 | 免费又黄又爽视频 | 五月天婷亚洲天综合网鲁鲁鲁 | 在线亚洲欧美日韩 | 国产91在线播放 | 天天草天天草 | 欧美精品视 | 国产黄免费 | 天天操天天色天天射 | 亚洲精品午夜久久久久久久久久久 | 精品视频免费看 | 免费久久精品视频 | 国语对白少妇爽91 | 中文字幕首页 | 综合色伊人 | 中文字幕在线播放日韩 | 久久伊人操 | 色a4yy| 国产二区视频在线观看 | 中文字幕av在线免费 | 99国内精品久久久久久久 | 黄色美女免费网站 | 欧美日韩精品在线一区二区 | 天天色天天操综合 | 一区二区久久久久 | 天天人人综合 | 91视频 - 88av | 亚洲视频免费在线看 | 少妇搡bbbb搡bbb搡69 | 免费久久网站 | 国产色久 | 久久免费观看少妇a级毛片 久久久久成人免费 | 在线观看av的网站 | 久久久久久久久影视 | 国内精品久久久久影院男同志 | 国产中文字幕视频在线观看 | 狠狠五月天 | 99久久婷婷国产一区二区三区 | 精品不卡视频 | 亚洲三级视频 | 一区二区在线不卡 | 国产一区福利 | 国产视频在线免费观看 | 日日干夜夜干 | 欧美日高清视频 | 人人射| 久久精品国产v日韩v亚洲 | 国产美女视频免费 | 久久久久久久国产精品视频 | 日韩精品免费在线观看视频 | 日b视频在线观看网址 | 国产香蕉视频在线观看 | 欧美激情综合五月 | 高清视频一区 | 黄色小说免费在线观看 | 亚洲日本激情 | 成人黄色免费在线观看 | 久久久久成人精品 | 中文字幕日韩精品有码视频 | 日韩欧美国产免费播放 | 天天操·夜夜操 | 成人av片在线观看 | 天天搞天天干天天色 | 五月色婷 | 亚洲一区二区视频在线 | 91c网站色版视频 | 亚洲欧洲国产视频 | 丝袜+亚洲+另类+欧美+变态 | av三级在线免费观看 | 韩国av免费| 亚洲一区二区三区毛片 | 婷婷六月丁香激情 | 日韩av快播电影网 | 91网址在线 | 国产精品成人国产乱 | 91探花系列在线播放 | 国产午夜亚洲精品 | 美女在线免费视频 | 黄色在线观看污 | 国产精品一区二区在线 | 在线免费亚洲 | 亚洲午夜精品久久久 | 99riav1国产精品视频 | 玖玖在线免费视频 | 在线视频免费观看 | 五月天综合网站 | 亚洲人久久 | 狠狠干婷婷 | 99精品视频在线观看免费 | 亚洲精品国偷拍自产在线观看蜜桃 | 麻豆传媒在线免费看 | 成人毛片在线观看 | a国产精品 | 91成人网在线观看 | av看片网| 超碰在线网 | 在线观看视频免费大全 | 美女视频黄网站 | 日批视频在线播放 | 最近的中文字幕大全免费版 | 麻豆91在线看 | a成人v在线 | 99精品国产福利在线观看免费 | 92国产精品久久久久首页 | 免费在线成人 | 国产精品大尺度 | 最近中文字幕第一页 | 99久久精品免费看国产一区二区三区 | 亚洲aⅴ免费在线观看 | 色噜噜噜| 中文av字幕在线观看 | 中文字幕 国产专区 | 中文字幕高清有码 | 在线观看 国产 | 国产精品毛片久久蜜 | 欧美 亚洲 另类 激情 另类 | 亚洲日本三级 | www.97视频 | 激情伊人五月天久久综合 | 91看片淫黄大片一级在线观看 | 四虎国产永久在线精品 | 欧美不卡在线 | 91精品入口 | 国产精品久久久视频 | 日韩一级电影在线 | 国产一区二区久久 | 中文字幕婷婷 | 久久美女精品 | 日韩高清在线一区 | 久久久久久免费 | 久久毛片高清国产 | 五月婷婷一区二区三区 | 婷婷深爱 | 中文一二区 | 久久久免费国产 | 一二三区在线 | 国产亚洲视频系列 | 日韩免费视频一区二区 | 日韩免费电影在线观看 | 91成人午夜 | 久久综合婷婷 | 国产成人av电影在线 | 亚洲精品视频免费在线 | 国产97碰免费视频 | 在线观看黄色国产 | 国产伦理精品一区二区 | 伊人久在线 | 亚洲精品日韩在线观看 | 久久一精品| 韩国三级av在线 | 精品国内自产拍在线观看视频 | 国产精品中文久久久久久久 | 欧美中文字幕第一页 | 国产精品情侣视频 | 一区二区三区免费在线 | 欧美性生交大片免网 | 精品国产大片 | 黄色大片免费播放 | 中文字幕在线看 | 亚洲黄色大片 | 狠狠操综合 | 探花视频免费观看 | 国产一区二区三区高清播放 | 久久歪歪 | 国产亚洲视频系列 | 日韩毛片精品 | 丁香久久婷婷 | 色网影音先锋 | 超碰97成人 | 一区二区三区在线视频111 | 波多野结衣小视频 | 免费黄色看片 | 91精品综合在线观看 | 日韩精品一区二区三区在线视频 | 国产精品丝袜久久久久久久不卡 | 美女黄久久 | 日本 在线 视频 中文 有码 | 成人av在线一区二区 | 91免费国产在线观看 | 91c网站色版视频 | 国产麻豆剧传媒免费观看 | 免费麻豆网站 | 久草在线综合网 | 欧美成人精品三级在线观看播放 | 少妇精69xxtheporn| 欧美日韩亚洲在线 | 国产成人福利在线观看 | 国产精品欧美日韩在线观看 | 91成人精品一区在线播放 | 精品久久久久国产免费第一页 | 国产精品福利在线播放 | 五月婷婷六月丁香 | 国产成人精品福利 | 911香蕉 | 国产精品久久久久影院日本 | 欧美日韩一区二区免费在线观看 | 99精品免费在线 | www色婷婷com| 97超碰色 | 午夜精品久久久久久久久久久 | 水蜜桃亚洲一二三四在线 | 在线精品在线 | 久久色中文字幕 | 免费成人在线视频网站 | 天天干中文字幕 | 天天天天天干 | 亚洲一级电影视频 | 99视频在线免费观看 | 婷婷久久网站 | 国产经典av | 久久er99热精品一区二区 | 狠狠干综合网 | 天天插天天狠 | 久久视频精品在线观看 | 免费观看一级视频 | 激情 亚洲| 日韩在线视频网站 | 玖玖玖国产精品 | 亚洲欧美日韩国产一区二区三区 | 久久免费中文视频 | 国产精品福利视频 | 天天操夜夜爱 | 久久国产精品久久w女人spa | 久久毛片高清国产 | 久久女教师| 久久精品看 | 激情丁香5月| 美女在线黄 | 午夜少妇 | 国产精品成人av在线 | 在线看不卡av | 国产亚洲精品女人久久久久久 | 国产精品青青 | 视频在线播放国产 | 久久久免费电影 | 黄色小说视频网站 | 欧美在线不卡一区 | 亚洲精品九九 | 狠狠久久 | 99久久国产免费免费 | 国产偷v国产偷∨精品视频 在线草 | 国产精彩视频 | 玖玖玖影院| 天天爽网站 | 男女靠逼app| 九色福利视频 | 成年人看片 | www日韩在线 | 精品一区二区三区久久久 | 午夜精品久久久久久中宇69 | 成年人在线观看免费视频 | 手机在线永久免费观看av片 | 国产精品成人久久久久 | 久久久天堂 | 日韩精品免费在线观看 | 干干干操操操 | 国产裸体视频网站 | 国产99久久精品一区二区永久免费 | 中文字幕亚洲不卡 | 麻豆视频入口 | 久久97精品 | 日韩欧美在线综合网 | 日日夜夜骑 | 91精品视频一区二区三区 | 亚洲精品久久在线 | 天天夜操 | 亚洲电影图片小说 | 国产日韩欧美在线 | 日韩成人精品一区二区 | 日本在线观看一区二区三区 | 日韩午夜小视频 | www.狠狠操 | 久久这里只精品 | 亚洲精品视频免费在线 | 免费视频二区 | 操操综合| 亚洲国产精品第一区二区 | av经典在线 | 91一区啪爱嗯打偷拍欧美 | 成人一级 | 黄色毛片大全 | 日韩大片在线免费观看 | 亚洲 欧美日韩 国产 中文 | 91麻豆文化传媒在线观看 | 精品毛片久久久久久 | 青青草久草在线 | 99精品欧美一区二区 | 中文字幕精品一区二区精品 | 免费看黄20分钟 | 久久露脸国产精品 | 久久久久久久久综合 | 99视频精品视频高清免费 | 在线看片一区 | 不卡视频在线 | 中文字幕在线播放第一页 | 国产成人在线网站 | 五月天综合网站 | 欧美综合在线视频 | 射久久久 | 国产精品久久久久久久久久久久午夜 | 国产成人黄色网址 | 96精品在线| 射九九| 激情av资源网 | 久久色亚洲| 97在线免费观看视频 | 国产视频一二区 | 亚洲乱码在线 | 天天综合网久久综合网 | 日韩电影中文字幕在线 | 中文字幕丝袜 | 久久久精品99 | 91热爆在线观看 | 久草在线视频中文 | 日韩欧美高清视频在线观看 | 免费日韩av电影 | 中文字幕在线观看免费高清电影 | 国产精品自产拍在线观看 | 有没有在线观看av | 2023av| 香蕉视频在线免费 | 99精品偷拍视频一区二区三区 | 另类老妇性bbwbbw高清 | 国产精品久久久久久一二三四五 | 国产一二三在线视频 | 碰超在线 | 丁香花在线视频观看免费 | 久久精品视频在线播放 | 2023国产精品自产拍在线观看 | 日韩在线观看第一页 | 国内精品久久久久久 | 亚洲精选在线 | 国产精品久久一区二区三区, | 偷拍福利视频一区二区三区 | 久久国产成人午夜av影院宅 | 国产一级在线 | 日韩免费看| 国产精品久久久久一区二区三区共 | 麻豆91在线看 | 久久免费视频在线观看30 | 久久免费国产精品1 | 国产九九热视频 | 欧美一级激情 | 久久免费视频6 | 一区二区在线电影 | 国产69精品久久久久久久久久 | 国产69精品久久99不卡的观看体验 | 在线国产日本 | 欧美在线日韩在线 | 五月丁色 | 99热都是精品 | 91麻豆精品国产91久久久使用方法 | 婷婷五月色综合 | 最近免费中文视频 | 久久午夜精品 | 国产一区二区久久久 | 免费色网站 | 中文字幕在线视频精品 | 久草视频在线免费 | 亚洲成人av一区 | 日韩动态视频 | 手机在线黄色网址 | 日韩欧美在线中文字幕 | 精品国产一区二区三区在线观看 | 麻豆91精品91久久久 | 日韩欧美在线观看一区 | 国产美女精品 | 日韩一区二区三区高清免费看看 | 国产精品欧美久久久久三级 | 草久草久 | 麻豆精品在线 | 久久手机免费观看 | 2020天天干天天操 | 日韩精品视频在线观看网址 | 欧洲一区二区在线观看 | 久久久久久久久久久精 | 啪啪资源 | 日韩欧美精品在线观看视频 | 国产精品久久99精品毛片三a | 黄色成人在线 | 成人黄色片免费 | av在线电影网站 | 国产福利精品一区二区 | 色婷婷久久 | 精品在线亚洲视频 | 精品无人国产偷自产在线 | 欧亚日韩精品一区二区在线 | 999电影免费在线观看 | 免费视频a| 国产精品高清在线 | 欧美a级成人淫片免费看 | 97视频免费看| 久久精品一区二 | 久久影视中文字幕 | 九九免费观看视频 | 欧美精品久久久久久 | 永久免费视频国产 | 色丁香综合 | 国产999视频在线观看 | 91丨九色丨国产女 | 久久精品99北条麻妃 | 日韩精品一区二区三区不卡 | 狠狠操狠狠干天天操 | 国产美女视频一区 | 五月综合在线观看 | 久久久久国产精品厨房 | 日韩特黄av | 国产精品99久久久久久久久 | 国产精品永久久久久久久久久 | 色婷婷国产精品一区在线观看 | 中文字幕专区高清在线观看 | 999国产 | 91手机视频 | 久久国产精品成人免费浪潮 | 爱av在线网 | av一区二区三区在线 | 国产91精品一区二区麻豆网站 | 黄色在线看网站 | 国产69精品久久久久久 | 色99久久| 99精品视频免费观看视频 | 玖玖视频精品 | 97在线精品 | 国产精品视频99 | 精品自拍网 | 久久久天天操 | 国产剧情一区二区在线观看 | 亚洲aⅴ在线观看 | 在线精品一区二区 | 就要色综合 | 色视频在线免费 | 在线视频一区二区 | 国产视频一 | 91大片网站 | 久久久精品视频成人 | 国产精品免费视频观看 | 精品福利网站 | 久久精品久久99精品久久 | 天堂va欧美va亚洲va老司机 | 久久人人爽av | 欧美日韩国产一区二区三区在线观看 | 五月天天天操 | 粉嫩一区二区三区粉嫩91 | 国产一二区视频 | 亚洲精品www. | 米奇四色影视 | 日韩免费一区二区 | 伊香蕉大综综综合久久啪 | 射综合网 | 97超碰人人模人人人爽人人爱 | 午夜视频播放 | 日日爽日日操 | a在线观看视频 | 国产精品黄色影片导航在线观看 | 日韩免费观看高清 | 人人狠狠综合久久亚洲婷 | 在线观看免费成人 | 欧美激情视频三区 | 国产精品视频久久 | 久草在线资源免费 | 日韩在线一二三区 | 国产精品久久久久久久久久久久冷 | 久要激情网 | 天天综合天天综合 | 狠狠色丁香婷婷综合久小说久 | 亚洲欧美日本A∨在线观看 青青河边草观看完整版高清 | 午夜精品久久久久久久久久久久 | 99在线精品视频 | 亚洲精品字幕 | 久久国产色 | 亚洲国产大片 | 婷婷国产精品 | 又黄又刺激又爽的视频 | 91av在线视频播放 | 日韩久久在线 | 国产精品永久 | 国产免费视频一区二区裸体 | 爱射综合| 日本一区二区不卡高清 | 亚洲视频大全 | 亚洲国产视频直播 | 免费无遮挡动漫网站 | 四虎在线观看 | 亚洲欧美日本A∨在线观看 青青河边草观看完整版高清 | 国产成人福利 | 丁香综合五月 | 免费看污的网站 | 久久午夜视频 | 在线观看成人一级片 | 国产精品影音先锋 | 99热这里只有精品国产首页 | 欧美日韩在线免费观看 | 精品久久久精品 | 久久黄色a级片 | 99久久久久久久久久 | 天天人人 | 蜜桃久久久 | 波多野结衣在线观看视频 | 九九色网| 91九色在线 | av在线日韩| 怡红院成人在线 | 不卡的一区二区三区 | 精品国产一区二区三区四 | 日日干夜夜操视频 | 麻豆视频免费在线观看 | 亚洲激情视频 | 久久99国产一区二区三区 | 欧美一级艳片视频免费观看 | 日韩欧美在线视频一区二区 | 国产一区二区在线播放视频 | 免费在线看成人av | www天天干 | 91麻豆国产 | 久久久久久久久久久网站 | 久久影视精品 | 国产午夜免费视频 | 97在线视频免费 | 国产成人av网站 | 国产在线毛片 | 91精选| 欧美日韩国内在线 | 国产日韩在线观看一区 | 午夜成人免费影院 | 久久www免费视频 | 91精品国产91热久久久做人人 | 色综合久久网 | 狠狠的日日 | 九九激情视频 | 久草在线免费电影 | 国产精品999久久久 久产久精国产品 | 国产精品自拍av | 国产视频精品网 | 亚洲理论电影 | 久久精品成人热国产成 | 人人舔人人插 | 日韩精品视频免费专区在线播放 | 在线观看91av | 日韩中文在线字幕 | 国产亚洲欧美精品久久久久久 | 福利电影久久 | 欧美精选一区二区三区 | 欧美日韩免费视频 | 精品国产亚洲一区二区麻豆 | 欧美日韩精品在线观看视频 | 精品视频123区在线观看 | 亚洲视频高清 | 人人看人人草 | 国产色视频网站2 | 99色免费| 色婷婷午夜 | 麻豆国产露脸在线观看 | 97**国产露脸精品国产 | 久久97超碰 | 久久久久国产成人精品亚洲午夜 | 热久在线| 国产一区二区不卡视频 | 精品视频| 韩国精品在线观看 | 婷婷丁香av | 99久久久国产精品免费99 | 国产青青青 | 国产福利一区二区在线 | 狠狠狠色丁香综合久久天下网 | 国产呻吟在线 | 精品福利视频在线观看 | 免费看一及片 | 成年人在线观看视频免费 | 午夜精品久久久久久久久久久久久久 | 欧美日韩高清在线观看 | 日日操操操 | 日日夜夜狠狠操 | 四虎成人在线 | 国产精品欧美久久久久天天影视 | 久草网视频在线观看 | 人人干网| 国产一区免费视频 | 欧美久久久久久久久 | 日日夜操 | 国产精品999久久久 久产久精国产品 | 久久精品视频免费观看 | 日韩一区二区免费播放 | 亚洲国产日韩欧美在线 | 欧美一级片在线 | av字幕在线 | 国产一级三级 | 亚洲视频 在线观看 | 国产成人99久久亚洲综合精品 | 黄色av高清 | 视频91在线 | 亚洲va欧美va人人爽春色影视 | 久久久国产精品一区二区三区 | 午夜久久影视 | 在线观看视频h | 免费亚洲精品 | 欧美午夜视频在线 | 国内精品久久久精品电影院 | 成人一级片视频 | 五月综合色婷婷 | 亚洲国产精品电影在线观看 | 欧美精品乱码久久久久久按摩 | 一区二区视频在线免费观看 | 国产成人精品免费在线观看 | 免费成人短视频 | 久久官网 | 欧美日韩成人一区 | 国产亚洲片 | 久久久久久久久久久久久久电影 | 国产乱码精品一区二区三区介绍 | 免费观看完整版无人区 | 毛片网站免费 | 日韩av在线网站 | 免费观看黄色av | 欧美精品一区二区三区四区在线 | 日本久久久久久久久久 | 黄网站免费看 | 96久久欧美麻豆网站 | 久久久色 | 国产人成在线视频 | 久久新视频 | 国产精品久久久精品 | 日韩在线免费电影 | 国产aaa免费视频 | a黄色| 99精品在线 | 久久极品| 色中色亚洲| 美女视频国产 | 欧美福利久久 | 国产精品久久久久久久久久免费看 | 黄色av播放 | 美女视频黄是免费的 | 黄色一及电影 | 久久中文欧美 | 免费视频91蜜桃 | 久久免费看视频 | 成人小视频在线观看免费 | 狠狠色丁香婷婷 | 久久久午夜视频 | 天天天干夜夜夜操 | 玖玖视频免费在线 | 国产精品毛片一区二区 | 久久久精品网站 | 成人av影院在线观看 | 久久99操| 日韩精品国产一区 | 黄色在线免费观看网址 | 麻豆精品在线 | 精品视频免费 | 99精品视频在线免费观看 | 久久免费视频这里只有精品 | 麻豆免费精品视频 | 日韩在线观看第一页 | 久草在线观看视频免费 | 91大神精品视频在线观看 | 丝袜制服综合网 | 亚洲日本黄色 | 天天综合网天天综合色 | 日本精品一区二区三区在线观看 | 99视频在线观看视频 | 欧美在线观看视频一区二区 | 日韩免费观看av | 中文字幕免费高清 | 亚洲四虎在线 | 91成人精品在线 | 亚洲国产精品电影 | 精品久久久99 | 欧美精品在线观看 | 久久艹艹 | 99国产精品一区 | www.久久com| 午夜国产一区二区 | 国产专区视频在线观看 | 国产人在线成免费视频 | 91香蕉视频黄 | 最近中文字幕 | 天天操天天射天天 | 精品乱码一区二区三四区 | 92国产精品久久久久首页 | 日韩欧美在线观看一区二区 | 毛片的网址 | 香蕉久久久久 | 天天av天天| 中文在线a在线 | 丰满少妇一级 | 国偷自产中文字幕亚洲手机在线 | 99精品99 | 久久精品麻豆 | 日韩在线观看视频免费 | 国产精品系列在线播放 | 中文字幕资源在线观看 | 日韩av偷拍 | 亚洲免费av在线播放 | 久要激情网 | 免费看的毛片 | 色国产精品 | 国产福利在线免费观看 | 国产精品18videosex性欧美 | 韩国精品在线观看 | 亚洲日本va午夜在线影院 | 国产日韩视频在线观看 | 国产成人99久久亚洲综合精品 | 天天碰天天操视频 | 欧美精品一区二区蜜臀亚洲 | 黄污网站在线 | 最近中文字幕久久 | 精品人妖videos欧美人妖 | 亚洲国产午夜 | 久久久久国产精品免费 | 99视频一区 | 三上悠亚一区二区在线观看 | 欧美日韩另类在线观看 | 不卡视频在线 | 日韩av电影免费在线观看 | 激情综合婷婷 | 国产高清福利在线 | 国产高清在线观看av | 久久五月天色综合 | 五月天中文在线 | 亚洲精品男人天堂 | av电影一区二区三区 | 国产精品免费一区二区三区 | 96精品视频 | 久久精品日本啪啪涩涩 | 中文字幕第一页在线播放 | 成人av电影免费在线播放 | 国产免费黄视频在线观看 | 自拍超碰在线 | 最新国产精品久久精品 | 亚洲成人av在线播放 | 久久成人一区二区 | 亚洲欧美视频 | 中文字幕永久免费 | 欧美日韩在线免费观看视频 | 国产精品久久一 | 婷婷激情五月综合 | av亚洲产国偷v产偷v自拍小说 | 制服丝袜亚洲 | 在线观看完整版免费 | 欧美日韩在线观看一区二区 | 日韩欧美一区二区三区黑寡妇 | 国产专区精品 | 色综合色综合色综合 | 国产中文字幕国产 | 久久久久久久久久久久国产精品 | 黄色电影在线免费观看 | 国产在线a视频 | 免费手机黄色网址 | 免费看的黄网站 | 操天天操| 久久久久免费网站 | 精品国产免费久久 | 亚洲1区在线 | 日韩av一卡二卡三卡 | 在线一二三四区 | 一区在线观看 | 麻豆视频在线观看免费 | 国产永久免费 | 91女神的呻吟细腰翘臀美女 | 国产二区av | 在线观看免费成人 | 国产成人精品综合久久久久99 | 特级aaa毛片 | 国产精品毛片一区二区 | 特级黄色电影 | 黄色www免费| 亚洲欧美日韩精品一区二区 | 亚洲在线不卡 | 精品在线观看视频 | 激情五月婷婷综合网 | 欧美性另类 | 亚洲第一伊人 | 91精品国产乱码在线观看 | av福利网址导航大全 | 国产麻豆电影在线观看 | 国产一区在线免费观看视频 | 国产黄免费在线观看 | 日韩欧美在线一区二区 | 亚洲全部视频 | 日韩在线播放欧美字幕 | 久久网站最新地址 | www.啪啪.com | 欧美成人高清 | 国产视频首页 | 丁香婷婷在线观看 | 8x成人在线 | 手机在线中文字幕 | 免费国产视频 | 国产成人一区在线 | 伊香蕉大综综综合久久啪 | 五月婷婷在线播放 | h网站免费在线观看 | 国产99免费视频 | 色视频网站在线 | 国产精品午夜av | 激情综合亚洲精品 | 久久精品导航 | 久久99视频免费观看 | 黄色网大全 | 天天操天 | av解说在线观看 | 国产精品毛片一区二区在线看 | 国产精品视频999 | 亚洲一级免费观看 | 欧美在线视频不卡 | 国语黄色片 | 色婷婷国产精品一区在线观看 | 99热这里只有精品1 av中文字幕日韩 | 天天操天天射天天爱 | 一本一本久久a久久精品综合 | 久久欧美精品 | 操久| 成人免费在线视频 | 豆豆色资源网xfplay | 国产亚洲精品成人av久久影院 | 一级一片免费观看 | 久久久高清一区二区三区 | 国产不卡片 | 成年人视频免费在线播放 | av黄色国产 | 丁香激情五月婷婷 |