ffmpeg api-band-test.c 讲解

2024-01-08 16:52
文章标签 讲解 ffmpeg test api band

本文主要是介绍ffmpeg api-band-test.c 讲解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

ffmpeg 4.0.4 源码讲解

draw_horiz_band

av_pix_fmt_desc_get 函数是 FFmpeg 中用于获取像素格式描述信息的函数。它的作用是根据给定的像素格式(AVPixelFormat)返回对应的像素格式描述结构体(AVPixFmtDescriptor),该结构体包含了关于像素格式的详细信息,如分量数、每个分量的位深度、颜色空间等。

参数说明:
pix_fmt:要查询的像素格式,是一个枚举值,表示视频帧的像素格式,如 AV_PIX_FMT_YUV420P、AV_PIX_FMT_YUV422P 等。
返回值:
如果成功找到对应的像素格式描述结构体,则返回该结构体的指针;如果未找到,则返回 NULL。


static void draw_horiz_band(AVCodecContext *ctx, const AVFrame *fr, int offset[4],int slice_position, int type, int height)
{int i;const AVPixFmtDescriptor *pix_fmt_desc;int chroma_w, chroma_h;int shift_slice_position;int shift_height;// 标记 draw_horiz_band 被调用draw_horiz_band_called = 1;// 获取像素格式描述信息pix_fmt_desc = av_pix_fmt_desc_get(ctx->pix_fmt);// 计算色度分量宽高chroma_w = -((-ctx->width) >> pix_fmt_desc->log2_chroma_w);chroma_h = -((-height) >> pix_fmt_desc->log2_chroma_h);/*ctx->width 表示视频帧的宽度,height 表示视频帧的高度,而 pix_fmt_desc->log2_chroma_w 和 pix_fmt_desc->log2_chroma_h 分别表示色度分量宽度和高度的对数值(即以2为底的对数)。
举个例子,假设视频帧的宽度 ctx->width 为 1920,高度 height 为 1080,而色度分量的采样率为 4:2:0(常见的视频编码格式之一),即色度分量的宽度和高度都是亮度分量的一半。
现在假设 pix_fmt_desc->log2_chroma_w 和 pix_fmt_desc->log2_chroma_h 都为 1(因为 2^1 = 2,所以色度分量的宽度和高度都是亮度分量的一半)。
根据这些假设,我们来看一下这两行代码的计算过程:首先,将视频帧的宽度 ctx->width 取反,即 -1920。然后右移 pix_fmt_desc->log2_chroma_w 位,
即右移 1 位,得到 -960。再次取反,得到最终的 chroma_w 值为 960。将视频帧的高度 height 取反,即 -1080。然后右移 pix_fmt_desc->log2_chroma_h 位,即右移 1 位,
得到 -540。再次取反,得到最终的 chroma_h 值为 540。*/// 计算偏移值shift_slice_position = -((-slice_position) >> pix_fmt_desc->log2_chroma_h);shift_height = -((-ctx->height) >> pix_fmt_desc->log2_chroma_h);// 处理 Y 分量数据for (i = 0; i < height; i++) {// 拷贝 Y 分量数据到缓冲区memcpy(slice_byte_buffer + ctx->width * slice_position + i * ctx->width,fr->data[0] + offset[0] + i * fr->linesize[0], ctx->width);}// 处理 U 分量数据for (i = 0; i < chroma_h; i++) {// 拷贝 U 分量数据到缓冲区memcpy(slice_byte_buffer + ctx->width * ctx->height + chroma_w * shift_slice_position + i * chroma_w,fr->data[1] + offset[1] + i * fr->linesize[1], chroma_w);}// 处理 V 分量数据for (i = 0; i < chroma_h; i++) {// 拷贝 V 分量数据到缓冲区memcpy(slice_byte_buffer + ctx->width * ctx->height + chroma_w * shift_height + chroma_w * shift_slice_position + i * chroma_w,fr->data[2] + offset[2] + i * fr->linesize[2], chroma_w);}
}

video_decode

static int video_decode(const char *input_filename)
{AVCodec *codec = NULL; // 视频编解码器AVCodecContext *ctx = NULL; // 编解码器上下文AVCodecParameters *origin_par = NULL; // 视频流参数uint8_t *byte_buffer = NULL; // 存储解码后图像数据的缓冲区AVFrame *fr = NULL; // 存储解码后图像帧的结构体AVPacket pkt; // 存储视频数据的数据包AVFormatContext *fmt_ctx = NULL; // 存储格式相关的上下文信息int number_of_written_bytes; // 写入字节数int video_stream; // 视频流索引int got_frame = 0; // 是否解码到一帧图像int byte_buffer_size; // 缓冲区大小int result; // 操作结果int end_of_stream = 0; // 是否到达视频流末尾draw_horiz_band_called = 0; // 标志位,用于检查 draw_horiz_band 函数是否被调用过// 打开输入文件并获取格式相关的上下文信息result = avformat_open_input(&fmt_ctx, input_filename, NULL, NULL);if (result < 0) {av_log(NULL, AV_LOG_ERROR, "Can't open file\n");return result;}// 获取流信息result = avformat_find_stream_info(fmt_ctx, NULL);if (result < 0) {av_log(NULL, AV_LOG_ERROR, "Can't get stream info\n");return result;}// 找到视频流video_stream = av_find_best_stream(fmt_ctx, AVMEDIA_TYPE_VIDEO, -1, -1, NULL, 0);if (video_stream < 0) {av_log(NULL, AV_LOG_ERROR, "Can't find video stream in input file\n");return -1;}// 获取视频流的参数origin_par = fmt_ctx->streams[video_stream]->codecpar;// 查找视频编解码器codec = avcodec_find_decoder(origin_par->codec_id);if (!codec) {av_log(NULL, AV_LOG_ERROR, "Can't find decoder\n");return -1;}// 分配编解码器上下文ctx = avcodec_alloc_context3(codec);if (!ctx) {av_log(NULL, AV_LOG_ERROR, "Can't allocate decoder context\n");return AVERROR(ENOMEM);}// 将流参数拷贝到编解码器上下文result = avcodec_parameters_to_context(ctx, origin_par);if (result) {av_log(NULL, AV_LOG_ERROR, "Can't copy decoder context\n");return result;}// 设置 draw_horiz_band 函数ctx->draw_horiz_band = draw_horiz_band;ctx->thread_count = 1; // 设置线程数量为 1// 打开编解码器result = avcodec_open2(ctx, codec, NULL);if (result < 0) {av_log(ctx, AV_LOG_ERROR, "Can't open decoder\n");return result;}// 分配图像帧结构体fr = av_frame_alloc();if (!fr) {av_log(NULL, AV_LOG_ERROR, "Can't allocate frame\n");return AVERROR(ENOMEM);}// 分配图像数据缓冲区byte_buffer_size = av_image_get_buffer_size(ctx->pix_fmt, ctx->width, ctx->height, 32);byte_buffer = av_malloc(byte_buffer_size);if (!byte_buffer) {av_log(NULL, AV_LOG_ERROR, "Can't allocate buffer\n");return AVERROR(ENOMEM);}// 分配 slice_byte_buffer 存储 slice 数据slice_byte_buffer = av_malloc(byte_buffer_size);if (!slice_byte_buffer) {av_log(NULL, AV_LOG_ERROR, "Can't allocate buffer\n");return AVERROR(ENOMEM);}memset(slice_byte_buffer, 0, byte_buffer_size);slice_byte_buffer_size = byte_buffer_size;// 初始化数据包av_init_packet(&pkt);do {// 读取数据包if (!end_of_stream) {if (av_read_frame(fmt_ctx, &pkt) < 0) {end_of_stream = 1;}}if (end_of_stream) {pkt.data = NULL;pkt.size = 0;}// 解码视频帧if (pkt.stream_index == video_stream || end_of_stream) {got_frame = 0;result = avcodec_decode_video2(ctx, fr, &got_frame, &pkt);if (result < 0) {av_log(NULL, AV_LOG_ERROR, "Error decoding frame\n");return result;}// 如果解码到一帧图像if (got_frame) {// 处理解码后的图像数据process_frame(fr, byte_buffer, byte_buffer_size);}}// 释放数据包av_packet_unref(&pkt);} while (!end_of_stream || got_frame);// 释放资源avcodec_free_context(&ctx);avformat_close_input(&fmt_ctx);av_frame_free(&fr);av_free(byte_buffer);av_free(slice_byte_buffer);return 0;
}

这篇关于ffmpeg api-band-test.c 讲解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/584150

相关文章

SpringBoo WebFlux+MongoDB实现非阻塞API过程

《SpringBooWebFlux+MongoDB实现非阻塞API过程》本文介绍了如何使用SpringBootWebFlux和MongoDB实现非阻塞API,通过响应式编程提高系统的吞吐量和响应性能... 目录一、引言二、响应式编程基础2.1 响应式编程概念2.2 响应式编程的优势2.3 响应式编程相关技术

java中4种API参数传递方式统一说明

《java中4种API参数传递方式统一说明》在Java中,我们可以使用不同的方式来传递参数给方法或函数,:本文主要介绍java中4种API参数传递方式的相关资料,文中通过代码介绍的非常详细,需要的... 目录1. 概述2. 参数传递方式分类2.1 Query Parameters(查询参数)2.2 Path

Java调用DeepSeek API的8个高频坑与解决方法

《Java调用DeepSeekAPI的8个高频坑与解决方法》现在大模型开发特别火,DeepSeek因为中文理解好、反应快、还便宜,不少Java开发者都用它,本文整理了最常踩的8个坑,希望对... 目录引言一、坑 1:Token 过期未处理,鉴权异常引发服务中断问题本质典型错误代码解决方案:实现 Token

Python+FFmpeg实现视频自动化处理的完整指南

《Python+FFmpeg实现视频自动化处理的完整指南》本文总结了一套在Python中使用subprocess.run调用FFmpeg进行视频自动化处理的解决方案,涵盖了跨平台硬件加速、中间素材处理... 目录一、 跨平台硬件加速:统一接口设计1. 核心映射逻辑2. python 实现代码二、 中间素材处

SQL Server中行转列方法详细讲解

《SQLServer中行转列方法详细讲解》SQL行转列、列转行可以帮助我们更方便地处理数据,生成需要的报表和结果集,:本文主要介绍SQLServer中行转列方法的相关资料,需要的朋友可以参考下... 目录前言一、为什么需要行转列二、行转列的基本概念三、使用PIVOT运算符进行行转列1.创建示例数据表并插入数

C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解

《C++,C#,Rust,Go,Java,Python,JavaScript的性能对比全面讲解》:本文主要介绍C++,C#,Rust,Go,Java,Python,JavaScript性能对比全面... 目录编程语言性能对比、核心优势与最佳使用场景性能对比表格C++C#RustGoJavapythonjav

VS Code中的Python代码格式化插件示例讲解

《VSCode中的Python代码格式化插件示例讲解》在Java开发过程中,代码的规范性和可读性至关重要,一个团队中如果每个开发者的代码风格各异,会给代码的维护、审查和协作带来极大的困难,这篇文章主... 目录前言如何安装与配置使用建议与技巧如何选择总结前言在 VS Code 中,有几款非常出色的 pyt

Java中实现对象的拷贝案例讲解

《Java中实现对象的拷贝案例讲解》Java对象拷贝分为浅拷贝(复制值及引用地址)和深拷贝(递归复制所有引用对象),常用方法包括Object.clone()、序列化及JSON转换,需处理循环引用问题,... 目录对象的拷贝简介浅拷贝和深拷贝浅拷贝深拷贝深拷贝和循环引用总结对象的拷贝简介对象的拷贝,把一个

使用Go调用第三方API的方法详解

《使用Go调用第三方API的方法详解》在现代应用开发中,调用第三方API是非常常见的场景,比如获取天气预报、翻译文本、发送短信等,Go作为一门高效并发的编程语言,拥有强大的标准库和丰富的第三方库,可以... 目录引言一、准备工作二、案例1:调用天气查询 API1. 注册并获取 API Key2. 代码实现3

PHP应用中处理限流和API节流的最佳实践

《PHP应用中处理限流和API节流的最佳实践》限流和API节流对于确保Web应用程序的可靠性、安全性和可扩展性至关重要,本文将详细介绍PHP应用中处理限流和API节流的最佳实践,下面就来和小编一起学习... 目录限流的重要性在 php 中实施限流的最佳实践使用集中式存储进行状态管理(如 Redis)采用滑动