【FFmpeg】理解 av_packet_from_data 和 av_packet_unref 接口

问题:FFmpeg 8.0 使用下面这段代码有问题吗?里面说 外部缓冲 pStream 不会被 av_packet_unref 释放,所以不会影响原始外部缓冲 pStream ,对不对?

cpp 复制代码
BOOL32 SaveMP4(u8* pStream, u32 dwSize, u32 dwMediaType )
{

	AVFormatContext *fmt_ctx = NULL;
	AVStream *stream = NULL;
	AVPacket *pkt = NULL;
	
	// 分配 packet
	pkt = av_packet_alloc();
	
	// 将外部数据缓冲 pStream 绑定到 packet(零拷贝)
	ret = av_packet_from_data(pkt, pStream, (int)dwSize);
	
	//TODO.....
	
	// 清理 packet(为下一次调用准备)
	av_packet_unref(pkt);// 解引用 packet(不会释放 pStream,因来自 av_packet_from_data)
	        // 注意:pStream已被 pkt 拥有,av_packet_unref 不会 free 它
	        // 所以这里不需要 free(frame_data) ------ 由 av_packet_free 内部处理
	        // 但 av_packet_from_data 后,内存由 packet 管理,无需手动 free

//TODO.....
}

不对,这个理解是错误的!

🔍 源码佐证(FFmpeg 8.0 libavcodec/packet.c)

av_packet_from_data 的实现:

c 复制代码
/**
 * Initialize a reference-counted packet from av_malloc()ed data.
 *
 * @param pkt packet to be initialized. This function will set the data, size,
 *        and buf fields, all others are left untouched.
 * @param data Data allocated by av_malloc() to be used as packet data. If this
 *        function returns successfully, the data is owned by the underlying AVBuffer.
 *        The caller may not access the data through other means.
 * @param size size of data in bytes, without the padding. I.e. the full buffer
 *        size is assumed to be size + AV_INPUT_BUFFER_PADDING_SIZE.
 *
 * @return 0 on success, a negative AVERROR on error
 */

int av_packet_from_data(AVPacket *pkt, uint8_t *data, int size)
{
    if (size >= INT_MAX - AV_INPUT_BUFFER_PADDING_SIZE)
        return AVERROR(EINVAL);

    pkt->buf = av_buffer_create(data, size + AV_INPUT_BUFFER_PADDING_SIZE,
                                av_buffer_default_free, NULL, 0);
    if (!pkt->buf)
        return AVERROR(ENOMEM);

    pkt->data = data;
    pkt->size = size;

    return 0;
}

av_packet_unref 的实现:

c 复制代码
/**
 * Wipe the packet.
 *
 * Unreference the buffer referenced by the packet and reset the
 * remaining packet fields to their default values.
 *
 * @param pkt The packet to be unreferenced.
 */

void av_packet_unref(AVPacket *pkt)
{
    av_packet_free_side_data(pkt);
    av_buffer_unref(&pkt->opaque_ref);
    av_buffer_unref(&pkt->buf);
    get_packet_defaults(pkt);
}

根据贴出的 FFmpeg 8.0 的真实源码,恰恰说明:

av_packet_unref(pkt) 会释放 pStream


🔍 源码分析(FFmpeg 8.0)

1. av_packet_from_data 做了什么?
c 复制代码
pkt->buf = av_buffer_create(data, size + AV_INPUT_BUFFER_PADDING_SIZE,
                            av_buffer_default_free, NULL, 0);
  • 它用 av_buffer_create 创建了一个 AVBufferRef
  • 引用的数据就是你传入的 data(即 pStream
  • 释放回调函数是 av_buffer_default_free

av_buffer_default_free 的定义是:

c 复制代码
void av_buffer_default_free(void *opaque, uint8_t *data)
{
    av_free(data);  // ← 最终会调用 av_free(pStream)!
}
2. av_packet_unref 做了什么?
c 复制代码
av_buffer_unref(&pkt->buf);  // ← 释放 buf,触发 av_buffer_default_free
  • buf 的引用计数归零时,会调用 av_buffer_default_free
  • 从而调用 av_free(pStream)

❌ 所以结论是:

av_packet_from_data(pkt, pStream, size) + av_packet_unref(pkt)
⇒ 会导致 av_free(pStream) 被调用!

这意味着:

pStream 来源 是否安全
通过 av_malloc 分配 ✅ 安全(av_free 匹配)
通过 malloc 分配 ⚠️ 可能安全(若 av_free 就是 free
通过 new[] 分配 ❌ 危险(av_freedelete[]
栈上数组(如 uint8_t buf[1000] 崩溃!非法指针
网络/解码器内部 buffer(只读) 提前释放或 double-free

📌 官方文档怎么说?

FFmpeg 官方对 av_packet_from_data 的说明:

The packet data buffer is managed by the packet , and is freed when the packet is unreferenced .

The data must have been allocated with av_malloc() or a compatible allocator.

明确要求:data 必须由 av_malloc() 或兼容分配器分配!


✅ 正确使用方式

情况 1:你想"零拷贝"且拥有 pStream 生命周期
cpp 复制代码
// 必须用 av_malloc 分配 pStream!
uint8_t* my_data = (uint8_t*)av_malloc(dwSize + AV_INPUT_BUFFER_PADDING_SIZE);
memcpy(my_data, original_stream, dwSize);
memset(my_data + dwSize, 0, AV_INPUT_BUFFER_PADDING_SIZE); // padding required

av_packet_from_data(pkt, my_data, dwSize);
// ...
av_packet_unref(pkt); // 安全:内部 av_free(my_data)
情况 2:pStream 是外部只读 buffer(绝大多数情况)

不要用 av_packet_from_data!改用拷贝:

cpp 复制代码
av_new_packet(pkt, dwSize);      // 内部 av_malloc
memcpy(pkt->data, pStream, dwSize);
// ...
av_packet_unref(pkt); // 安全释放内部 buffer

❌ 你的注释为什么错?

// av_packet_unref 不会 free 它
// pStream 已被 pkt 拥有,无需手动 free

事实正好相反

  • pStreampkt 拥有
  • 并且 av_packet_unreffree 它!

所以如果你传入的是普通指针(非 av_malloc),程序极可能崩溃


✅ 总结

说法 正确? 说明
"av_packet_unref 不会释放 pStream" ❌ 错误 会释放!
"pStream 是外部 buffer,可安全传给 av_packet_from_data" ❌ 危险 除非它是 av_malloc 分配的
"应使用 av_new_packet + memcpy 处理外部数据" ✅ 正确 推荐做法

🔧 建议

在你的 SaveMP4 函数中,只要 pStream 不是你用 av_malloc 分配的,就绝对不要用 av_packet_from_data

使用:

cpp 复制代码
av_new_packet(pkt, dwSize);
memcpy(pkt->data, pStream, dwSize);

这是最安全、最清晰、最符合常规用法的方式。

🔧 附加建议

如果你担心性能(频繁 memcpy),可以考虑:

  • 使用 AVBufferRef 自定义内存管理(高级用法)
  • 或者确保上游提供 av_malloc 分配的 buffer(不现实)

但在 99% 的封装场景中,一次 memcpy 的开销远小于播放失败或 crash 的代价


请立即修改代码,避免使用 av_packet_from_data 处理外部 buffer!

相关推荐
1234567890@world2 小时前
FFmpeg | Day1 FFmpege音视频开发与学习
学习·ffmpeg·音视频
MaximusCoder1 天前
等保测评命令——达梦数据库 DM
linux·运维·数据库·安全·ffmpeg·安全威胁分析
EasyCVR1 天前
从流媒体转发到智能分析:EasyCVR的视频技术演进
人工智能·ffmpeg·音视频·视频监控·gb28181
南宫乘风2 天前
从零开发AI诊断Agent:拆解LLM+Tools+Prompt三大核心
人工智能·ffmpeg·prompt
l1t3 天前
使用ffmpeg把英语四级听力MP3文件转成wav供模型识别
ffmpeg·语音识别
liulilittle5 天前
Windows 11 上搭建 YouTube 视频下载工具:yt-dlp + FFmpeg
windows·ffmpeg·音视频
指尖在键盘上舞动5 天前
Cannot find matching video player interface for ‘ffpyplayer‘.解决方案
linux·ubuntu·ffmpeg·psychopy·ffpyplayer
glumes5 天前
音视频教程-第二节
ffmpeg
softbangong6 天前
899-批量视频处理工具
ffmpeg·格式转换·批量处理·音频提取·视频处理工具·视频截取
智能工业品检测-奇妙智能7 天前
SpringBoot整合FFmpeg的方法
人工智能·ffmpeg·springboot·deepseek·openclaw