news 2026/6/6 16:21:33

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/6/5 13:20:53

ATX自动化测试终极指南:从零开始快速上手

ATX自动化测试终极指南&#xff1a;从零开始快速上手 【免费下载链接】ATX Smart phone automation tool. Support iOS, Android, WebApp and game. 项目地址: https://gitcode.com/gh_mirrors/at/ATX ATX&#xff08;AutomatorX&#xff09;是一款由网易游戏团队开发的…

作者头像 李华
网站建设 2026/6/6 15:20:52

跨国企业合规痛点:多地区法规差异下的许可统一管理

跨国企业合规痛点&#xff1a;多地区法规差异下的许可统一管理在全球化加速发展的今天&#xff0c;越来越多的企业选择拓展国际市场&#xff0c;走向海外。跨国经营并非一条坦途&#xff0c;在许可合规管理方面&#xff0c;企业常常面临多地区法规差异带来的复杂问题。一个企业…

作者头像 李华
网站建设 2026/5/24 7:58:26

面向AI系统的数据隐私保护测试框架设计与实践

随着《个人信息保护法》和《数据安全法》的深入实施&#xff0c;AI系统数据隐私保护已成为测试工作的核心议题。2025年&#xff0c;全球数据泄露事件同比激增67%&#xff0c;其中AI模型训练与推理环节占比达41%。本文提出覆盖全生命周期的测试方案&#xff0c;帮助测试团队构建…

作者头像 李华
网站建设 2026/6/1 19:16:18

移动端自动化测试:工具选择与实战技巧深度解析

一、移动端自动化测试现状与挑战随着智能设备渗透率持续攀升&#xff0c;移动应用功能复杂度呈指数级增长。截至2025年&#xff0c;全球移动用户日均使用时长突破4小时&#xff0c;这对测试工作提出更高要求。传统手工测试面临三大瓶颈&#xff1a;设备碎片化&#xff08;需覆盖…

作者头像 李华
网站建设 2026/6/6 15:07:54

升级竞价思维:从被动跟随到主动布局的策略进化

在亚马逊广告投放的竞技场中&#xff0c;“系统建议出价”曾像海妖的歌声一般&#xff0c;诱惑着无数卖家踏入高成本低回报的迷雾&#xff0c;直到越来越多的人发现&#xff0c;这一“友好”的工具背后&#xff0c;潜藏着一套精密的算法逻辑——它并非以卖家的利润最大化为目的…

作者头像 李华
网站建设 2026/6/4 6:09:20

汽车行业最严重漏洞:20家知名车企API暴露车主个人信息

汽车行业最严重漏洞&#xff1a;20家知名车企API暴露车主个人信息 近日安全研究人员Sam Curry披露了近20家知名汽车制造商在线服务中的API安全漏洞&#xff0c;这些漏洞可能允许黑客执行恶意活动&#xff0c;包括从解锁、启动、跟踪汽车到窃取客户个人信息。这可能是汽车行业迄…

作者头像 李华