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

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 >

PulseAudio 设计和实现浅析

發布時間:2024/4/11 64 豆豆
生活随笔 收集整理的這篇文章主要介紹了 PulseAudio 设计和实现浅析 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

PulseAudio 是一個 POSIX 操作系統的音頻服務器系統,它是我們的音頻應用程序訪問系統音頻設備的代理。它是所有相關的現代 Linux 發行版的組成部分,并被多個供應商用在了各種各樣的移動設備中。它在應用程序和硬件設備間傳遞音頻數據時,可以對音頻數據執行一些高級操作。比如,把音頻數據傳給不同的機器,修改樣本格式或通道數,或者混音多路音頻到一路輸入/輸出,這些用 PulseAudio 實現都很簡單。PulseAudio 主頁。

很多學習了編程語言,操作系統,計算機網絡和標準庫及操作系統 API 的同學都會有一些疑問:自己好像已經學習了不少知識了,但這些知識能用來做些什么,基于這些知識怎么才能實現一個有一定價值的應用,應用程序究竟是如何訪問硬件設備的,如音頻設備、視頻設備等。

通過了解 PulseAudio 系統的設計和實現,可以讓我們看到許許多多 POSIX 系統,特別是 Linux 系統平臺下,開發一個實用的軟件,所需要了解的許許多多跟系統相關的點,如基于 udev 的設備發現和設備狀態變化監聽,各種各樣的進程間通信方式,如基于 IPv4 和 IPv6 的 TCP,Unix 域 socket,D-Bus 等,訪問音頻設備的 alsa 架構等。

這里簡單看一下 PulseAudio 的設計和實現。

PulseAudio 的代碼下載及編譯

編譯出調試版代碼,并在調試器中把代碼運行起來,可以為我們研究一個軟件項目提供極大的便利。我們在 Ubuntu 20.04 上編譯 PulseAudio 并把它跑起來。

我們可以通過 Git 下載代碼,當前 PulseAudio 最新發布的版本為 14.2,我們切換到這個分支:

$ git clone https://github.com/pulseaudio/pulseaudio.git $ git checkout -t origin/stable-14.x

PulseAudio 是一個通過 meson + ninja 來構建的工程,更多關于 meson 構建系統的信息,可以參考 meson 主頁。在開始構建 PulseAudio 前,需要先下載 meson:

$ sudo apt install meson

還需要下載 PulseAudio 本身及測試用例編譯的依賴庫:

$ sudo apt-get install libltdl-dev libsndfile1-dev cmake libsbc-dev check libbluetooth-dev

編譯 PulseAudio

$ cd pulseaudio $ mkdir build $ cd build/ $ meson .. $ ninja -C . . . . . . . . . . Database: tdbLegacy Database Entry Support: truemodule-stream-restore:Clear old devices: falseRunning from build tree: trueSystem User: pulseSystem Group: pulseAccess Group: pulse-access meson.build:899: WARNING: You do not have speex support enabled. It is strongly recommended that you enable speex support if your platform supports it as it is the primary method used for audio resampling and is thus a critical part of PulseAudio on that platform. Build targets in project: 167Found ninja-1.8.2 at ~/bin/depot_tools/ninja

PulseAudio 默認已經運行在 Ubuntu 20.04 中了。為了能夠運行我們編譯出來的 PulseAudio 服務,需要先暫停系統中已經存在的 pulseaudio 服務:

systemctl --user stop pulseaudio.socket systemctl --user stop pulseaudio.service

注意,執行這些命令不需要超級用戶權限 sudo。

調試完成之后,還可以通過如下命令重新啟動系統的 pulseaudio 服務:

systemctl --user start pulseaudio.socket systemctl --user start pulseaudio.service

通過如下命令,可以將 pulseaudio 服務運行起來:

~/data/opensource/pulseaudio$ build/src/daemon/pulseaudio -n -F build/src/daemon/default.pa -p $(pwd)/build/src/modules/ W: [pulseaudio] caps.c: Normally all extra capabilities would be dropped now, but that's impossible because PulseAudio was built without capabilities support. N: [pulseaudio] daemon-conf.c: Detected that we are run from the build tree, fixing search path. N: [pulseaudio] alsa-util.c: Disabling timer-based scheduling because running inside a VM. E: [alsa-sink-ES1371/1] alsa-sink.c: ALSA 提醒我們在該設備中寫入新數據,但實際上沒有什么可以寫入的! E: [alsa-sink-ES1371/1] alsa-sink.c: 這很可能是 ALSA 驅動程序 'snd_ens1371' 中的一個 bug。請向 ALSA 開發人員報告這個問題。 E: [alsa-sink-ES1371/1] alsa-sink.c: 我們因 POLLOUT 被設置而喚醒 -- 但結果是 snd_pcm_avail() 返回 0 或者另一個小于最小可用值的數值。 N: [pulseaudio] alsa-util.c: Disabling timer-based scheduling because running inside a VM. E: [pulseaudio] stdin-util.c: Unable to read or parse data from client. E: [pulseaudio] module.c: Failed to load module "module-gsettings" (argument: ""): initialization failed.

PulseAudio 的設計和實現

pulse audio 工程有許多測試用例,可以用于幫助我們了解這個工程設計與實現。其中許多測試用例是單元測試,主要用于測試 pulse audio 基礎組件的 API,但也有一些測試用例會跑完整的錄制和播放流程,可以讓我們聽到通過 pulse audio 播放出來的音頻數據,這些測試用例有 lo-latency-test (pulseaudio/src/tests/lo-latency-test.c----pulseaudio/build/src/tests/lo-latency-test) 和 sync-playback (pulseaudio/src/tests/sync-playback.c----pulseaudio/build/src/tests/sync-playback) 等。

這里我們通過觀察 sync-playback 和 pulseaudio 服務的交互來了解 PulseAudio 的實現。

從進程角度來看,pulse audio 是客戶端/服務器架構的。pulse audio 提供一個常駐系統的服務進程,它接收音頻相關的操作請求命令,管理音頻設備,實現與音頻設備間的數據交互等。各種接入 pulse audio 庫,通過 pulse audio API 訪問音頻設備的應用,是 pulse audio 系統服務的客戶端,它們通過 pulse audio 系統服務向音頻播放設備寫入數據將聲音播放出來,通過 pulse audio 系統服務讀取錄制設備以獲得采集的音頻數據。

pulse audio 系統服務和它的客戶端可以通過多種方式實現進程間通信。對于命令的傳遞,可以通過 UNIX 域 socket,IPv4 socket,IP v6 socket,Linux 系統的 D-Bus 等,對于大塊的音頻 PCM 數據,則通過共享內存來傳遞。

這里看一下 pulse audio 數據傳輸的基本過程。

  • 發生在 pulse audio 系統服務進程中的 socket 監聽
  • #0 pa_socket_server_new_unix ()at ../src/pulsecore/socket-server.c:175 #1 module_native_protocol_unix_LTX_pa__init () at ../src/modules/module-protocol-stub.c:330 #2 pa_module_load ()at ../src/pulsecore/module.c:191 #3 pa_cli_command_load ()at ../src/pulsecore/cli-command.c:437 #4 pa_cli_command_execute_line_stateful () at ../src/pulsecore/cli-command.c:2141 #5 pa_cli_command_execute_file_stream ()at ../src/pulsecore/cli-command.c:2181 #6 pa_cli_command_execute_file ()at ../src/pulsecore/cli-command.c:2212 #7 pa_cli_command_execute_line_stateful () at ../src/pulsecore/cli-command.c:2106 #8 pa_cli_command_execute ()at ../src/pulsecore/cli-command.c:2238 #9 main () at ../src/daemon/main.c:1112

    pulseaudio 系統服務在進程啟動時,會執行傳入的配置文件 build/src/daemon/default.pa 中的命令。如配置文件中的 load-module module-native-protocol-unix 行會使得 pulseaudio 系統服務去加載 module-native-protocol-unix 模塊,在這個模塊中會起動監聽路徑為 /run/user/1000/pulse/native 的 Unix 域 socket,詳細的執行路徑如上面的堆棧信息,并設置連接建立等事件發生時的回調 socket_server_on_connection_cb() 等。Unix 域 socket 由模塊 module-native-protocol-unix 管理。

  • 客戶端建立連接
  • #0 pa_socket_client_new_unix ()at ../src/pulsecore/socket-client.c:227 #1 pa_socket_client_new_string () at ../src/pulsecore/socket-client.c:459 #2 try_next_connection () at ../src/pulse/context.c:884 #3 pa_context_connect () at ../src/pulse/context.c:1046 #4 sync_playback_test () at ../src/tests/sync-playback.c:165 #5 main () at ../src/tests/sync-playback.c:184

    客戶端應用程序,在調用 pa_context_connect() 接口時,會去連接 pulseaudio 系統服務監聽的 Unix 域 socket。

  • pulseaudio 系統服務將命令處理程序和 IO 通道關聯
  • pulseaudio 系統服務在收到客戶端的連接時,socket-server 在其回調中,根據新連接的文件描述符創建 pa_iochannel,并執行回調 socket_server_on_connection_cb(),這個回調通過 pa_native_protocol_connect() 函數為連接創建所需的資源,如 pa_client,pa_native_connection 和流 pa_pstream 等,為流 pa_pstream 設置各種事件的回調,并將命令處理程序關聯到流上:

    #0 pa_native_protocol_connect () at ../src/pulsecore/protocol-native.c:5194 #1 socket_server_on_connection_cb ()at ../src/modules/module-protocol-stub.c:202 #2 callback ()at ../src/pulsecore/socket-server.c:143 #3 dispatch_pollfds () at ../src/pulse/mainloop.c:655 #4 pa_mainloop_dispatch () at ../src/pulse/mainloop.c:896 #5 pa_mainloop_iterate () at ../src/pulse/mainloop.c:927 #6 pa_mainloop_run () at ../src/pulse/mainloop.c:942 #7 main () at ../src/daemon/main.c:1170

    pa_native_protocol_connect() 函數的部分代碼如下:

    c->pstream = pa_pstream_new(p->core->mainloop, io, p->core->mempool);pa_pstream_set_receive_packet_callback(c->pstream, pstream_packet_callback, c);pa_pstream_set_receive_memblock_callback(c->pstream, pstream_memblock_callback, c);pa_pstream_set_die_callback(c->pstream, pstream_die_callback, c);pa_pstream_set_drain_callback(c->pstream, pstream_drain_callback, c);pa_pstream_set_revoke_callback(c->pstream, pstream_revoke_callback, c);pa_pstream_set_release_callback(c->pstream, pstream_release_callback, c);c->pdispatch = pa_pdispatch_new(p->core->mainloop, true, command_table, PA_COMMAND_MAX);c->record_streams = pa_idxset_new(NULL, NULL);c->output_streams = pa_idxset_new(NULL, NULL);
  • 客戶端發送命令請求 pulseaudio 系統服務創建資源及 pulseaudio 系統服務處理命令
  • 客戶端在連接建立完成后,通過新創建的連接發送幾個命令,請求 pulseaudio 系統服務分配資源。這些命令的交互主要包括如下這些:

    command_get_info command_get_info command_auth command_register_memfd_shmid command_set_client_name command_get_info command_get_info command_enable_srbchannel command_create_playback_stream command_create_playback_stream command_create_playback_stream command_create_playback_stream command_get_info command_get_info command_get_info command_get_info command_get_info command_get_info command_cork_playback_stream command_get_info command_get_info command_get_info

    忽略不是很重要的 command_get_info 命令。

    (1). 如前面我們提到的,客戶端和 pulseaudio 系統服務通過共享內存相互傳遞大塊的音頻數據。客戶端的共享內存對象在創建 pa_context 對象時打開,如:

    #0 sharedmem_create ()at ../src/pulsecore/shm.c:141 #1 pa_shm_create_rw () at ../src/pulsecore/shm.c:241 #2 pa_mempool_new () at ../src/pulsecore/memblock.c:849 #3 pa_context_new_with_proplist () at ../src/pulse/context.c:185 #4 pa_context_new () at ../src/pulse/context.c:103 #5 sync_playback_test () at ../src/tests/sync-playback.c:160 #6 main () at ../src/tests/sync-playback.c:184

    (2). pulseaudio 系統服務是在客戶端發送了 PA_COMMAND_AUTH 命令,在PA_COMMAND_AUTH 命令處理程序中打開的共享內存對象。客戶端在連接建立成功時發送 PA_COMMAND_AUTH 命令:

    #0 pa_pstream_send_packet () at ../src/pulsecore/pstream.c:442 #1 pa_pstream_send_tagstruct_with_ancil_data ()at ../src/pulsecore/pstream-util.c:45 #2 pa_pstream_send_tagstruct_with_creds ()at ../src/pulsecore/pstream-util.c:58 #3 setup_context () at ../src/pulse/context.c:650 #4 on_connection () at ../src/pulse/context.c:926 #5 do_call () at ../src/pulsecore/socket-client.c:159 #6 connect_defer_cb () at ../src/pulsecore/socket-client.c:172 #7 dispatch_defer () at ../src/pulse/mainloop.c:680 #8 pa_mainloop_dispatch () at ../src/pulse/mainloop.c:887 #9 pa_mainloop_iterate () at ../src/pulse/mainloop.c:927 #10 pa_mainloop_run () at ../src/pulse/mainloop.c:942 #11 sync_playback_test () at ../src/tests/sync-playback.c:170 #12 main () at ../src/tests/sync-playback.c:184

    (3). pulseaudio 系統服務在 PA_COMMAND_AUTH 命令處理程序中,創建 pa_srbchannel 對象,并打開共享內存對象:

    #0 sharedmem_create ()at ../src/pulsecore/shm.c:141 #1 pa_shm_create_rw () at ../src/pulsecore/shm.c:241 #2 pa_mempool_new (t) at ../src/pulsecore/memblock.c:849 #3 setup_srbchannel () at ../src/pulsecore/protocol-native.c:2502 #4 command_auth ()at ../src/pulsecore/protocol-native.c:2740 #5 pa_pdispatch_run ()at ../src/pulsecore/pdispatch.c:346 #6 pstream_packet_callback ()at ../src/pulsecore/protocol-native.c:5027 #7 do_read () at ../src/pulsecore/pstream.c:1020 #8 do_pstream_read_write () at ../src/pulsecore/pstream.c:260 #9 io_callback () at ../src/pulsecore/pstream.c:312 #10 callback ()at ../src/pulsecore/iochannel.c:158 #11 dispatch_pollfds () at ../src/pulse/mainloop.c:655 #12 pa_mainloop_dispatch () at ../src/pulse/mainloop.c:896 #13 pa_mainloop_iterate () at ../src/pulse/mainloop.c:927 #14 pa_mainloop_run () at ../src/pulse/mainloop.c:942 #15 main () at ../src/daemon/main.c:1170

    pulseaudio 系統服務在 PA_COMMAND_AUTH 命令的處理中,創建共享內存對象之后,還會通過 pa_pstream_register_memfd_mempool() 將共享內存對象的文件描述符傳給客戶端,如下面的 setup_srbchannel () 函數定義所示:

    static void setup_srbchannel(pa_native_connection *c, pa_mem_type_t shm_type) {pa_srbchannel_template srbt;pa_srbchannel *srb;pa_memchunk mc;pa_tagstruct *t;int fdlist[2];#ifndef HAVE_CREDSpa_log_debug("Disabling srbchannel, reason: No fd passing support");return; #endifif (!c->options->srbchannel) {pa_log_debug("Disabling srbchannel, reason: Must be enabled by module parameter");return;}if (c->version < 30) {pa_log_debug("Disabling srbchannel, reason: Protocol too old");return;}if (!pa_pstream_get_shm(c->pstream)) {pa_log_debug("Disabling srbchannel, reason: No SHM support");return;}if (c->rw_mempool) {pa_log_debug("Ignoring srbchannel setup, reason: received COMMAND_AUTH ""more than once");return;}if (!(c->rw_mempool = pa_mempool_new(shm_type, c->protocol->core->shm_size, true))) {pa_log_warn("Disabling srbchannel, reason: Failed to allocate shared ""writable memory pool.");return;}if (shm_type == PA_MEM_TYPE_SHARED_MEMFD) {const char *reason;if (pa_pstream_register_memfd_mempool(c->pstream, c->rw_mempool, &reason)) {pa_log_warn("Disabling srbchannel, reason: Failed to register memfd mempool: %s", reason);goto fail;}}pa_mempool_set_is_remote_writable(c->rw_mempool, true);srb = pa_srbchannel_new(c->protocol->core->mainloop, c->rw_mempool);if (!srb) {pa_log_debug("Failed to create srbchannel");goto fail;}pa_log_debug("Enabling srbchannel...");pa_srbchannel_export(srb, &srbt);/* Send enable command to client */t = pa_tagstruct_new();pa_tagstruct_putu32(t, PA_COMMAND_ENABLE_SRBCHANNEL);pa_tagstruct_putu32(t, (size_t) srb); /* tag */fdlist[0] = srbt.readfd;fdlist[1] = srbt.writefd;pa_pstream_send_tagstruct_with_fds(c->pstream, t, 2, fdlist, false);/* Send ringbuffer memblock to client */mc.memblock = srbt.memblock;mc.index = 0;mc.length = pa_memblock_get_length(srbt.memblock);pa_pstream_send_memblock(c->pstream, 0, 0, 0, &mc);c->srbpending = srb;return;fail:if (c->rw_mempool) {pa_mempool_unref(c->rw_mempool);c->rw_mempool = NULL;} }

    setup_srbchannel () 函數還會向客戶端發送 PA_COMMAND_ENABLE_SRBCHANNEL 命令并向共享內存中寫入一段數據。

    (4). 隨后,客戶端收到 pulseaudio 系統服務發過來的共享內存對象文件描述符,attach 到這塊共享內存對象上:

    #0 shm_attach () at ../src/pulsecore/shm.c:347 #1 pa_shm_attach ()at ../src/pulsecore/shm.c:424 #2 segment_attach ()at ../src/pulsecore/memblock.c:1123 #3 pa_memimport_attach_memfd () at ../src/pulsecore/memblock.c:1213 #4 pa_pstream_attach_memfd_shmid () at ../src/pulsecore/pstream.c:377 #5 pa_pstream_register_memfd_mempool ()at ../src/pulsecore/pstream-util.c:174 #6 setup_complete_callback ()at ../src/pulse/context.c:554 #7 run_action () at ../src/pulsecore/pdispatch.c:288 #8 pa_pdispatch_run ()at ../src/pulsecore/pdispatch.c:341 #9 pstream_packet_callback ()at ../src/pulse/context.c:353 #10 do_read () at ../src/pulsecore/pstream.c:1020 #11 do_pstream_read_write () at ../src/pulsecore/pstream.c:260 #12 io_callback () at ../src/pulsecore/pstream.c:312 #13 callback ()at ../src/pulsecore/iochannel.c:158 #14 dispatch_pollfds () at ../src/pulse/mainloop.c:655 #15 pa_mainloop_dispatch () at ../src/pulse/mainloop.c:896 #16 pa_mainloop_iterate () at ../src/pulse/mainloop.c:927 #17 pa_mainloop_run () at ../src/pulse/mainloop.c:942 #18 sync_playback_test () at ../src/tests/sync-playback.c:170 #19 main () at ../src/tests/sync-playback.c:184

    (5). pulseaudio 系統服務創建的共享內存對象的文件描述符被傳給客戶端,客戶端除了 attach 到這塊共享內存對象上之外,還會通過發送 PA_COMMAND_REGISTER_MEMFD_SHMID 命令向 pulseaudio 系統服務注冊它自己創建的共享內存對象的文件描述符:

    #0 pa_pstream_send_packet () at ../src/pulsecore/pstream.c:442 #1 pa_pstream_send_tagstruct_with_ancil_data ()at ../src/pulsecore/pstream-util.c:45 #2 pa_pstream_send_tagstruct_with_fds ()at ../src/pulsecore/pstream-util.c:81 #3 pa_pstream_register_memfd_mempool ()at ../src/pulsecore/pstream-util.c:186 #4 setup_complete_callback ()at ../src/pulse/context.c:554

    (6). pulseaudio 系統服務處理客戶端發過來的注冊共享內存對象文件描述符請求:

    #0 shm_attach () at ../src/pulsecore/shm.c:347 #1 pa_shm_attach ()at ../src/pulsecore/shm.c:424 #2 segment_attach ()at ../src/pulsecore/memblock.c:1123 #3 pa_memimport_attach_memfd () at ../src/pulsecore/memblock.c:1213 #4 pa_pstream_attach_memfd_shmid () at ../src/pulsecore/pstream.c:377 #5 pa_common_command_register_memfd_shmid ()at ../src/pulsecore/native-common.c:67 #6 command_register_memfd_shmid ()at ../src/pulsecore/protocol-native.c:2750 #7 pa_pdispatch_run ()at ../src/pulsecore/pdispatch.c:346

    (7). 客戶端發送 PA_COMMAND_ENABLE_SRBCHANNEL 命令:

    #0 handle_srbchannel_memblock () at ../src/pulse/context.c:359 #1 pstream_memblock_callback () at ../src/pulse/context.c:413 #2 do_read () at ../src/pulsecore/pstream.c:1066 #3 do_pstream_read_write () at ../src/pulsecore/pstream.c:260 #4 io_callback () at ../src/pulsecore/pstream.c:312

    pulseaudio 系統服務處理了客戶端注冊的共享內存對象文件描述符之后,向共享內存中寫入數據,客戶端收到數據之后,向 pulseaudio 系統服務發送 PA_COMMAND_ENABLE_SRBCHANNEL 命令。

    (8). pulseaudio 系統服務處理 PA_COMMAND_ENABLE_SRBCHANNEL 命令

    #0 pa_pstream_set_srbchannel () at ../src/pulsecore/pstream.c:1272 #1 command_enable_srbchannel ()at ../src/pulsecore/protocol-native.c:2559 #2 pa_pdispatch_run ()at ../src/pulsecore/pdispatch.c:346 #3 pstream_packet_callback ()at ../src/pulsecore/protocol-native.c:5027

    PA_COMMAND_ENABLE_SRBCHANNEL 命令處理函數中,將 srbchannel 與 pa_pstream 關聯起來。

    (9). 客戶端發送 PA_COMMAND_CREATE_PLAYBACK_STREAM 命令:

    #0 pa_pstream_send_packet () at ../src/pulsecore/pstream.c:442 #1 pa_pstream_send_tagstruct_with_ancil_data ()at ../src/pulsecore/pstream-util.c:45 #2 pa_pstream_send_tagstruct_with_creds () at ../src/pulsecore/pstream-util.c:61 #3 create_stream () at ../src/pulse/stream.c:1382 #4 pa_stream_connect_playback () at ../src/pulse/stream.c:1402 #5 context_state_callback () at ../src/tests/sync-playback.c:129

    客戶端創建 pa_stream 對象,并在調用 pa_stream_connect_playback () 接口時向pulseaudio 系統服務發送 PA_COMMAND_CREATE_PLAYBACK_STREAM 命令,請求 pulseaudio 系統服務創建播放流。

    (10). pulseaudio 系統服務處理 PA_COMMAND_CREATE_PLAYBACK_STREAM 命令

    #0 playback_stream_new () at ../src/pulsecore/protocol-native.c:964 #1 command_create_playback_stream ()at ../src/pulsecore/protocol-native.c:2067 #2 pa_pdispatch_run )at ../src/pulsecore/pdispatch.c:346

    pulseaudio 系統服務處理 PA_COMMAND_CREATE_PLAYBACK_STREAM 命令,創建 playback_stream 對象,這里會給流的 sink_input 設置許多回調,如:

    s->sink_input->parent.process_msg = sink_input_process_msg;s->sink_input->pop = sink_input_pop_cb;s->sink_input->process_underrun = sink_input_process_underrun_cb;s->sink_input->process_rewind = sink_input_process_rewind_cb;s->sink_input->update_max_rewind = sink_input_update_max_rewind_cb;s->sink_input->update_max_request = sink_input_update_max_request_cb;s->sink_input->kill = sink_input_kill_cb;s->sink_input->moving = sink_input_moving_cb;s->sink_input->suspend = sink_input_suspend_cb;s->sink_input->send_event = sink_input_send_event_cb;s->sink_input->userdata = s;start_index = ssync ? pa_memblockq_get_read_index(ssync->memblockq) : 0;fix_playback_buffer_attr(s);pa_sink_input_get_silence(sink_input, &silence);memblockq_name = pa_sprintf_malloc("native protocol playback stream memblockq [%u]", s->sink_input->index);s->memblockq = pa_memblockq_new(memblockq_name,start_index,s->buffer_attr.maxlength,s->buffer_attr.tlength,&sink_input->sample_spec,s->buffer_attr.prebuf,s->buffer_attr.minreq,0,&silence);pa_xfree(memblockq_name);

    (11). 客戶端發送 PA_COMMAND_CORK_PLAYBACK_STREAM 命令:

    #0 pa_pstream_send_packet () at ../src/pulsecore/pstream.c:442 #1 pa_pstream_send_tagstruct_with_ancil_data ()at ../src/pulsecore/pstream-util.c:45 #2 pa_pstream_send_tagstruct_with_creds () at ../src/pulsecore/pstream-util.c:61 #3 pa_stream_cork () at ../src/pulse/stream.c:2293 #4 stream_state_callback () at ../src/tests/sync-playback.c:95

    客戶端調用 pa_stream_cork () 接口向 pulseaudio 系統服務發送 PA_COMMAND_CORK_PLAYBACK_STREAM 命令。這個接口用于暫停或恢復播放或錄制流。

    (12). pulseaudio 系統服務處理 PA_COMMAND_CORK_PLAYBACK_STREAM 命令

    #0 pa_sink_input_cork () at ../src/pulsecore/sink-input.c:1577 #1 command_cork_playback_stream ()at ../src/pulsecore/protocol-native.c:3988 #2 pa_pdispatch_run ()at ../src/pulsecore/pdispatch.c:346 #3 pstream_packet_callback ()at ../src/pulsecore/protocol-native.c:5027

    pulseaudio 系統服務設置 sink_input 的狀態。

    此外,客戶端還向 pulseaudio 系統服務發送了 command_set_client_name 命令和多個 command_get_info 命令 ,這里不再詳述這些命令的發送和接收處理過程。

    客戶端和 pulseaudio 系統服務經過上面的這些命令交互,完成整個的協商過程,它們之間可以開始進行音頻數據的互相發送了。

  • 客戶端發送數據
  • 客戶端通過調用 pa_stream_write () 向播放流中寫入數據,將播放數據發送給 pulseaudio 系統服務:

    #0 pa_pstream_send_memblock () at ../src/pulsecore/pstream.c:477 #1 pa_stream_write_ext_free () at ../src/pulse/stream.c:1549 #2 pa_stream_write () at ../src/pulse/stream.c:1616 #3 stream_state_callback () at ../src/tests/sync-playback.c:87

    pa_stream_write_ext_free () 將要發送的數據切成一個個的塊,然后通過 pa_pstream_send_memblock () 發送出去:

    while (t_length > 0) {pa_memchunk chunk;chunk.index = 0;if (free_cb && !pa_pstream_get_shm(s->context->pstream)) {chunk.memblock = pa_memblock_new_user(s->context->mempool, (void*) t_data, t_length, free_cb, free_cb_data, 1);chunk.length = t_length;} else {void *d;size_t blk_size_max;/* Break large audio streams into _aligned_ blocks or the* other endpoint will happily discard them upon arrival. */blk_size_max = pa_frame_align(pa_mempool_block_size_max(s->context->mempool), &s->sample_spec);chunk.length = PA_MIN(t_length, blk_size_max);chunk.memblock = pa_memblock_new(s->context->mempool, chunk.length);d = pa_memblock_acquire(chunk.memblock);memcpy(d, t_data, chunk.length);pa_memblock_release(chunk.memblock);}pa_pstream_send_memblock(s->context->pstream, s->channel, t_offset, t_seek, &chunk);t_offset = 0;t_seek = PA_SEEK_RELATIVE;t_data = (const uint8_t*) t_data + chunk.length;t_length -= chunk.length;pa_memblock_unref(chunk.memblock);}

    pa_pstream_send_memblock () 發送數據是異步的,它將一個個塊的數據再按需切成一個個 item,放進發送隊列里:

    void pa_pstream_send_memblock(pa_pstream*p, uint32_t channel, int64_t offset, pa_seek_mode_t seek_mode, const pa_memchunk *chunk) {size_t length, idx;size_t bsm;pa_assert(p);pa_assert(PA_REFCNT_VALUE(p) > 0);pa_assert(channel != (uint32_t) -1);pa_assert(chunk);if (p->dead)return;idx = 0;length = chunk->length;bsm = pa_mempool_block_size_max(p->mempool);while (length > 0) {struct item_info *i;size_t n;if (!(i = pa_flist_pop(PA_STATIC_FLIST_GET(items))))i = pa_xnew(struct item_info, 1);i->type = PA_PSTREAM_ITEM_MEMBLOCK;n = PA_MIN(length, bsm);i->chunk.index = chunk->index + idx;i->chunk.length = n;i->chunk.memblock = pa_memblock_ref(chunk->memblock);i->channel = channel;i->offset = offset;i->seek_mode = seek_mode; #ifdef HAVE_CREDSi->with_ancil_data = false; #endifpa_queue_push(p->send_queue, i);idx += n;length -= n;}p->mainloop->defer_enable(p->defer_event, 1); }

    在主事件循環中,通過 srbchannel 將數據發送出去:

    #0 pa_memexport_put () at ../src/pulsecore/memblock.c:1455 #1 prepare_next_write_item () at ../src/pulsecore/pstream.c:664 #2 do_write () at ../src/pulsecore/pstream.c:751 #3 do_pstream_read_write () at ../src/pulsecore/pstream.c:266 #4 srb_callback () at ../src/pulsecore/pstream.c:295 #5 srbchannel_rwloop () at ../src/pulsecore/srbchannel.c:190 #6 semread_cb ()at ../src/pulsecore/srbchannel.c:210 #7 dispatch_pollfds () at ../src/pulse/mainloop.c:655

    pa_pstream_send_memblock () 函數是如何觸發 srbchannel 的回調執行的呢?這是由于客戶端通過如下過程創建 pa_srbchannel:

    #0 pa_srbchannel_new_from_template () at ../src/pulsecore/srbchannel.c:291 #1 handle_srbchannel_memblock () at ../src/pulse/context.c:380 #2 pstream_memblock_callback () at ../src/pulse/context.c:413 #3 do_read () at ../src/pulsecore/pstream.c:1066 #4 do_pstream_read_write () at ../src/pulsecore/pstream.c:260 #5 io_callback () at ../src/pulsecore/pstream.c:312 #6 callback ()at ../src/pulsecore/iochannel.c:158 #7 dispatch_pollfds () at ../src/pulse/mainloop.c:655

    且會為 srb_channel 設置 defer_event 回調:

    #0 pa_srbchannel_set_callback () at ../src/pulsecore/srbchannel.c:340 #1 check_srbpending () at ../src/pulsecore/pstream.c:738 #2 do_write () at ../src/pulsecore/pstream.c:755 #3 do_pstream_read_write () at ../src/pulsecore/pstream.c:266 #4 0x00007ffff7bcd19e in io_callback () at ../src/pulsecore/pstream.c:312

    pa_srbchannel_set_callback () 函數實現如下:

    void pa_srbchannel_set_callback(pa_srbchannel *sr, pa_srbchannel_cb_t callback, void *userdata) {if (sr->callback)pa_fdsem_after_poll(sr->sem_read);sr->callback = callback;sr->cb_userdata = userdata;if (sr->callback) {/* If there are events to be read already in the ringbuffer, we will not get any IO event for that,because that's how pa_fdsem works. Therefore check the ringbuffer in a defer event instead. */if (!sr->defer_event)sr->defer_event = sr->mainloop->defer_new(sr->mainloop, defer_cb, sr);sr->mainloop->defer_enable(sr->defer_event, 1);} }

    pa_srbchannel_new_from_template () 函數的實現如下:

    pa_srbchannel* pa_srbchannel_new_from_template(pa_mainloop_api *m, pa_srbchannel_template *t) {int temp;struct srbheader *srh;pa_srbchannel* sr = pa_xmalloc0(sizeof(pa_srbchannel));sr->mainloop = m;sr->memblock = t->memblock;pa_memblock_ref(sr->memblock);srh = pa_memblock_acquire(sr->memblock);sr->rb_read.capacity = sr->rb_write.capacity = srh->capacity;sr->rb_read.count = &srh->read_count;sr->rb_write.count = &srh->write_count;sr->rb_read.memory = (uint8_t*) srh + srh->readbuf_offset;sr->rb_write.memory = (uint8_t*) srh + srh->writebuf_offset;sr->sem_read = pa_fdsem_open_shm(&srh->read_semdata, t->readfd);if (!sr->sem_read)goto fail;sr->sem_write = pa_fdsem_open_shm(&srh->write_semdata, t->writefd);if (!sr->sem_write)goto fail;pa_srbchannel_swap(sr);temp = t->readfd; t->readfd = t->writefd; t->writefd = temp;#ifdef DEBUG_SRBCHANNELpa_log("Enabling io event on fd %d", t->readfd); #endifsr->read_event = m->io_new(m, t->readfd, PA_IO_EVENT_INPUT, semread_cb, sr);m->io_enable(sr->read_event, PA_IO_EVENT_INPUT);return sr;fail:pa_srbchannel_free(sr);return NULL; }

    這里會給 io event (read_event)關聯回調。

    在 pa_pstream_send_memblock () 函數中通過 p->mainloop->defer_enable(p->defer_event, 1); 喚醒主循環,這個回調的實際實現函數為 mainloop_defer_enable():

    static void mainloop_defer_enable(pa_defer_event *e, int b) {pa_assert(e);pa_assert(!e->dead);if (e->enabled && !b) {pa_assert(e->mainloop->n_enabled_defer_events > 0);e->mainloop->n_enabled_defer_events--;} else if (!e->enabled && b) {e->mainloop->n_enabled_defer_events++;pa_mainloop_wakeup(e->mainloop);}e->enabled = b; }

    mainloop_defer_enable() 函數更新狀態 n_enabled_defer_events,并喚醒主事件循環。被喚醒的主事件循環中,pa_mainloop_dispatch() 在看到這個狀態時,會將事件派發給各個 defer event:

    static unsigned dispatch_defer(pa_mainloop *m) {pa_defer_event *e;unsigned r = 0;if (m->n_enabled_defer_events <= 0)return 0;PA_LLIST_FOREACH(e, m->defer_events) {if (m->quit)break;if (e->dead || !e->enabled)continue;pa_assert(e->callback);e->callback(&m->api, e, e->userdata);r++;}return r; } . . . . . . int pa_mainloop_dispatch(pa_mainloop *m) {unsigned dispatched = 0;pa_assert(m);pa_assert(m->state == STATE_POLLED);if (m->quit)goto quit;if (m->n_enabled_defer_events)dispatched += dispatch_defer(m);else {

    mainloop_defer_enable() 在出發 defer event 被回調之外,也會觸發 srb_channel 的 read event 被調用。

  • 服務端接收數據
  • 在 Linux 平臺上,pulseaudio 系統服務通過 ALSA 與系統音頻硬件交互。ALSA 項目的主頁為 ALSA。ALSA 項目有一份文檔 A Tutorial on Using the ALSA Audio API 簡單說明了 ALSA 庫接口的用法。ALSA 庫的接口還可以參考 ALSA Library API 和 ALSA library API reference,特別是 PCM interface 部分。

    有兩種方法可以用來傳輸音頻樣本數據,第一種是標準的讀寫接口。第二種是使用直接音頻緩沖區,來與設備通信。標準的讀寫接口包括 snd_pcm_writei()?/ snd_pcm_readi() 和 snd_pcm_writen()?/?snd_pcm_readn()。直接讀寫傳輸借助于 mmap 的區域來傳輸數據,這些接口包括 snd_pcm_mmap_begin() / snd_pcm_mmap_commit()。pulse audio 中用這兩種方式都支持,一般用的是直接讀寫傳輸。

    alsa-sink 內部有一個線程,在需要的時候,會去取播放的數據并寫入設備(pulseaudio/src/modules/alsa/alsa-sink.c):

    static int mmap_write(struct userdata *u, pa_usec_t *sleep_usec, bool polled, bool on_timeout) {bool work_done = false; . . . . . .for (;;) {pa_memchunk chunk;void *p;int err;const snd_pcm_channel_area_t *areas;snd_pcm_uframes_t offset, frames;snd_pcm_sframes_t sframes;size_t written;frames = (snd_pcm_uframes_t) (n_bytes / u->frame_size); /* pa_log_debug("%lu frames to write", (unsigned long) frames); */if (PA_UNLIKELY((err = pa_alsa_safe_mmap_begin(u->pcm_handle, &areas, &offset, &frames, u->hwbuf_size, &u->sink->sample_spec)) < 0)) {if (!after_avail && err == -EAGAIN)break;if ((r = try_recover(u, "snd_pcm_mmap_begin", err)) == 0)continue;if (r == 1)break;return r;}/* Make sure that if these memblocks need to be copied they will fit into one slot */frames = PA_MIN(frames, u->frames_per_block);if (!after_avail && frames == 0)break;pa_assert(frames > 0);after_avail = false;/* Check these are multiples of 8 bit */pa_assert((areas[0].first & 7) == 0);pa_assert((areas[0].step & 7) == 0);/* We assume a single interleaved memory buffer */pa_assert((areas[0].first >> 3) == 0);pa_assert((areas[0].step >> 3) == u->frame_size);p = (uint8_t*) areas[0].addr + (offset * u->frame_size);written = frames * u->frame_size;chunk.memblock = pa_memblock_new_fixed(u->core->mempool, p, written, true);chunk.length = pa_memblock_get_length(chunk.memblock);chunk.index = 0;pa_sink_render_into_full(u->sink, &chunk);pa_memblock_unref_fixed(chunk.memblock);if (PA_UNLIKELY((sframes = snd_pcm_mmap_commit(u->pcm_handle, offset, frames)) < 0)) {if ((int) sframes == -EAGAIN)break;if ((r = try_recover(u, "snd_pcm_mmap_commit", (int) sframes)) == 0)continue;if (r == 1)break;return r;}

    如上所示,alsa-sink 模塊通過 snd_pcm_mmap_commit() 接口將接收的音頻數據送給音頻設備。

    alsa-sink 線程取數據的過程如下:

    #0 sink_input_pop_cb () at ../src/pulsecore/protocol-native.c:1504 #1 pa_sink_input_peek ()at ../src/pulsecore/sink-input.c:931 #2 fill_mix_info () at ../src/pulsecore/sink.c:1100 #3 pa_sink_render_into () at ../src/pulsecore/sink.c:1340 #4 pa_sink_render_into_full () at ../src/pulsecore/sink.c:1424 #5 mmap_write () at ../src/modules/alsa/alsa-sink.c:737 #6 thread_func () at ../src/modules/alsa/alsa-sink.c:1921 #7 internal_thread_func () at ../src/pulsecore/thread-posix.c:81 #8 start_thread ( at pthread_create.c:477

    pulseaudio 的 ALSA 模塊在 sink_input_pop_cb() 函數中請求音頻數據:

    /* Called from thread context */ static int sink_input_pop_cb(pa_sink_input *i, size_t nbytes, pa_memchunk *chunk) {playback_stream *s;pa_sink_input_assert_ref(i);s = PLAYBACK_STREAM(i->userdata);playback_stream_assert_ref(s);pa_assert(chunk);#ifdef PROTOCOL_NATIVE_DEBUGpa_log("%s, pop(): %lu", pa_proplist_gets(i->proplist, PA_PROP_MEDIA_NAME), (unsigned long) pa_memblockq_get_length(s->memblockq)); #endifif (!handle_input_underrun(s, false))s->is_underrun = false;/* This call will not fail with prebuf=0, hence we check forunderrun explicitly in handle_input_underrun */if (pa_memblockq_peek(s->memblockq, chunk) < 0)return -1;chunk->length = PA_MIN(nbytes, chunk->length);if (i->thread_info.underrun_for > 0)pa_asyncmsgq_post(pa_thread_mq_get()->outq, PA_MSGOBJECT(s), PLAYBACK_STREAM_MESSAGE_STARTED, NULL, 0, NULL, NULL);pa_memblockq_drop(s->memblockq, chunk->length);playback_stream_request_bytes(s);return 0; }

    sink_input_pop_cb() 函數先處理 underrun ,即緩沖區中數據不足的情況:

    #0 playback_stream_request_bytes () at ../src/pulsecore/protocol-native.c:1116 #1 handle_input_underrun () at ../src/pulsecore/protocol-native.c:1488 #2 sink_input_pop_cb () at ../src/pulsecore/protocol-native.c:1516 #3 pa_sink_input_peek ()at ../src/pulsecore/sink-input.c:931

    此時會通過 playback_stream_request_bytes () 給客戶端發消息讓它發數據過來。隨后,從內存塊緩存隊列中取一部分數據出來:

    #0 pa_memblockq_peek () at ../src/pulsecore/memblockq.c:474 #1 sink_input_pop_cb () at ../src/pulsecore/protocol-native.c:1521 #2 pa_sink_input_peek ()at ../src/pulsecore/sink-input.c:931

    最后sink_input_pop_cb() 函數還是會通過 playback_stream_request_bytes () 給客戶端發消息請求數據。playback_stream_request_bytes () 函數定義如下:

    static void playback_stream_request_bytes(playback_stream *s) {size_t m;playback_stream_assert_ref(s);m = pa_memblockq_pop_missing(s->memblockq);if (m <= 0)return;#ifdef PROTOCOL_NATIVE_DEBUGpa_log("request_bytes(%lu)", (unsigned long) m); #endifif (pa_atomic_add(&s->missing, (int) m) <= 0)pa_asyncmsgq_post(pa_thread_mq_get()->outq, PA_MSGOBJECT(s), PLAYBACK_STREAM_MESSAGE_REQUEST_DATA, NULL, 0, NULL, NULL); }

    這最終會導致 pulseaudio 服務向客戶端發送一個請求數據的命令(pulseaudio/src/pulsecore/protocol-native.c)

    static int playback_stream_process_msg(pa_msgobject *o, int code, void*userdata, int64_t offset, pa_memchunk *chunk) {playback_stream *s = PLAYBACK_STREAM(o);playback_stream_assert_ref(s);if (!s->connection)return -1;switch (code) {case PLAYBACK_STREAM_MESSAGE_REQUEST_DATA: {pa_tagstruct *t;int l = 0;for (;;) {if ((l = pa_atomic_load(&s->missing)) <= 0)return 0;if (pa_atomic_cmpxchg(&s->missing, l, 0))break;}t = pa_tagstruct_new();pa_tagstruct_putu32(t, PA_COMMAND_REQUEST);pa_tagstruct_putu32(t, (uint32_t) -1); /* tag */pa_tagstruct_putu32(t, s->index);pa_tagstruct_putu32(t, (uint32_t) l);pa_pstream_send_tagstruct(s->connection->pstream, t);

    播放數據這塊的處理是典型的生產者-消費者模型。前面這里看到的都是消費者的處理,不過那生產者是怎么把數據放進隊列里的呢?

    sink_input_process_msg() 函數在處理 SINK_INPUT_MESSAGE_SEEK 消息和 SINK_INPUT_MESSAGE_POST_DATA 消息時將讀取的數據放進隊列里,從 backtrace 可以看到,放數據的動作是在 IO 線程,也就是 alsa-sink 線程,里完成的:

    #0 sink_input_process_msg () at ../src/pulsecore/protocol-native.c:1318 #1 pa_asyncmsgq_dispatch ()at ../src/pulsecore/asyncmsgq.c:323 #2 asyncmsgq_read_work () at ../src/pulsecore/rtpoll.c:566 #3 pa_rtpoll_run () at ../src/pulsecore/rtpoll.c:238 #4 thread_func () at ../src/modules/alsa/alsa-sink.c:2003

    SINK_INPUT_MESSAGE_POST_DATA 和 SINK_INPUT_MESSAGE_POST_DATA 消息,讀取到客戶端發送過來的數據時發送,pstream.c 下的 do_read() 通過 pa_memimport_get () 獲得客戶端發送過來的內存塊:

    #0 pa_memimport_get () at ../src/pulsecore/memblock.c:1231 #1 do_read () at ../src/pulsecore/pstream.c:1042 #2 do_pstream_read_write () at ../src/pulsecore/pstream.c:253 #3 srb_callback () at ../src/pulsecore/pstream.c:295 #4 srbchannel_rwloop () at ../src/pulsecore/srbchannel.c:190 #5 semread_cb ()at ../src/pulsecore/srbchannel.c:210

    在 pstream_memblock_callback() 函數中,發送 SINK_INPUT_MESSAGE_POST_DATA 和 SINK_INPUT_MESSAGE_POST_DATA 消息出去:

    #0 pstream_memblock_callback () at ../src/pulsecore/protocol-native.c:5033 #1 do_read () at ../src/pulsecore/pstream.c:1066 #2 do_pstream_read_write () at ../src/pulsecore/pstream.c:253

    從 PulseAudio Git repo master 分支的 commit 歷史可以看到,第一筆 commit 是在 2004 年提交的,從第一筆 commit 提交到現在已經有近 20 年了,PulseAudio 是一個經過了非常多年發展,非常有歷史的一個項目,想必 PulseAudio 的很多代碼也是經過了相當長的歷史演化變成今天這個樣子的。一天兩天似乎是很難將這個項目完全吃透的。這里對于 PulseAudio 的說明還很粗淺。

    PulseAudio 項目不僅僅是要做一個應用和音頻設備之間的數據通道,它更想成為一個音頻框架。這里的介紹只是了解 PulseAudio 設計和實現的一個角度。要想對 PulseAudio 有更深入的了解,對一些技術基礎的了解不可或缺:

    • 進程間通信的庫接口的詳細用法,這包括 D-Bus,Unix 域 socket API,IPv4/IPv6 tcp socket 的 API,共享內存接口如 memfd_create() / shm_open() 等。
    • udev 庫接口的詳細用法及其相關機制
    • ALSA 庫接口的詳細用法
    • timerfd 和 eventfd 的用法

    還有很多 PulseAudio 相關的內容這里還沒有涉及:

    • PulseAudio 的線程模型及線程基礎設施,如 pa_mainloop、pa_mainloop_api 和 pa_threaded_mainloop 等。
    • PulseAudio 創建的一些概念和抽象的語義,如 pa_context,pa_stream,pa_iochannel,pa_pstream,pa_srbchannel,pa_sink,pa_sink_input,pa_source,pa_source_output,pa_module 和 pa_card 等等等
    • pulseaudio 系統服務管理音頻設備文件
    • pulseaudio 系統服務的模塊化架構設計
    • 音頻 pipeline 的搭建
    • 客戶端和 pulseaudio 系統服務間消息和數據交換的詳細設計
    • 等等等。

    參考文檔:
    如何暫時禁用PulseAudio?
    Instructions for building and installing the current development version
    README.md of pulseaudio

    總結

    以上是生活随笔為你收集整理的PulseAudio 设计和实现浅析的全部內容,希望文章能夠幫你解決所遇到的問題。

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

    久久全国免费视频 | 黄色大全免费网站 | 欧美作爱视频 | 丝袜av网站| 免费看的国产视频网站 | 夜夜夜影院 | 9热精品 | 三级午夜片 | 91丨九色丨国产在线观看 | 午夜婷婷在线观看 | 亚洲一级免费电影 | 99综合电影在线视频 | 久久91网 | wwwwww色| 免费av网址在线观看 | 五月天久久 | 国产成人精品在线 | 亚洲精品在线免费 | 狠狠网亚洲精品 | 亚洲综合精品在线 | 免费在线播放视频 | 国产精品一区二区在线看 | 国产综合91| 99国产视频在线 | 日日干美女 | 黄色av一区二区 | 日韩av美女 | 国产一级黄色电影 | 国产亚洲成av人片在线观看桃 | av成人免费在线观看 | 国产99在线播放 | 最近最新mv字幕免费观看 | 亚洲黄色在线观看 | 国产精品久久久久久久久久久久久 | 69国产成人综合久久精品欧美 | 国产成人精品av久久 | 亚洲aⅴ在线 | 在线观看深夜福利 | 黄色a视频| 亚洲激情视频在线观看 | av超碰在线观看 | www.com.黄 | 亚洲精品国产综合99久久夜夜嗨 | 国产一区在线视频 | 日韩一区二区三区高清在线观看 | 久久综合色8888 | 日韩欧美在线观看一区二区三区 | 免费看毛片在线 | 开心色婷婷 | 久久 精品一区 | 97在线免费 | 亚洲天天做 | 久草久草在线观看 | 亚洲一级免费观看 | 91九色网站 | 久久久毛片 | 在线观看成人小视频 | 久久精品国产一区二区三区 | 免费看三级 | 国产69精品久久久久99尤 | 一级做a爱片性色毛片www | 久久99精品久久久久婷婷 | 国产精品久久久久久五月尺 | 亚洲欧美一区二区三区孕妇写真 | 五月天综合色激情 | 日韩欧美精品在线观看 | 亚洲精品国产欧美在线观看 | 美女视频一区 | 国产亚洲aⅴaaaaaa毛片 | 免费黄色av片 | 亚洲精品在线观看av | av在线成人 | 日韩精品免费专区 | 狠狠色丁香久久婷婷综合_中 | 91在线精品秘密一区二区 | 免费在线观看日韩欧美 | 香蕉网在线观看 | 免费黄色激情视频 | 热re99久久精品国产66热 | 麻豆小视频在线观看 | 91日韩在线| 国产人成一区二区三区影院 | 久久av免费电影 | 精品国产三级a∨在线欧美 免费一级片在线观看 | 亚洲高清视频在线 | 日韩久久午夜一级啪啪 | 久久新| 三级av在线播放 | 伊人婷婷网 | 日韩精品视频一二三 | 天天干天天看 | 国产精品破处视频 | 人人爽人人干 | 日韩在线视 | 欧美做受xxx | 国产成人精品一区一区一区 | 日韩精品中文字幕久久臀 | 超碰在线人人艹 | 丁香婷婷综合激情五月色 | 亚洲视屏 | 中文字幕精品视频 | 亚洲精品资源在线观看 | 国产精品理论在线观看 | 97在线视频免费 | 国产免费区 | 国产亚洲精品久久久久秋 | 国产成人免费网站 | 在线视频 国产 日韩 | 天天干天天操av | 91麻豆高清视频 | 91在线公开视频 | 99riav1国产精品视频 | av成人免费在线看 | 午夜久久美女 | 日本中文字幕网 | 91久久精品一区二区三区 | 在线精品视频在线观看高清 | 亚洲欧美日韩国产 | 人人干人人做 | 成人va天堂 | 久久超碰网 | 在线观看免费视频 | 91在线免费播放 | 日韩三区在线观看 | 中文字幕资源在线 | 免费在线观看av的网站 | 国产亚洲欧美精品久久久久久 | 精品一区三区 | 成人97人人超碰人人99 | 天天爽天天摸 | 黄网站色视频免费观看 | 婷婷激情网站 | 亚洲午夜剧场 | 国产精品久一 | 国产精品a级 | 国产成人av免费在线观看 | 欧美精品v国产精品 | 成人午夜电影在线 | 日本精品一区二区 | 国产青草视频在线观看 | 亚洲欧洲成人精品av97 | 国产69精品久久99不卡的观看体验 | 77国产精品 | 国产视频每日更新 | 在线免费中文字幕 | 黄色1级大片 | 久久人人97超碰精品888 | 久爱综合 | 国产中文字幕在线 | 成人免费网站视频 | 草久久久久| 波多野结衣久久资源 | 日韩在线免费看 | 久久久久久黄色 | 色丁香色婷婷 | 国产日韩欧美在线一区 | 热99在线| av丝袜在线 | 免费看三级 | 久久久久久久久久久精 | 黄色激情网址 | 在线观看成人 | 蜜臀av性久久久久蜜臀aⅴ四虎 | 色综合久久中文字幕综合网 | 国产永久免费观看 | 亚洲国产欧洲综合997久久, | 精品国产免费观看 | 国产精品女同一区二区三区久久夜 | 欧美日韩99 | 亚洲精区二区三区四区麻豆 | 96国产在线 | 中文字幕av免费在线观看 | 成人午夜免费剧场 | 国产黄色片一级 | 欧美视频在线观看免费网址 | 亚洲电影第一页av | 奇米影视999| 色婷婷综合在线 | 欧美资源在线观看 | 国产精品久久久久影视 | 成人h在线观看 | 日韩免费看 | 99热在线免费观看 | 午夜私人影院 | 精品国产精品国产偷麻豆 | 色综合久久精品 | 国产精品久久久久影院 | 国产精品欧美久久久久天天影视 | 激情片av| 久久综合九色九九 | 免费在线观看a v | 久久视频在线观看免费 | av电影免费在线看 | 九九综合在线 | 中文字幕一区二区三区四区在线视频 | 久久99国产精品免费 | 色99中文字幕 | 国产精品美女久久久免费 | 欧美另类xxxxx | 久久成人国产 | 日韩精品视频在线免费观看 | 在线观看国产区 | 精品在线一区二区 | 日韩日韩日韩日韩 | 91av在线国产 | 91自拍视频在线观看 | 国产精品自产拍在线观看蜜 | 激情网在线视频 | www.超碰| 国产精品国产毛片 | 热久久这里只有精品 | 中字幕视频在线永久在线观看免费 | 久久av不卡 | 爱情影院aqdy鲁丝片二区 | 丝袜网站在线观看 | 亚洲精品综合在线 | 免费日韩 精品中文字幕视频在线 | 欧美一级黄大片 | 亚洲在线成人精品 | 亚洲精品一区中文字幕乱码 | 欧美 亚洲 另类 激情 另类 | 丁香激情综合国产 | 日本精品一区二区 | 狠狠操天天射 | 99热精品久久 | 色黄久久久久久 | 成人国产精品久久久久久亚洲 | 成人精品一区二区三区中文字幕 | 日韩91在线| 一区二区电影在线观看 | 亚洲国产精品第一区二区 | 天天干天天爽 | 午夜体验区 | 国产精品18久久久久久久久 | 国产成人精品午夜在线播放 | 日韩女同av | 国产乱码精品一区二区三区介绍 | 亚洲欧美激情精品一区二区 | 天天曰天天曰 | 色.www| 日本aaaa级毛片在线看 | 欧美日韩综合在线观看 | 日日夜夜草| 免费在线观看污网站 | 最近中文字幕免费观看 | 日韩在线观看a | 精品在线观看一区二区三区 | 久久久精品高清 | av免费看在线 | 久久人91精品久久久久久不卡 | 91完整版在线观看 | 国产一区二区三区 在线 | 91在线91拍拍在线91 | 国产免费成人 | www在线观看视频 | 国产日产精品久久久久快鸭 | 在线观看免费成人 | 最新日韩精品 | 激情视频一区二区三区 | 男女免费视频观看 | 亚州免费视频 | 久久伦理视频 | 国产日本三级 | 日本99精品 | 黄在线免费观看 | 日韩丝袜在线观看 | 美女露久久 | 97超碰精品 | 韩日精品中文字幕 | 国产精品a级| 区一区二区三在线观看 | 亚洲精选久久 | 久久伊人国产精品 | 欧美极品久久 | 欧美精品久久久久久久久免 | 成年人app网址 | 欧洲精品码一区二区三区免费看 | 国产亚洲无 | 亚洲国产精品久久久久 | 国产成人精品久久二区二区 | 91高清一区| 久久精品视频免费 | 亚洲在线色 | 国产日韩视频在线观看 | 一级黄色片在线免费看 | 就色干综合 | 亚洲爽爽网 | 久久精品8 | 欧美aaa大片 | 91九色porny蝌蚪视频 | 在线观看免费av片 | 在线观看中文字幕一区二区 | 午夜黄色影院 | 天天综合导航 | 婷婷色六月天 | 婷婷中文字幕 | 四虎在线观看精品视频 | 超碰人人乐 | 在线中文字幕网站 | 婷婷深爱五月 | 国产色久 | 久久精品一区二区国产 | 亚洲午夜久久久久久久久 | 九九热精品视频在线观看 | 日本女人在线观看 | 久草在线视频首页 | 高清免费av在线 | 狠狠操操操 | 99久久综合狠狠综合久久 | 国内精品久久久久久久 | 91久久在线观看 | 欧美精品乱码久久久久久按摩 | 97色国产| 欧美一进一出抽搐大尺度视频 | 国精产品一二三线999 | 精品国产精品久久一区免费式 | 国产精品片 | 91天堂素人约啪 | 久草在线观 | 免费黄色在线播放 | 国产91精品一区二区 | 日韩精品资源 | 欧美日韩国语 | 伊人天天综合 | 日韩电影在线观看一区 | av在线进入 | 国产精品久久久久av福利动漫 | 国产精品久久久久久久久久久久久久 | 欧美日韩精| 国产一区二区久久 | 天天射天 | 久草亚洲视频 | www.99av| 亚洲精品在线观看视频 | 成人av在线资源 | 免费在线一区二区 | 国产精品黄| www免费在线观看 | 99精品欧美一区二区蜜桃免费 | a级国产乱理伦片在线观看 亚洲3级 | 日韩高清不卡在线 | 成人免费看片网址 | 色网站在线免费观看 | 国产久草在线观看 | 日韩三级不卡 | 日韩在线精品视频 | 日日躁夜夜躁xxxxaaaa | 黄色av观看| av网站大全免费 | 亚洲精品综合久久 | 久久黄色免费视频 | 国产精品岛国久久久久久久久红粉 | 91亚洲狠狠婷婷综合久久久 | 国内视频在线观看 | 91探花在线| 日日夜av| 成人欧美一区二区三区在线观看 | 激情久久综合 | 丁香视频全集免费观看 | 婷婷激情网站 | 午夜电影av| 日日操天天操夜夜操 | 日韩久久久久久久久 | 成人av电影网址 | 九九九热精品免费视频观看 | 久久91久久久久麻豆精品 | 中文字幕免费高清在线观看 | 狠狠天天 | 中文字幕高清 | 久久亚洲人 | 亚洲高清国产视频 | 在线电影中文字幕 | 亚洲欧洲在线视频 | 国产 亚洲 欧美 在线 | 久久人人爽av | 天天色综合久久 | 国产精品 中文字幕 亚洲 欧美 | 日韩精品1区2区 | 精品字幕| 国产一区欧美一区 | 99久久精品久久久久久清纯 | 国产亚洲在线观看 | 麻豆视频国产在线观看 | 国产二区视频在线 | 日韩欧美综合在线视频 | 国产精品毛片久久久久久久久久99999999 | 成人欧美一区二区三区在线观看 | 青春草国产视频 | 91视频91自拍 | 91大神精品视频在线观看 | 免费看国产一级片 | 日韩亚洲国产中文字幕 | 一区 二区电影免费在线观看 | 在线观看免费观看在线91 | 国产尤物视频在线 | 成年人天堂com | 91免费在线播放 | 国产不卡在线看 | 日本一区二区三区免费观看 | 伊在线视频| 九九精品毛片 | 国产精品自拍av | 久久久私人影院 | 亚洲黄色在线免费观看 | 97福利在线观看 | 激情动态 | 亚洲国产精品视频在线观看 | 99久久精品免费看国产四区 | 丝袜网站在线观看 | 免费能看的黄色片 | 天天曰天天爽 | 中文字幕免费看 | 久久精品电影院 | av黄色av| 有码中文字幕在线观看 | 麻豆视传媒官网免费观看 | 日韩电影久久久 | 免费在线观看av片 | 国产福利a| 午夜精品剧场 | 九九导航| 国产高清视频在线免费观看 | 午夜久久久久 | 国产精品视频免费看 | 69精品在线观看 | 91色蜜桃| 日韩精品在线视频免费观看 | 91精品黄色 | 91精品在线观看视频 | 中文字幕激情 | 免费久久99精品国产 | 日韩精品一区二区三区免费观看视频 | 日韩精品一区二区三区免费视频观看 | 人人舔人人 | 91网址在线看 | 丁香六月色| 狠狠久久综合 | av片在线观看免费 | 美女免费电影 | 国产高清99 | 2019av在线视频 | 日韩毛片在线一区二区毛片 | 国产成人精品一区二三区 | 五月丁色 | 婷婷久久国产 | 亚洲三级网站 | 日韩中文字幕视频在线 | 久久99精品国产99久久6尤 | av片在线看| 97超碰人人澡人人爱 | 激情一区二区三区欧美 | 日本高清中文字幕有码在线 | 91亚洲精品久久久蜜桃网站 | 亚洲一级二级三级 | 伊人夜夜 | 国产99久久九九精品免费 | 免费黄色看片 | 免费一级日韩欧美性大片 | 波多野结衣久久资源 | 精品久久久久久综合日本 | 国产精品福利av | 日本中文字幕在线一区 | 成年人免费观看在线视频 | 日日夜夜精品视频 | 狠狠的干| 国产色黄网站 | 国产最顶级的黄色片在线免费观看 | 香蕉视频在线播放 | 日日爽日日操 | 天堂网一区| 欧美成人h版在线观看 | 日韩av成人在线观看 | 久久精品亚洲一区二区三区观看模式 | 国产日产精品一区二区三区四区的观看方式 | 国产黄大片在线观看 | 久久久精品免费看 | 成人免费观看完整版电影 | 999ZYZ玖玖资源站永久 | 亚洲欧美色婷婷 | 在线观看视频免费大全 | 精品国产伦一区二区三区观看方式 | 黄色a级片在线观看 | 久久一区二区三区超碰国产精品 | 狠狠地操 | 中文字幕在线免费 | 免费网站污 | 午夜视频久久久 | 国产精品美乳一区二区免费 | 99精品黄色片免费大全 | 日韩理论在线播放 | 在线观看精品一区 | 精品国产欧美一区二区 | 一级免费看视频 | 精品久久久久国产免费第一页 | 成人国产精品免费 | 亚洲成人av影片 | 中文在线中文a | 国模吧一区 | 国内精品小视频 | 日韩免费看 | 97免费 | 久久国产精品久久精品国产演员表 | 在线www色 | 色国产精品一区在线观看 | 久久久免费精品视频 | 日韩视频免费观看高清完整版在线 | 日本中文字幕观看 | 亚洲精品在线一区二区三区 | www免费视频com | 久久国产热 | 欧美91精品国产自产 | 午夜精品一二区 | 精品国产伦一区二区三区观看说明 | 亚洲一区二区三区四区精品 | 国产中文视频 | 久久久福利影院 | 人人澡人人舔 | 色婷婷99 | 奇米影视999 | 天天玩夜夜操 | 国产精品久久99精品毛片三a | 久草在线综合 | 在线播放视频一区 | 丰满少妇在线观看资源站 | 中文字幕欧美三区 | 日本中文字幕一二区观 | 91麻豆高清视频 | 久久99电影 | 色综合夜色一区 | a级成人毛片| 久久不射网站 | 国产高清在线观看av | 免费看一级片 | 久久久免费国产 | 久久精彩免费视频 | 久草视频视频在线播放 | 亚洲欧美视频在线播放 | 久久tv视频 | 在线99热| 亚洲精品国偷拍自产在线观看蜜桃 | 极品久久久久 | 午夜精品婷婷 | 啪嗒啪嗒免费观看完整版 | 欧美疯狂性受xxxxx另类 | 久久午夜羞羞影院 | av亚洲产国偷v产偷v自拍小说 | 福利二区视频 | 久久精品麻豆 | 92精品国产成人观看免费 | 中日韩在线视频 | 久久久香蕉视频 | 91精品视频网站 | 日日摸日日添夜夜爽97 | www久久国产 | 国产激情小视频在线观看 | 日韩一级电影在线观看 | 人人草人人草 | 亚洲最大的av网站 | 日韩精品一区在线播放 | 中文字幕在线视频第一页 | 天天干天天射天天操 | 在线观看爱爱视频 | 国产在线p | 久久视频精品 | 亚洲黄色片 | 欧洲在线免费视频 | 中文字幕刺激在线 | 在线综合 亚洲 欧美在线视频 | 91精品欧美一区二区三区 | 91丨九色丨首页 | 日本少妇高清做爰视频 | 天天躁日日躁狠狠躁av中文 | 久久久91精品国产一区二区精品 | 国产剧情在线一区 | 欧美日韩性视频 | 69国产精品视频免费观看 | 久草在线视频免赞 | 欧美日韩天堂 | 国产美女精品久久久 | 久久精品免费观看 | 手机av在线网站 | 91精品一区二区三区蜜臀 | 人人爽人人爽人人 | 欧美在线视频免费 | 91传媒在线观看 | 丝袜护士aⅴ在线白丝护士 天天综合精品 | 天天天综合网 | 中文字幕一区二区三区久久 | 久久久久久福利 | 午夜精品久久久久久久99婷婷 | 日日天天干 | 亚洲欧美视频一区二区三区 | 手机成人av在线 | 96av在线视频 | 亚洲色图22p| 天天天干天天天操 | 九月婷婷人人澡人人添人人爽 | 欧美精品在线免费 | 91激情视频在线观看 | 亚洲黄色av网址 | 国产精品一区二区久久精品爱微奶 | 9999在线视频 | 久久伊人热 | 亚洲一级黄色片 | 精品国产理论片 | 成人国产精品av | 九九热re | 日韩精品欧美精品 | 91免费网站在线观看 | 日产乱码一二三区别免费 | 精品欧美一区二区在线观看 | 久久婷婷国产色一区二区三区 | 亚洲精品乱码久久久久久9色 | 欧美一区二区在线看 | 三级av网 | 国产精品一区二区久久国产 | 中文字幕在线播放一区二区 | 中文字幕在线视频免费播放 | 亚洲第一久久久 | 免费观看完整版无人区 | 99 色 | 狠狠狠色 | av中文字幕av | 成人午夜电影在线 | 欧美有色| av在线a | 日韩电影中文字幕 | 久久久免费看片 | 欧美xxxxx在线视频 | 丁香5月婷婷久久 | 97av视频 | 久久精品人人做人人综合老师 | 97在线观看免费观看 | 婷婷视频在线观看 | 欧美在线视频二区 | 91麻豆网站 | 亚洲精品字幕在线 | 亚洲一二三区精品 | 国产成人精品一区二区 | 99久久精品久久亚洲精品 | 欧美日韩亚洲精品在线 | 成人黄性视频 | 国产精品久久久久久久久久免费看 | 波多野结衣电影一区二区三区 | 黄色在线视频网址 | 久久综合九色综合网站 | 久久福利小视频 | av免费在线播放 | 最近日本韩国中文字幕 | 亚洲天堂网在线视频观看 | 亚洲最新合集 | 中文字幕一区二区三 | 久久综合久久综合这里只有精品 | 在线中文字幕一区二区 | 狠狠狠操 | 人人爽人人爽人人片av免 | 国产视频在线观看一区 | 99久久精品国产毛片 | 免费精品视频 | 久久久久久久久久久免费视频 | 成人一级 | 麻豆视频免费观看 | 久久久免费看视频 | 国产亚洲精品久久久久久大师 | 久久五月激情 | 日韩精品欧美精品 | 国产黄色免费 | 日韩一区正在播放 | 婷婷深爱网 | 成人在线视频网 | 精品国产一区二区三区免费 | 国产在线播放一区二区三区 | 婷婷综合伊人 | 中文字幕在线一区二区三区 | 欧美日韩性生活 | 成人黄色中文字幕 | 97超碰在线资源 | 国产色婷婷精品综合在线手机播放 | 国产精品福利视频 | 久久国产精品99久久久久久老狼 | 五月婷婷综合激情网 | 男女激情片在线观看 | 精品国产综合区久久久久久 | 久久五月婷婷丁香社区 | 国产又粗又硬又长又爽的视频 | 国产精品二区在线 | 九九免费在线看完整版 | 久久久精品小视频 | 色综合久久久久综合体桃花网 | 亚洲午夜精品久久久久久久久久久久 | 日韩欧美高清在线观看 | 色播五月激情五月 | 久久 地址| 少妇av片 | 日韩精品中文字幕在线 | 国产精品久久久久亚洲影视 | 一区二区三区播放 | 久久九九久久九九 | 久久99视频精品 | 国产流白浆高潮在线观看 | 日韩精品中文字幕一区二区 | 亚洲精品成人 | 日韩激情av在线 | 国产精品黄色在线观看 | 日韩在线网址 | 人人干天天射 | 久久综合成人网 | 91中文字幕网 | 欧美午夜激情网 | 国产手机视频在线 | 欧美色噜噜| 超碰人人草 | 成人免费视频网 | 免费看黄电影 | av中文在线观看 | 日韩av一区二区三区在线观看 | 国产成人一区二区在线观看 | 成人一级免费电影 | 免费国产黄线在线观看视频 | 天天操人人要 | 国产 在线观看 | 国产色啪| 美女av免费 | 超碰夜夜 | 日本久久中文字幕 | 国产亚洲精品福利 | 久久精品亚洲精品国产欧美 | 亚州精品在线视频 | 精品免费国产一区二区三区四区 | 日韩精品一区二区三区第95 | 精品久久久久久久久久久久久久久久 | 成人av网站在线观看 | 天天射天天操天天干 | 欧美另类tv | 午夜精品视频福利 | 日韩欧美综合在线视频 | 国产短视频在线播放 | 97在线视| 九热精品 | www.色综合.com | 亚洲精品在线二区 | 久久久久激情视频 | 91在线免费观看网站 | 91视频免费播放 | 精品久久久久久亚洲综合网站 | av黄色大片| 欧美日韩不卡一区二区 | 国产精品va | 97在线视频免费看 | 久久草草影视免费网 | 四虎影视精品 | 在线天堂亚洲 | 欧美国产日韩一区 | 久久久国产精品人人片99精片欧美一 | 国产视频一二区 | 亚洲精品高清一区二区三区四区 | 日本爱爱免费视频 | 亚洲国产片 | www黄色av| 国产97在线视频 | 看av免费网站 | 久久免费的精品国产v∧ | 最近中文字幕高清字幕在线视频 | 亚洲成人黄色在线观看 | 日日综合 | 色中文字幕在线观看 | 日韩av中文| 午夜黄色影院 | 色在线网 | 人人澡澡人人 | 免费h视频| 亚洲三级网站 | av爱干| 一区二区三区在线观看免费视频 | 久草在线在线视频 | 99精彩视频 | 久久精品8| 色婷婷伊人 | 亚洲免费在线看 | 国产精品99久久久久久久久 | 在线导航av | 九九综合九九综合 | 久久99久久99精品免费看小说 | 顶级欧美色妇4khd | 免费观看全黄做爰大片国产 | 久久人人爽人人人人片 | 日本黄色片一区二区 | a一片一级 | 欧美精品资源 | 婷婷社区五月天 | 黄色网址国产 | 久久久久国产精品视频 | 超碰av在线免费观看 | 中文字幕第一页在线vr | 在线日本看片免费人成视久网 | 在线观看亚洲国产 | 日韩三级一区 | 麻豆传媒精品 | 久久精品99视频 | 西西444www高清大胆 | 婷色| 欧美一进一出抽搐大尺度视频 | 国产白浆在线观看 | 伊人资源站 | 日韩在线观看影院 | 精品国产理论 | 欧美 亚洲 另类 激情 另类 | 色婷婷播放 | 一区在线免费观看 | 9在线观看免费高清完整 | 国产日韩一区在线 | 夜色资源站wwwcom | 91精品一| 6080yy精品一区二区三区 | 亚洲精品美女久久久久 | 亚洲少妇自拍 | 久草视频手机在线 | 午夜精品视频一区二区三区在线看 | 成人国产精品久久久春色 | 日韩黄色影院 | 色婷婷丁香 | 国产精品18久久久久久vr | 日韩三级久久 | 免费看黄在线网站 | 日韩一区二区三区免费电影 | 国产成免费视频 | 国产精品18久久久久vr手机版特色 | 开心综合网 | 国产精品一区电影 | 久久久精品视频成人 | 伊人久久精品久久亚洲一区 | 国产精品久久中文字幕 | 久产久精国产品 | 黄色免费大片 | 福利视频精品 | 久久涩涩网站 | 99精品国产99久久久久久福利 | 免费的国产精品 | 人人玩人人添人人澡超碰 | 久久久免费精品国产一区二区 | 国产原创中文在线 | 91精品久久久久久久久久入口 | 色播五月激情综合网 | 337p西西人体大胆瓣开下部 | 欧美成人免费在线 | 久久久久久久久久网 | 欧美在线日韩在线 | av电影一区二区三区 | 91麻豆精品 | 国产成人久久av | 在线91网| 在线观看精品一区 | 中文字幕精品一区二区精品 | 二区三区在线视频 | 操操色 | 美女视频黄网站 | www久久99| 伊在线视频| 西西44人体做爰大胆视频 | 久草在线免费看视频 | 国产高清永久免费 | 亚洲精品在线观看中文字幕 | 521色香蕉网站在线观看 | 国产传媒一区在线 | 久久久婷 | 福利精品在线 | 友田真希av| 国产成人黄色在线 | 国产香蕉97碰碰久久人人 | 中文字幕亚洲欧美日韩2019 | 97国产精品久久 | 胖bbbb搡bbbb擦bbbb | 香蕉精品在线观看 | 可以免费看av | 国内精品久久久久久久久久清纯 | 婷婷在线观看视频 | 91麻豆精品国产 | 日韩精品在线视频免费观看 | av电影av在线 | 久久精品国产久精国产 | 日b视频在线观看网址 | 亚洲五月 | 日韩一区二区三免费高清在线观看 | 亚洲欧美日韩精品久久久 | 中文字幕韩在线第一页 | 久久中文网 | 九九久久电影 | 天天草天天干天天 | 午夜电影中文字幕 | 国产成人精品999 | 深夜成人av | 中文字幕亚洲欧美日韩 | 婷婷www| 在线看成人 | 成人免费观看视频大全 | 蜜桃视频日本 | 99久久婷婷国产综合精品 | 青青草国产在线 | 黄色成人小视频 | 国产不卡精品 | 日韩精品中文字幕在线不卡尤物 | 亚洲日韩中文字幕 | 精品一区二区在线看 | 天天爱综合 | 成人免费大片黄在线播放 | 日韩天堂在线观看 | 一区二区欧美日韩 | 久久视屏网| 69视频永久免费观看 | 精品一区二区三区四区在线 | 日韩精品中文字幕在线不卡尤物 | 国产精品男女 | 永久av免费在线观看 | 手机色在线 | 国产精品wwwwww | 久久国产精品99国产精 | 国产精品99久久久 | 九色在线视频 | 久久免费看a级毛毛片 | 国产一级片免费观看 | 国产一级一片免费播放放a 一区二区三区国产欧美 | 日韩欧美第二页 | 亚洲国产资源 | 日韩欧美在线高清 | 天天操夜夜操国产精品 | 一区三区视频在线观看 | 中文字幕在线观看免费 | 久久尤物电影视频在线观看 | 久久综合偷偷噜噜噜色 | 丝袜美女在线观看 | 欧美日韩在线精品一区二区 | 91av久久 | 国产精品久久久久久久久久久久午夜片 | 国产麻豆精品传媒av国产下载 | 亚洲天堂精品 | 91人人揉日日捏人人看 | 国产一区国产二区在线观看 | 亚洲免费av一区二区 | 美女网站色在线观看 | 人人爱人人做人人爽 | 日本一区二区不卡高清 | 国产对白av| 欧美最猛性xxxxx(亚洲精品) | 中文字幕a∨在线乱码免费看 | 激情综合网五月 | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 精品视频成人 | 免费黄色看片 | 日本精品视频免费 | 午夜电影av | 99精品一区二区 | 欧美在一区 | 免费开视频 | 久久免费99精品久久久久久 | 天天操天天爽天天干 | 丁香色婷| 久久久www成人免费精品张筱雨 | 久久精品黄 | 午夜 久久 tv | 中文字幕国产一区 | 久久久亚洲影院 | 亚洲japanese制服美女 | 欧美一级大片在线观看 | 在线国产中文字幕 | 中国一级片在线 | 国产婷婷一区二区 | 免费成人在线网站 | 国产a级免费 | 韩国在线一区 | 久久黄色小说视频 | 91精品在线视频 | 国产午夜三级一区二区三桃花影视 | 国产在线精品一区二区不卡了 | 香蕉免费在线 | 在线精品视频免费播放 | 超碰97免费观看 | 超碰av在线播放 | 国产在线国偷精品产拍免费yy | 精品一区二区免费视频 | 免费黄色特级片 | 精品在线二区 | 粉嫩一区二区三区粉嫩91 | 久久毛片视频 | 欧美日比视频 | 免费国产在线精品 | 免费a网| 久久久夜色 | 久久99精品久久久久蜜臀 | 亚洲精品乱码久久 | 欧美一二区在线 | 看国产黄色大片 | 精品久久网站 | 久久婷婷综合激情 | 国产精品久久久久999 | 麻豆超碰 | 久久一二区 | 国产一区二区久久久 | 特级毛片在线免费观看 |