简介: CSDN博客专家、《Android系统多媒体进阶实战》作者
博主新书推荐:《Android系统多媒体进阶实战》🚀
Android Audio工程师专栏地址:Audio工程师进阶系列【原创干货持续更新中...... 】🚀
Android多媒体专栏地址:多媒体系统工程师系列【原创干货持续更新中...... 】🚀
专题一 二:AAOS车载系统+AOSP14系统攻城狮入门视频实战课 🚀
专题三:Android14 Binder之HIDL与AIDL通信实战课 🚀
专题四:Android15快速自定义与集成音效实战课 🚀
专题五:Android15音频策略实战课 🚀
专题六:Android15音频性能实战课(无声/杂音/断音/爆音实战案例) 🚀
人生格言: 人生从来没有捷径,只有行动才是治疗恐惧和懒惰的唯一良药.
更多原创,欢迎关注:Android系统攻城狮

🍉🍉🍉文章目录🍉🍉🍉
- [🌻1. 前言](#🌻1. 前言)
- [🌻2. 应用场景与用法](#🌻2. 应用场景与用法)
- 应用场景
- [🌻3. 调用流程剖析](#🌻3. 调用流程剖析)
-
- [🌻3.1 核心步骤](#🌻3.1 核心步骤)
-
- [1. 应用层发起请求](#1. 应用层发起请求)
- [2. 分配 pa_mainloop 对象](#2. 分配 pa_mainloop 对象)
- [3. 创建自唤醒管道](#3. 创建自唤醒管道)
- [4. 设置管道为非阻塞](#4. 设置管道为非阻塞)
- [5. 初始化事件 API 表](#5. 初始化事件 API 表)
- [6. 初始化 mainloop 状态](#6. 初始化 mainloop 状态)
- [7. 返回 mainloop 对象](#7. 返回 mainloop 对象)
- [8. 后续被 mainloop_run 消费](#8. 后续被 mainloop_run 消费)
- [🌻3.2 调用流程图](#🌻3.2 调用流程图)
- [🌻3.3 mainloop 创建生命周期图](#🌻3.3 mainloop 创建生命周期图)
- [🌻4. 实战应用案例](#🌻4. 实战应用案例)
- [🌻5. 一句话总结](#🌻5. 一句话总结)
🌻1. 前言
本篇目的:
Linux PulseAudio 深度解析之 pa_mainloop_new 调用流程与实战。
要点概括
- 核心功能 :创建一个
pa_mainloop对象,作为 libpulse 异步 API 的事件循环基础。 - 工作机制 :函数内部分配
pa_mainloop结构体,创建自唤醒管道wakeup_pipe[0/1],初始化事件 API 表、初始状态和 poll 相关字段,最后返回pa_mainloop*给应用层。 - 典型用途 :异步连接 PulseAudio Server、驱动
pa_context状态机、驱动pa_stream回调、管理 IO/Time/Defer 事件。
🌻2. 应用场景与用法
pa_mainloop_new() 是 PulseAudio 异步客户端编程中的事件循环创建接口。
在 PulseAudio 中,pa_context_connect()、pa_stream_connect_playback()、pa_stream_set_write_callback() 这类异步 API,都需要依赖 mainloop 驱动状态变化和回调触发。
而该接口用于:
创建一个可供 libpulse 异步 API 使用的
pa_mainloop对象。
函数原型
c
pa_mainloop* pa_mainloop_new(void);
参数说明
c
无参数
pa_mainloop_new() 不需要应用层传入参数,它内部负责创建并初始化 pa_mainloop 对象。
返回值
c
返回 pa_mainloop 对象
用于后续获取 mainloop API、创建 pa_context、注册回调,并通过 pa_mainloop_run() 或 pa_mainloop_iterate() 驱动整个异步流程。
应用场景
pa_mainloop_new() 常见应用场景主要有三类。
第一类是创建 PulseAudio 异步客户端。应用程序调用 pa_mainloop_new() 创建事件循环对象,然后通过 pa_mainloop_get_api() 获取事件 API 表,再传给 pa_context_new() 创建 pa_context,这是 libpulse 异步模型的标准入口。
第二类是驱动连接和 Stream 状态机。pa_context_connect()、pa_stream_connect_playback() 等接口本身不会阻塞等待完成,它们依赖 mainloop 监听 socket、dispatch 事件并触发状态回调。也就是说,mainloop 不运行,Context 和 Stream 的异步回调就不会继续推进。
第三类是管理 IO/Time/Defer 事件。PulseAudio 的异步模型并不是简单的 while 循环,而是通过 mainloop 统一管理 IO 事件、定时事件和 defer 事件。pa_mainloop_new() 创建的对象,就是后续 prepare -> poll -> dispatch 主循环的基础容器。
🌻3. 调用流程剖析
🌻3.1 核心步骤
1. 应用层发起请求
c
pa_mainloop *mainloop;
mainloop = pa_mainloop_new();
2. 分配 pa_mainloop 对象
内部创建:
c
pa_mainloop
用于保存事件 API、wakeup pipe、poll 状态、事件链表和 mainloop 当前状态。
3. 创建自唤醒管道
内部创建:
text
wakeup_pipe[0]
wakeup_pipe[1]
其中:
text
wakeup_pipe[0]:读端
wakeup_pipe[1]:写端
当 mainloop 阻塞在 poll/ppoll 时,其他事件可以通过写端写入字节,让读端变成可读,从而唤醒 mainloop。
4. 设置管道为非阻塞
对 wakeup_pipe[0] 和 wakeup_pipe[1] 设置非阻塞,避免读写唤醒管道时卡住 mainloop。
5. 初始化事件 API 表
内部设置:
c
m->api = vtable;
m->api.userdata = m;
其中 m->api 是暴露给 pa_context_new()、pa_stream_new() 等对象使用的事件操作表,m->api.userdata = m 表示这些事件 API 后续可以反向找到当前 pa_mainloop 对象。
6. 初始化 mainloop 状态
设置:
text
m->state = STATE_PASSIVE
m->poll_func_ret = -1
m->rebuild_pollfds = true
表示 mainloop 已创建,但还没有进入主动运行状态;后续第一次进入 poll 前,需要重新构建 pollfd 数组。
7. 返回 mainloop 对象
函数最终返回:
c
pa_mainloop*
应用层拿到该对象后,可以继续调用 pa_mainloop_get_api()、pa_context_new()、pa_context_connect() 和 pa_mainloop_run()。
8. 后续被 mainloop_run 消费
创建出来的 pa_mainloop 最终会被:
c
pa_mainloop_run(mainloop, NULL);
或:
c
pa_mainloop_iterate(mainloop, 1, NULL);
驱动,从而进入 prepare -> poll -> dispatch 事件循环。
🌻3.2 调用流程图

🌻3.3 mainloop 创建生命周期图

🌻4. 实战应用案例
c
#include <pulse/pulseaudio.h>
#include <stdio.h>
static void context_state_cb(
pa_context *c,
void *userdata) {
pa_context_state_t state;
state = pa_context_get_state(c);
if (state == PA_CONTEXT_READY)
printf("context ready\n");
if (state == PA_CONTEXT_FAILED ||
state == PA_CONTEXT_TERMINATED) {
pa_mainloop *mainloop;
mainloop = userdata;
pa_mainloop_quit(
mainloop,
0);
}
}
int main() {
pa_mainloop *mainloop;
pa_mainloop_api *api;
pa_context *context;
/*
* 创建 mainloop 对象
*/
mainloop =
pa_mainloop_new();
/*
* 获取 mainloop API 表
*/
api =
pa_mainloop_get_api(
mainloop);
/*
* 创建 context
*/
context =
pa_context_new(
api,
"pa_mainloop_new_demo");
/*
* 注册 context 状态回调
*/
pa_context_set_state_callback(
context,
context_state_cb,
mainloop);
/*
* 发起异步连接
*/
pa_context_connect(
context,
NULL,
0,
NULL);
/*
* 启动事件循环
*/
pa_mainloop_run(
mainloop,
NULL);
pa_context_unref(
context);
pa_mainloop_free(
mainloop);
return 0;
}
🌻5. 一句话总结
pa_mainloop_new() 本质上是:
"创建 libpulse 异步模型的事件循环对象"。
它负责分配 pa_mainloop、创建自唤醒管道、初始化事件 API 表和初始运行状态,是 pa_context_connect()、pa_stream 回调、IO 事件、Time 事件和 Defer 事件能够被驱动起来的基础入口。