news 2026/4/16 10:58:26

FFmepg-- 32-ffplay源码- PacketQueue 的线程安全机制 以及 serial 字段的作用

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
FFmepg-- 32-ffplay源码- PacketQueue 的线程安全机制 以及 serial 字段的作用

文章目录

      • 一 PacketQueue 的线程安全设计
      • 线程同步手段
      • 二 serial 字段的作用详解
        • 为什么需要 serial?
        • serial 的工作机制
      • 三 简化版示例代码
        • 使用场景(解码线程伪代码)
        • Seek 发生时
      • 四 总结

一 PacketQueue 的线程安全设计

在 ffplay.c 中,PacketQueue 是一个典型的生产者-消费者队列:

生产者:read_thread(从文件/网络读取 AVPacket 并放入队列)
消费者:解码线程(如 audio_thread / video_thread)从队列中取出 AVPacket 解码

线程同步手段

typedefstructPacketQueue{AVPacketList*first_pkt,*last_pkt;intnb_packets;// 当前包数量intsize;// 总字节数(用于限流)int64_tduration;// 总时长(毫秒)intabort_request;// 是否请求终止intserial;// 👈 关键字段:播放序列号SDL_mutex*mutex;// 互斥锁SDL_cond*cond;// 条件变量}PacketQueue;

SDL_mutex:保护对队列结构体(如 first_pkt, last_pkt, nb_packets 等)的并发访问。
SDL_cond:用于阻塞/唤醒:
消费者调用 packet_queue_get(…, block=1) 时,若队列为空,则 SDL_CondWait(cond, mutex) 阻塞;
生产者调用 packet_queue_put() 后,调用 SDL_CondSignal(cond) 唤醒等待的消费者。

二 serial 字段的作用详解

为什么需要 serial?

当用户执行 seek(快进/快退)操作时,旧的 AVPacket 已经无效,必须被丢弃。此时可能出现以下情况:

read_thread 可能还在往队列里塞旧数据;
解码线程可能还在处理旧数据;
如果不清除这些“过期”数据,就会出现:
音频“回放杂音”
视频跳帧混乱
音画不同步

serial 用于标识“当前播放上下文”的版本号。

serial 的工作机制

初始化时:q->serial = 0

执行 seek 时:
调用packet_queue_flush(&is->audioq)清空队列;
插入一个特殊的flush_pkt(其data == NULL);
执行q->serial++(例如从 0 → 1)

此后所有新入队的 packet 都会设置pkt->serial = q->serial

解码线程在取到 packet 后,会检查:

if(pkt->serial!=decoder->pkt_serial){av_packet_unref(pkt);continue;// 丢弃旧序列数据}

其中decoder->pkt_serial会在 flush 后被更新为新的 serial。

三 简化版示例代码

以下是一个高度简化但功能完整的PacketQueue实现,突出serial和线程安全逻辑:

#include<SDL2/SDL.h>#include<libavcodec/avcodec.h>#defineMAX_QUEUE_SIZE(15*1024*1024)typedefstructMyAVPacketList{AVPacket pkt;intserial;structMyAVPacketList*next;}MyAVPacketList;typedefstructPacketQueue{MyAVPacketList*first,*last;intnb_packets;intsize;int64_tduration;intabort_request;intserial;// 序列号SDL_mutex*mutex;SDL_cond*cond;}PacketQueue;voidpacket_queue_init(PacketQueue*q){memset(q,0,sizeof(PacketQueue));q->mutex=SDL_CreateMutex();q->cond=SDL_CreateCond();q->serial=0;}intpacket_queue_put(PacketQueue*q,AVPacket*pkt){MyAVPacketList*pkt1;SDL_LockMutex(q->mutex);if(q->abort_request){SDL_UnlockMutex(q->mutex);return-1;}pkt1=av_malloc(sizeof(MyAVPacketList));if(!pkt1)gotofail;pkt1->pkt=*pkt;pkt1->serial=q->serial;// 绑定当前 serialpkt1->next=NULL;if(!q->last)q->first=pkt1;elseq->last->next=pkt1;q->last=pkt1;q->nb_packets++;q->size+=pkt1->pkt.size+sizeof(*pkt1);q->duration+=pkt1->pkt.duration;SDL_CondSignal(q->cond);// 唤醒消费者SDL_UnlockMutex(q->mutex);return0;fail:SDL_UnlockMutex(q->mutex);return-1;}// 获取 packet,block=1 表示阻塞等待intpacket_queue_get(PacketQueue*q,AVPacket*pkt,intblock,int*serial){MyAVPacketList*pkt1;intret;SDL_LockMutex(q->mutex);for(;;){if(q->abort_request){ret=-1;break;}pkt1=q->first;if(pkt1){q->first=pkt1->next;if(!q->first)q->last=NULL;q->nb_packets--;q->size-=pkt1->pkt.size+sizeof(*pkt1);q->duration-=pkt1->pkt.duration;*pkt=pkt1->pkt;if(serial)*serial=pkt1->serial;// 返回 packet 的 serialav_free(pkt1);ret=1;break;}elseif(!block){ret=0;break;}else{SDL_CondWait(q->cond,q->mutex);// 阻塞等待}}SDL_UnlockMutex(q->mutex);returnret;}// seek 时调用:清空队列 + serial++voidpacket_queue_flush(PacketQueue*q){MyAVPacketList*pkt,*pkt1;SDL_LockMutex(q->mutex);for(pkt=q->first;pkt;pkt=pkt1){pkt1=pkt->next;av_packet_unref(&pkt->pkt);av_free(pkt);}q->first=q->last=NULL;q->nb_packets=0;q->size=0;q->duration=0;q->serial++;// 关键:序列号递增SDL_UnlockMutex(q->mutex);}
使用场景(解码线程伪代码)
intwanted_serial=is->audioq.serial;// 期望的 serialwhile(1){intserial;AVPacket pkt;if(packet_queue_get(&is->audioq,&pkt,1,&serial)<0)break;if(serial!=wanted_serial){av_packet_unref(&pkt);continue;// 丢弃旧序列数据}// 正常解码...decode_audio(&pkt);av_packet_unref(&pkt);}
Seek 发生时
// 用户 seek 到新位置packet_queue_flush(&is->audioq);// serial 自增packet_queue_flush(&is->videoq);// read_thread 会重新开始读取,并给新 packet 打上新 serial

四 总结

机制作用
SDL_mutex + SDL_cond实现线程安全的生产者-消费者队列
serial 字段标识“播放上下文”,避免 seek 后旧数据污染
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/14 20:43:27

从“画图员“到“方案师“:AI如何释放设计师创造力?

摘要&#xff1a;在《人工智能法》保障使用者知识产权的背景下&#xff0c;设计院若继续将工程师困于重复绘图工作&#xff0c;则不仅浪费人力&#xff0c;还可能丧失对AI生成内容的合法主张权。当前&#xff0c;70%的设计时间被耗费在格式调整和规范查对等低创造性任务上&…

作者头像 李华
网站建设 2026/4/13 15:45:44

性价比高的袋式过滤器生产厂家

如何甄选性价比高的袋式过滤器生产厂家&#xff1f;行业采购指南在工业流体处理领域&#xff0c;袋式过滤器因其经济、效率高、维护简便的特点&#xff0c;被广泛应用于化工、电子、新能源、食品饮料、水处理等多个行业。对于采购方而言&#xff0c;面对市场上众多的供应商&…

作者头像 李华
网站建设 2026/4/3 4:36:02

LobeChat是否具备内存泄漏检测?长期运行稳定性评估

LobeChat 是否具备内存泄漏检测&#xff1f;长期运行稳定性评估 在构建企业级 AI 助手门户的今天&#xff0c;一个看似简单的聊天界面背后&#xff0c;往往隐藏着复杂的性能挑战。LobeChat 作为当前最受欢迎的开源 ChatGPT 替代前端之一&#xff0c;凭借其现代化的设计和灵活的…

作者头像 李华
网站建设 2026/4/16 8:39:33

提示工程实战指南:5大维度提升Gemini模型输出质量

提示工程实战指南&#xff1a;5大维度提升Gemini模型输出质量 【免费下载链接】generative-ai Sample code and notebooks for Generative AI on Google Cloud 项目地址: https://gitcode.com/GitHub_Trending/ge/generative-ai 你是否经常遇到Gemini模型输出不稳定、不…

作者头像 李华
网站建设 2026/4/16 10:14:09

研发团队许可抢占:错峰使用与优先级分配策略解析

研发团队许可抢占&#xff1a;错峰使用与优先级分配策略解析作为一名长期在技术团队中负责系统规划与资源管理的工程师&#xff0c;我经常会遇到这样的问题&#xff1a;当多个研发团队同时申请使用相同资源时&#xff0c;如何高效、公平地分配&#xff0c;以避免资源冲突、效率…

作者头像 李华
网站建设 2026/4/16 5:56:57

WAN2.2-14B-Rapid-AllInOne:重新定义多模态视频创作效率的智能引擎

在AI视频生成技术快速迭代的当下&#xff0c;WAN2.2-14B-Rapid-AllInOne以其独特的"一体化"设计理念&#xff0c;为内容创作者带来了前所未有的操作便利性。这款融合了WAN 2.2核心架构与多种优化技术的创新方案&#xff0c;正在改写传统视频制作的工作流程。 【免费下…

作者头像 李华