news 2026/3/8 9:22:25

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/3/7 11:29:07

《余行论》第七篇:历史篇

第七篇&#xff1a;历史篇——文明如长河&#xff0c;兴衰皆余行引言&#xff1a;历史是宇宙的备忘录&#xff0c;还是人类的独白&#xff1f;历史是什么&#xff1f;是帝王将相的谱系&#xff0c;是战争与和平的循环&#xff0c;是生产力进步的阶梯&#xff0c;还是偶然事件的…

作者头像 李华
网站建设 2026/3/7 9:19:40

《余行论》第九篇:证验篇

第九篇&#xff1a;证验篇——理可玄思&#xff0c;亦可实证引言&#xff1a;从哲学构想到科学纲领前八篇构建的“余行论”&#xff0c;以其宏大的视野与内在的融贯性&#xff0c;描绘了一幅从宇宙本源到人类文明的统一图景。然而&#xff0c;一个深邃的哲学思辨若想真正成为推…

作者头像 李华
网站建设 2026/3/8 2:33:09

构建动态响应式动画架构:lottie-ios与现代数据流技术融合实践

构建动态响应式动画架构&#xff1a;lottie-ios与现代数据流技术融合实践 【免费下载链接】lottie-ios airbnb/lottie-ios: Lottie-ios 是一个用于 iOS 平台的动画库&#xff0c;可以将 Adobe After Effects 动画导出成 iOS 应用程序&#xff0c;具有高性能&#xff0c;易用性和…

作者头像 李华
网站建设 2026/3/7 0:15:43

小程序商城搭建 自带拼团砍价功能 快速引爆销量

小程序商城搭建的核心技术框架微信小程序开发基础&#xff1a;WXML/WXSS/JavaScript后端技术选型&#xff1a;Node.js/Python&#xff08;Django/Flask&#xff09;或PHP&#xff08;ThinkPHP/Laravel&#xff09;数据库设计&#xff1a;MySQL/MongoDB存储用户、订单、拼团数据…

作者头像 李华
网站建设 2026/3/7 20:21:49

海外网红营销:超越促销,用“圣诞故事”绑定品牌情感

每到圣诞季&#xff0c;全球品牌都会涌入折扣大战&#xff0c;但真正能在用户心里留下印记的&#xff0c;往往不是促销力度&#xff0c;而是能让消费者“感同身受”的情绪触点。圣诞节承载着人们对家庭、传统、温暖、陪伴与仪式感的期待&#xff0c;而海外网红营销真正的价值&a…

作者头像 李华