FFMPEG内存操作(一) avio_reading.c 回调读取数据到内存解析

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: 相关博客列表 :     FFMPEG内存操作(一) avio_reading.c 回调读取数据到内存解析      FFMPEG内存操作(二)从内存中读取数及数据格式的转换      FFmpeg内存操作(三)内存转码器    在FFMPEG的官方例程中,有个avio_reading.c 的文件,他的主要功能是将音视频文件读取到内存,如果FFMPEG需要使用输入文件的数据,则直接从内存中调用。

相关博客列表 :

    FFMPEG内存操作(一) avio_reading.c 回调读取数据到内存解析 

    FFMPEG内存操作(二)从内存中读取数及数据格式的转换 

    FFmpeg内存操作(三)内存转码器

   在FFMPEG的官方例程中,有个avio_reading.c 的文件,他的主要功能是将音视频文件读取到内存,如果FFMPEG需要使用输入文件的数据,则直接从内存中调用。初学FFMPEG,给avio_reading.c 文件做了一个注释,如果不对,欢迎指正。

 

[objc]  view plain  copy
 
 print?
  1. /* 
  2.  * Copyright (c) 2014 Stefano Sabatini 
  3.  * 
  4.  * Permission is hereby granted, free of charge, to any person obtaining a copy 
  5.  * of this software and associated documentation files (the "Software"), to deal 
  6.  * in the Software without restriction, including without limitation the rights 
  7.  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell 
  8.  * copies of the Software, and to permit persons to whom the Software is 
  9.  * furnished to do so, subject to the following conditions: 
  10.  * 
  11.  * The above copyright notice and this permission notice shall be included in 
  12.  * all copies or substantial portions of the Software. 
  13.  * 
  14.  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 
  15.  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 
  16.  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 
  17.  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 
  18.  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, 
  19.  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN 
  20.  * THE SOFTWARE. 
  21.  */  
  22.   
  23. /** 
  24.  * @file 
  25.  * libavformat AVIOContext API example. 
  26.  * 
  27.  * Make libavformat demuxer access media content through a custom 
  28.  * AVIOContext read callback. 
  29.  * @example avio_reading.c 
  30.  */  
  31.   
  32. #include <libavcodec/avcodec.h>  
  33. #include <libavformat/avformat.h>  
  34. #include <libavformat/avio.h>  
  35. #include <libavutil/file.h>  
  36.   
  37. struct buffer_data {  
  38.     uint8_t *ptr; /* 文件中对应位置指针 */  
  39.     size_t size; ///< size left in the buffer /* 文件当前指针到末尾 */   
  40. };  
  41.   
  42. /* 将文件中数据拷贝到缓冲区,同时文件指针位置偏移,数据大小改变 */  
  43. static int read_packet(voidvoid *opaque, uint8_t *buf, int buf_size)  
  44. {  
  45.     struct buffer_data *bd = (struct buffer_data *)opaque;  
  46.     buf_size = FFMIN(buf_size, bd->size);  
  47.   
  48.     printf("ptr:%p size:%zu\n", bd->ptr, bd->size);  
  49.   
  50.     /* copy internal buffer data to buf */  
  51.     memcpy(buf, bd->ptr, buf_size);  
  52.     bd->ptr  += buf_size;  
  53.     bd->size -= buf_size;  
  54.   
  55.     return buf_size;  
  56. }  
  57.   
  58. int main(int argc, charchar *argv[])  
  59. {  
  60.     AVFormatContext *fmt_ctx = NULL;  
  61.     AVIOContext *avio_ctx = NULL;  
  62.     uint8_t *buffer = NULL, *avio_ctx_buffer = NULL;  
  63.     size_t buffer_size, avio_ctx_buffer_size = 4096;  
  64.     charchar *input_filename = NULL;  
  65.     int ret = 0;  
  66.     struct buffer_data bd = { 0 };  
  67.   
  68.     if (argc != 2) {  
  69.         fprintf(stderr, "usage: %s input_file\n"  
  70.                 "API example program to show how to read from a custom buffer "  
  71.                 "accessed through AVIOContext.\n", argv[0]);  
  72.         return 1;  
  73.     }  
  74.     input_filename = argv[1];  
  75.   
  76.     /* register codecs and formats and other lavf/lavc components*/  
  77.     av_register_all();  
  78.   
  79.     /* slurp file content into buffer */  
  80.     /* input_filename : 输入文件的文件名 
  81.      * buffer : 文件开始地址 
  82.      * buffer_size : 文件大小 
  83.      * 类似于UNIX下的mmap函数所实现的功能,返回文件开始指针,文件大小 
  84.      * 经测试,并不耗内存,可视为UNIX下的文件映射 
  85.     */  
  86.     ret = av_file_map(input_filename, &buffer, &buffer_size, 0, NULL);  
  87.     if (ret < 0)  
  88.         goto end;  
  89.   
  90.     /* fill opaque structure used by the AVIOContext read callback */  
  91.     /* bd 是指经过文件映射之后的文件,并不是指需要缓存区 */  
  92.     bd.ptr  = buffer;  
  93.     bd.size = buffer_size;  
  94.   
  95.     /* 初始化文件格式的结构体,就是分配内存 */  
  96.     if (!(fmt_ctx = avformat_alloc_context())) {  
  97.         ret = AVERROR(ENOMEM);  
  98.         goto end;  
  99.     }  
  100.   
  101.     /* 分配内存, 可以自己设置缓冲大小,这里设置的是4K */  
  102.     avio_ctx_buffer = av_malloc(avio_ctx_buffer_size);  
  103.     if (!avio_ctx_buffer) {  
  104.         ret = AVERROR(ENOMEM);  
  105.         goto end;  
  106.     }  
  107.   
  108.     /* avio_ctx_buffer是缓冲区, 
  109.      * avio_ctx_buffer 的初始地址赋值到 avio_ctx->buffer 
  110.      * avio_ctx_buffer_size是缓冲区大小 , 也是每次读取数据的大小 
  111.      * bd 是输入文件文件的映射文件 
  112.      * read_packet 回调函数,读取数据的功能 , 具体在什么情况下才会回调 ? 
  113.      */  
  114.     avio_ctx = avio_alloc_context(avio_ctx_buffer, avio_ctx_buffer_size,  
  115.                                   0, &bd, &read_packet, NULL, NULL);  
  116.     if (!avio_ctx) {  
  117.         ret = AVERROR(ENOMEM);  
  118.         goto end;  
  119.     }  
  120.     fmt_ctx->pb = avio_ctx;  
  121.     /* 配置初始化信息 
  122.     * read_packet 回调函数会在这里被调用, 它将输入文件的所有数据都先存入缓存中, 
  123.     * 如果后面有需要用到数据,那么它就从缓存中直接调用数据 
  124.     */  
  125.     ret = avformat_open_input(&fmt_ctx, NULL, NULL, NULL);  
  126.     if (ret < 0) {  
  127.         fprintf(stderr, "Could not open input\n");  
  128.         goto end;  
  129.     }  
  130.   
  131.     ret = avformat_find_stream_info(fmt_ctx, NULL);  
  132.     if (ret < 0) {  
  133.         fprintf(stderr, "Could not find stream information\n");  
  134.         goto end;  
  135.     }  
  136.       
  137.     /* 输出基本信息 */  
  138.     av_dump_format(fmt_ctx, 0, input_filename, 0);  
  139.   
  140. end:  
  141.     avformat_close_input(&fmt_ctx);  
  142.     /* note: the internal buffer could have changed, and be != avio_ctx_buffer */  
  143.     //应该就是av_free(&avio_ctx_buffer),但位置不对  
  144.     //两者有差异  
  145.     //printf("%p %p\n",avio_ctx_buffer,avio_ctx->buffer);  
  146.     if (avio_ctx) {  
  147.         av_freep(&avio_ctx->buffer);  
  148.         av_freep(&avio_ctx);  
  149.     }  
  150.     av_file_unmap(buffer, buffer_size);  
  151.   
  152.     if (ret < 0) {  
  153.         fprintf(stderr, "Error occurred: %s\n", av_err2str(ret));  
  154.         return 1;  
  155.     }  
  156.   
  157.     return 0;  
  158. }  

    这里定义了一个avio_ctx_buffer_size 变量表示每次从文件中读取到少数据到内存。在该例子中,FFMPEG是先把输入的音视频数据全部读取到内存,而并不是需要使用的时候才去读取。

目录
相关文章
|
2月前
|
存储 缓存 安全
Java内存模型深度解析:从理论到实践####
【10月更文挑战第21天】 本文深入探讨了Java内存模型(JMM)的核心概念与底层机制,通过剖析其设计原理、内存可见性问题及其解决方案,结合具体代码示例,帮助读者构建对JMM的全面理解。不同于传统的摘要概述,我们将直接以故事化手法引入,让读者在轻松的情境中领略JMM的精髓。 ####
43 6
|
2月前
|
存储 Java 编译器
Java内存模型(JMM)深度解析####
本文深入探讨了Java内存模型(JMM)的工作原理,旨在帮助开发者理解多线程环境下并发编程的挑战与解决方案。通过剖析JVM如何管理线程间的数据可见性、原子性和有序性问题,本文将揭示synchronized关键字背后的机制,并介绍volatile关键字和final关键字在保证变量同步与不可变性方面的作用。同时,文章还将讨论现代Java并发工具类如java.util.concurrent包中的核心组件,以及它们如何简化高效并发程序的设计。无论你是初学者还是有经验的开发者,本文都将为你提供宝贵的见解,助你在Java并发编程领域更进一步。 ####
|
3月前
|
C++
【C++】深入解析C/C++内存管理:new与delete的使用及原理(二)
【C++】深入解析C/C++内存管理:new与delete的使用及原理
|
3月前
|
编译器 C++ 开发者
【C++】深入解析C/C++内存管理:new与delete的使用及原理(三)
【C++】深入解析C/C++内存管理:new与delete的使用及原理
|
3月前
|
存储 C语言 C++
【C++】深入解析C/C++内存管理:new与delete的使用及原理(一)
【C++】深入解析C/C++内存管理:new与delete的使用及原理
|
1月前
|
存储 算法 Java
Java内存管理深度解析####
本文深入探讨了Java虚拟机(JVM)中的内存分配与垃圾回收机制,揭示了其高效管理内存的奥秘。文章首先概述了JVM内存模型,随后详细阐述了堆、栈、方法区等关键区域的作用及管理策略。在垃圾回收部分,重点介绍了标记-清除、复制算法、标记-整理等多种回收算法的工作原理及其适用场景,并通过实际案例分析了不同GC策略对应用性能的影响。对于开发者而言,理解这些原理有助于编写出更加高效、稳定的Java应用程序。 ####
|
5月前
|
安全 Java 开发者
Java 内存模型解析与实践
在Java的世界中,理解内存模型对于编写高效、线程安全的代码至关重要。本文将深入探讨Java内存模型的核心概念,并通过实例分析其对并发编程的影响,旨在为读者提供一套实用的策略和思考方式来优化多线程应用的性能与安全性。
70 0
|
3月前
|
存储 监控 算法
Java中的内存管理与垃圾回收机制解析
本文深入探讨了Java编程语言中的内存管理方式,特别是垃圾回收机制。我们将了解Java的自动内存管理是如何工作的,它如何帮助开发者避免常见的内存泄漏问题。通过分析不同垃圾回收算法(如标记-清除、复制和标记-整理)以及JVM如何选择合适的垃圾回收策略,本文旨在帮助Java开发者更好地理解和优化应用程序的性能。
|
3月前
|
存储 安全 Java
JVM锁的膨胀过程与锁内存变化解析
在Java虚拟机(JVM)中,锁机制是确保多线程环境下数据一致性和线程安全的重要手段。随着线程对共享资源的竞争程度不同,JVM中的锁会经历从低级到高级的膨胀过程,以适应不同的并发场景。本文将深入探讨JVM锁的膨胀过程,以及锁在内存中的变化。
58 1
|
4月前
|
存储 算法 Java
深入解析 Java 虚拟机:内存区域、类加载与垃圾回收机制
本文介绍了 JVM 的内存区域划分、类加载过程及垃圾回收机制。内存区域包括程序计数器、堆、栈和元数据区,每个区域存储不同类型的数据。类加载过程涉及加载、验证、准备、解析和初始化五个步骤。垃圾回收机制主要在堆内存进行,通过可达性分析识别垃圾对象,并采用标记-清除、复制和标记-整理等算法进行回收。此外,还介绍了 CMS 和 G1 等垃圾回收器的特点。
134 0
深入解析 Java 虚拟机:内存区域、类加载与垃圾回收机制

推荐镜像

更多