提取mp4中的音频Pkt,以adts的方式写为aac文件

 提取mp4文件中的音频pkt,加上adts头写入到文件中(位运算的方式)

/*****************************************************************//*** \file   writeAAC01.cpp* \brief  提取出mp4文件中的音频包,加上adts_header, 写入到aac格式音频文件* * \author 13648* \date   April 2024*********************************************************************/
#define _CRT_SECURE_NO_WARNINGS
#include "myLog.h"
#include <iostream>extern "C"
{
#include <libavformat/avformat.h>
#include <libavcodec/avcodec.h>
}#pragma comment(lib, "avformat.lib")
#pragma comment(lib, "avcodec.lib")// 根据采样率查找对应的采样率下标
const int sampling_frequencies[] = {96000,  // 0x088200,  // 0x164000,  // 0x248000,  // 0x344100,  // 0x432000,  // 0x524000,  // 0x622050,  // 0x716000,  // 0x812000,  // 0x911025,  // 0xa8000   // 0xb// 0xc d e f是保留的
};int get_adts_header_buffer(char* const adts_header_buff, const int pkt_size, const int profile,const int sample_rate, const int channels)
{int sampling_frequency_index = 3;    // 默认使用48khzint adtsLen = pkt_size + 7;			 // 数据长度加7字节头长度int frequencies_size = sizeof(sampling_frequencies) / sizeof(sampling_frequencies[0]);for (int i = 0; i < frequencies_size; i++){sampling_frequency_index = i;if (sampling_frequencies[i] == sample_rate){break;}}if (sampling_frequency_index >= frequencies_size){LOG_WARNING("unsupport samplerate:%d\n", sample_rate);return -1;}// syncword:0xfff	adts frame start(同步字)adts_header_buff[0] = 0xff;adts_header_buff[1] = 0xf0;													// 12 bits// 0001// 0: MPEG-4 00(给00即可)  1: 表示无校验码		adts_header_buff[1] |= (0 << 3);    //MPEG Version:0 for MPEG-4,1 for MPEG-2  1 bitadts_header_buff[1] |= (0 << 1);    //Layer:0                                 2 bitsadts_header_buff[1] |= 1;           //protection absent:1(表示没有校验码)     1 bit// 01 0100 0 0// AAC级别:profile   2 bits  adts_header_buff[2] = (profile) << 6;// sampling frequency index:采样率索引  4bits   44100: index =  4adts_header_buff[2] |= (sampling_frequency_index & 0x0f) << 2;//private bit:0                   1bitadts_header_buff[2] |= (0 << 1);//channel configuration:channels  占3bit 此处剩高位1bitadts_header_buff[2] |= (channels & 0x04) >> 2;	// 0000 0010 & 0000 0100 (0 >> 2)// 10 0 0     0 0 00adts_header_buff[3] = (channels & 0x03) << 6;  //channel configuration:channels 低2bitsadts_header_buff[3] |= (0 << 5);               //original:0                1bitadts_header_buff[3] |= (0 << 4);               //home:0                    1bit// 变长adts_headeradts_header_buff[3] |= (0 << 3);               //copyright id bit:0        1bitadts_header_buff[3] |= (0 << 2);               //copyright id start:0      1bitadts_header_buff[3] |= ((adtsLen & 0x1800) >> 11);           //frame length:一共13bits   高2bits// 0000 0000adts_header_buff[4] = (uint8_t)((adtsLen & 0x7f8) >> 3);     //frame length:value    中间8bits// 000 11111adts_header_buff[5] = (uint8_t)((adtsLen & 0x7) << 5);       //frame length:value    低3bitsadts_header_buff[5] |= 0x1f;                                 //码率可变:0x7ff 高5bits// 1111 1100adts_header_buff[6] = 0xfc;     		 // buffer fullness:0x7ff 低6bits// 后还有 2bit 低位字节// 表示ADTS帧中有number_of_raw_data_blocks_in_frame + 1个AAC原始帧(数据块)return 0;
}int main_writeAAC01()
{const std::string in_filename = "./MediaFile/HTA_13s.mp4";AVFormatContext* ifmt_ctx = nullptr;size_t nRet = avformat_open_input(&ifmt_ctx, in_filename.c_str(), NULL, NULL);if (nRet != 0){LOG_WARNING("avformat_open_input error\n");return -1;}nRet = avformat_find_stream_info(ifmt_ctx, NULL);if (nRet < 0){LOG_WARNING("avformat_find_stream_info error\n");avformat_close_input(&ifmt_ctx);return -2;}int audio_idx = av_find_best_stream(ifmt_ctx, AVMEDIA_TYPE_AUDIO, -1, -1, NULL, 0);if (audio_idx < 0){LOG_WARNING("av_find_best_stream audio error\n");avformat_close_input(&ifmt_ctx);return -3;}// 不是AAC编码if (ifmt_ctx->streams[audio_idx]->codecpar->codec_id != AV_CODEC_ID_AAC){LOG_WARNING("this file audio not is aac\n");avformat_close_input(&ifmt_ctx);return -4;}AVStream* audio_stream = ifmt_ctx->streams[audio_idx];// 打印AAC级别LOG_INFO("audio profile: %d, FF_PROFILE_AAC_LOW:%d\n", audio_stream->codecpar->profile, FF_PROFILE_AAC_LOW);// 读取音频包写入到本地文件const std::string out_filename = "./MediaFile/out.aac";FILE* out_fp = fopen(out_filename.c_str(), "wb");if (NULL == out_fp){avformat_close_input(&ifmt_ctx);LOG_WARNING("open out_filename faild\n");return -5;}AVPacket* pkt = av_packet_alloc();while (av_read_frame(ifmt_ctx, pkt) >= 0){if (pkt->stream_index == audio_idx){char adts_header_buff[7] = { 0 };	// adts头信息nRet = get_adts_header_buffer(adts_header_buff, pkt->size, audio_stream->codecpar->profile, audio_stream->codecpar->sample_rate, audio_stream->codecpar->ch_layout.nb_channels);if (nRet >= 0){nRet = fwrite(adts_header_buff, 1, sizeof(adts_header_buff), out_fp);}nRet = fwrite(pkt->data, 1, pkt->size, out_fp);if (nRet != pkt->size){LOG_WARNING("write size != pkt->size\n");}}av_packet_unref(pkt);}av_packet_free(&pkt);fclose(out_fp);if (ifmt_ctx){avformat_close_input(&ifmt_ctx);}return 0;
}

 提取mp4文件中的音频pkt,加上adts头写入到文件中(位域的方式) 

/*****************************************************************//*** \file   writeAAC02.cpp* \brief  提取出mp4文件中的音频包,加上adts_header, 写入到aac格式音频文件(使用位域的方式)*			* \author 13648* \date   April 2024*********************************************************************/
#define _CRT_SECURE_NO_WARNINGS
#include "myLog.h"
#include <iostream>extern "C"
{
#include <libavformat/avformat.h>
#include <libavcodec/avcodec.h>
}#pragma comment(lib, "avformat.lib")
#pragma comment(lib, "avcodec.lib")// 根据采样率查找对应的采样率下标
const int sampling_frequencies[] = {96000,  // 0x088200,  // 0x164000,  // 0x248000,  // 0x344100,  // 0x432000,  // 0x524000,  // 0x622050,  // 0x716000,  // 0x812000,  // 0x911025,  // 0xa8000   // 0xb// 0xc d e f是保留的
};// adts_header
struct st_adts_header
{// adts_fixed_headerunsigned int syncword : 12;unsigned int id : 1;unsigned int layer : 2;unsigned int protection_absent : 1;unsigned int profile : 2;unsigned int sampling_frequency_index : 4;unsigned int private_bit : 1;unsigned int channel_configuration : 3;unsigned int original_copy : 1;unsigned int home : 1;// adts_var_headerunsigned int copyright_identification_bit : 1;unsigned int copyright_identification_start : 1;unsigned int aac_frame_length : 13;unsigned int adts_buffer_fullness : 11;unsigned int number_of_raw_data_blocks_in_frame : 2;
};unsigned long long copyAdtsHeaderToCharArray(st_adts_header* header)
{// 将fixed_header的各个字段按照ADTS的同步词格式排列// 同步词占用前12位,其余字段按照ADTS头部格式排列// (0000多的4bit) 1111 1111 1111 0 00 0 00 0000 0 000 0 0unsigned int fixed_part = (header->syncword << 16) |	// 1111 1111 1111(header->id << 15) |(header->layer << 13) |(header->protection_absent << 12) |(header->profile << 10) |(header->sampling_frequency_index << 6) |(header->private_bit << 5) |(header->channel_configuration << 2) |(header->original_copy << 1) |(header->home << 0);std::cout << std::hex << fixed_part << std::endl;// 将var_header的各个字段按照ADTS的格式排列// (0000多的bit) 0 0 0000000000000 00000000000 00unsigned int var_part = (header->copyright_identification_bit << 27) |(header->copyright_identification_start << 26) |(header->aac_frame_length << 13) |(header->adts_buffer_fullness << 2) |(header->number_of_raw_data_blocks_in_frame << 0);std::cout << std::hex << var_part << std::endl;unsigned long long ret = (fixed_part & 0XFFFFFFFFFFFFFFFF);	// 低32bit// 将fixed_part和var_part合并成一个64位整数unsigned long long adts_header_combined = (ret << 28) | var_part;std::cout << std::hex << adts_header_combined << std::endl;return adts_header_combined;
}static int get_adts_header_buffer(char* const adts_header_buff, const int pkt_size, const int profile,const int sample_rate, const int channels)
{int sampling_frequency_index = 3;    // 默认使用48khzint adtsLen = pkt_size + 7;			 // 数据长度加7字节头长度int frequencies_size = sizeof(sampling_frequencies) / sizeof(sampling_frequencies[0]);for (int i = 0; i < frequencies_size; i++){sampling_frequency_index = i;if (sampling_frequencies[i] == sample_rate){break;}}if (sampling_frequency_index >= frequencies_size){LOG_WARNING("unsupport samplerate:%d\n", sample_rate);return -1;}// 使用位域的方式填入adts_header数据st_adts_header adts_header;// 赋值ADTS的固定头部分adts_header.syncword = 0xFFF;adts_header.id = 0;adts_header.layer = 0;adts_header.protection_absent = 1;adts_header.profile = profile;adts_header.sampling_frequency_index = sampling_frequency_index;adts_header.private_bit = 0x00;adts_header.channel_configuration = channels;adts_header.original_copy = 0x00;adts_header.home = 0x00;// 赋值ADTS的可变头部分adts_header.copyright_identification_bit = 0x00;adts_header.copyright_identification_start = 0x00;adts_header.aac_frame_length = adtsLen;adts_header.adts_buffer_fullness = 0x7FF;adts_header.number_of_raw_data_blocks_in_frame = 0x00;// 获取一个64bit的值,然后逐个拷贝到adts_header_buff中unsigned long long value = copyAdtsHeaderToCharArray(&adts_header);// 00000000 00000000 00000000 00000000 00000000 00000000 00000000 (56 bit)for (int i = 0; i < 7; i++){adts_header_buff[i] = value >> ((7 - i - 1) * 8);}return 0;
}int main()
{const std::string in_filename = "./MediaFile/HTA_13s.mp4";AVFormatContext* ifmt_ctx = nullptr;size_t nRet = avformat_open_input(&ifmt_ctx, in_filename.c_str(), NULL, NULL);if (nRet != 0){LOG_WARNING("avformat_open_input error\n");return -1;}nRet = avformat_find_stream_info(ifmt_ctx, NULL);if (nRet < 0){LOG_WARNING("avformat_find_stream_info error\n");avformat_close_input(&ifmt_ctx);return -2;}int audio_idx = av_find_best_stream(ifmt_ctx, AVMEDIA_TYPE_AUDIO, -1, -1, NULL, 0);if (audio_idx < 0){LOG_WARNING("av_find_best_stream audio error\n");avformat_close_input(&ifmt_ctx);return -3;}// 不是AAC编码if (ifmt_ctx->streams[audio_idx]->codecpar->codec_id != AV_CODEC_ID_AAC){LOG_WARNING("this file audio not is aac\n");avformat_close_input(&ifmt_ctx);return -4;}AVStream* audio_stream = ifmt_ctx->streams[audio_idx];// 打印AAC级别LOG_INFO("audio profile: %d, FF_PROFILE_AAC_LOW:%d\n", audio_stream->codecpar->profile, FF_PROFILE_AAC_LOW);// 读取音频包写入到本地文件const std::string out_filename = "./MediaFile/out_02.aac";FILE* out_fp = fopen(out_filename.c_str(), "wb");if (NULL == out_fp){avformat_close_input(&ifmt_ctx);LOG_WARNING("open out_filename faild\n");return -5;}AVPacket* pkt = av_packet_alloc();while (av_read_frame(ifmt_ctx, pkt) >= 0){if (pkt->stream_index == audio_idx){char adts_header_buff[7] = { 0 };	// adts头信息nRet = get_adts_header_buffer(adts_header_buff, pkt->size, audio_stream->codecpar->profile,audio_stream->codecpar->sample_rate, audio_stream->codecpar->ch_layout.nb_channels);if (nRet >= 0){nRet = fwrite(adts_header_buff, 1, sizeof(adts_header_buff), out_fp);}nRet = fwrite(pkt->data, 1, pkt->size, out_fp);if (nRet != pkt->size){LOG_WARNING("write size != pkt->size\n");}}av_packet_unref(pkt);}av_packet_free(&pkt);fclose(out_fp);if (ifmt_ctx){avformat_close_input(&ifmt_ctx);}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/799014.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【单源最短路 图论】882. 细分图中的可到达节点

作者推荐 视频算法专题 本文涉及知识点 单源最短路 图论 LeetCode 882. 细分图中的可到达节点 给你一个无向图&#xff08;原始图&#xff09;&#xff0c;图中有 n 个节点&#xff0c;编号从 0 到 n - 1 。你决定将图中的每条边 细分 为一条节点链&#xff0c;每条边之间…

Spring Security——11,自定义权限校验方法

自定义权限校验方法 一键三连有没有捏~~ 我们也可以定义自己的权限校验方法&#xff0c;在PreAuthorize注解中使用我们的方法。 自定义一个权限检验方法&#xff1a; 在SPEL表达式中使用 ex相当于获取容器中bean的名字未ex的对象。然后再调用这个对象的 hasAuthority方法&am…

hadoop分布式计算组件

什么是计算、分布式计算&#xff1f; 计算&#xff1a;对数据进行处理&#xff0c;使用统计分析等手段得到需要的结果 分布式计算&#xff1a;多台服务器协同工作&#xff0c;共同完成一个计算任务 分布式计算常见的2种工作模式 分散->汇总(MapReduce就是这种模式)中心调…

【Linux系列】如何确定当前运行的是 RHEL 9 还是 RHEL 8?

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

物联网可视化平台

随着数字化转型的深入&#xff0c;物联网技术正在成为企业实现智能化、高效化运营的重要工具。物联网可视化平台&#xff0c;作为连接物理世界与数字世界的桥梁&#xff0c;为企业提供了直观、实时的数据展示和监控能力&#xff0c;从而在数字化转型中扮演着关键角色。 一、物…

抖音-引流私域转化模式1.0现场视频,从抖音源源不断把人加到私域,

抖音-引流私域转化模式1.0现场视频&#xff0c;从抖音源源不断把人加到私域&#xff0c;让加到私域的粉丝买单 抖音-引流私域转化模式1.0现场视频&#xff0c;从抖音源源不断把人加到私域 - 百创网-源码交易平台_网站源码_商城源码_小程序源码 课程内容&#xff1a; 01.第一…

后端nginx使用set_real_ip_from获取用户真实IP

随着nginx的迅速崛起&#xff0c;越来越多公司将apache更换成nginx. 同时也越来越多人使用nginx作为负载均衡, 并且代理前面可能还加上了CDN加速&#xff0c;但是随之也遇到一个问题&#xff1a;nginx如何获取用户的真实IP地址. 前言&#xff1a;Nginx ngx_http_realip_module…

深入理解计算机系统 家庭作业 2.96

题目出的很不好,感觉没有标准. #include <stdio.h>typedef unsigned float_bits;int float_f2i(float_bits f) {unsigned sign f >> (31);unsigned exp (f >> 23) & 0xff;unsigned frac f & 0x7fffff;unsigned add (frac & 0x3) 0x3;unsig…

Pytorch张量的数学运算:向量基础运算

文章目录 一、简单运算二、广播运算1.广播的基本规则2.广播操作的例子 三、运算函数 参考&#xff1a;与凤行 张量的数学运算是深度学习和科学计算中的基础。张量可以被视为一个多维数组&#xff0c;其在数学和物理学中有广泛的应用。这些运算包括但不限于加法、减法、乘法、除…

TLS是什么,为什么TLS1.0不安全

什么是TLS TLS&#xff0c;全名为 Transport Layer Security&#xff0c;即传输层安全协议。它是一种加密协议&#xff0c;用于在互联网上提供安全和数据完整性保证。TLS 的前身是 SSL&#xff08;Secure Sockets Layer&#xff0c;安全套接字层协议&#xff09;&#xff0c;由…

提升网络传输性能的误区

客机坠毁始终是种灾难&#xff0c;理论上至少存在 100 种不让客机坠毁的方案&#xff0c;其中之一是携带一个巨大的降落伞(合计展开面积 6000 平米)&#xff0c;该方案涉及包括材料学&#xff0c;力学等&#xff0c;用最轻的材料做展开面积最大的降落伞&#xff0c;当客机要坠毁…

Vscode连接WSL2当中的jupyter

主要解决办法参考自这篇博客 1. 在WSL当中安装jupyter 这个随便找一篇博客即可&#xff0c;比如这篇&#xff0c;也可以根据现有的环境参考其它博客内容 2. 使用jupyter创建一个虚拟环境 首先激活想要添加的虚拟环境后&#xff0c;输入命令安装库: pip install ipykernel …

企业级开源路由系统VyOS-构建和使用

介绍 VyOS是一个基于Linux的企业级路由器操作系统&#xff0c;被许多公司和个人用来驱动物理网络设备&#xff0c;如路由器和防火墙。它有一个统一的命令行界面来管理其所有的网络相关功能&#xff08;和Juniper Junos操作很像&#xff09;。VyOS使用Debian GNU/Linux作为其基…

每天学习一个Linux命令之nc

每天学习一个Linux命令之nc 简介 nc&#xff08;netcat&#xff09;是一种用于TCP/IP和UDP协议的通信工具。在Linux系统下&#xff0c;nc命令提供了丰富的选项&#xff0c;使其具备了很多强大的功能。本文将详细介绍nc命令的各种选项及其使用方法。 命令选项 以下是nc命令的…

HTTP 常见的状态码以及其适用场景

是什么 HTTP状态码&#xff08;英语&#xff1a;HTTP Status Code&#xff09;&#xff0c;用以表示网页服务器超文本传输协议响应状态的3位数字代码 它由 RFC 2616规范定义的&#xff0c;并得到 RFC 2518、RFC 2817、RFC 2295、RFC 2774与 RFC 4918等规范扩展 简单来讲&#…

文件输入/输出流(I/O)

文章目录 前言一、文件输入\输出流是什么&#xff1f;二、使用方法 1.FileInputStream与FileOutputStream类2.FileReader与FileWriter类总结 前言 对于文章I/O(输入/输出流的概述)&#xff0c;有了下文。这篇文章将具体详细展述如何向磁盘文件中输入数据&#xff0c;或者读取磁…

全面解析十七种数据分析方法,具象数据分析思维

一、介绍 在当今数据驱动的商业环境中&#xff0c;数据分析已经成为了企业获取竞争优势的关键工具。无论是为了优化运营效率&#xff0c;提高客户满意度&#xff0c;还是推动产品创新&#xff0c;企业都需要通过分析大量数据来做出明智的决策。数据分析方法多种多样&#xff0c…

【云开发笔记No.20】中台架构的分类

中台现在成了一个到处都在说的词了&#xff0c;甚至在组织架构中&#xff0c;弄几个万金油&#xff0c;也说有了一个中台支撑部门。一方面是滥用&#xff0c;另一个方面&#xff0c;也说明确实有它的作用和意义。 在云计算和数字化转型日益盛行的今天&#xff0c;中台架构已成…

软件无线电系列——CIC滤波器

本节目录 一、CIC滤波器 1、积分器 2、梳状滤波器 3、CIC滤波器的特性 二、单级CIC和多级CIC滤波器本节内容 一、CIC滤波器 CIC滤波器&#xff0c;英文名称为Cascade Integrator Comb&#xff0c;中文全称为级联积分梳状滤波器&#xff0c;从字面来看就是将积分器与梳状滤波器…

《云原生安全攻防》-- 云原生应用风险分析

为了满足每位朋友的学习需求&#xff0c;并且支持课程的持续更新&#xff0c;本系列课程提供了免费版和付费视频版两种方式来提供课程内容。我们会持续更新课程内容&#xff0c;以确保内容的度和实用性。 在本节课程中&#xff0c;我们将一起探讨云原生应用在新的架构模式下可能…