MediaCodec编译pcm为aac

2024-02-05 10:18
文章标签 编译 pcm aac mediacodec

本文主要是介绍MediaCodec编译pcm为aac,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

http://blog.csdn.net/tn0521/article/details/44980183


 

MediaCodec编译pcm为aac

标签: mediacodecaac编码adtsaudioandroid音频编码
  1514人阅读  评论(5)  收藏  举报
  分类:
 

先说说写这份博客的原因吧:

1.自己的第一篇博客,一味的拿来主义者太过于不厚道

2.记录下来也为自己以后使用,利人利己

3.代码都是亲自测试过


另外说一下需要改动的地方,

1,package是肯定需要改动的,你不可能跟我定义的一样吧

2,mediaFormat.setInteger(MediaFormat.KEY_MAX_INPUT_SIZE, 8192);这个参数是pcm buffer大小,可以根据自己的作适当修改,

3,如果还有不明白的地方可以参考以下链接

http://m.blog.csdn.NET/blog/cc_xueqin/41862131

里面讲解十分清楚,很好的文章


废话少说,代码献上

package com.sabine.codec;



import android.media.MediaCodec;
import android.media.MediaCodecInfo;
import android.media.MediaFormat;
import android.os.Environment;
import android.util.Log;


import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;


public class AudioEncoder {


private MediaCodec mediaCodec;
private BufferedOutputStream outputStream;
private String mediaType = "OMX.google.aac.encoder";


ByteBuffer[] inputBuffers = null;
ByteBuffer[] outputBuffers = null;


// "OMX.qcom.audio.decoder.aac";
// "audio/mp4a-latm";


public AudioEncoder() {
File f = new File(Environment.getExternalStorageDirectory(),
"sabine/audio_encoded.aac");
touch(f);
try {
outputStream = new BufferedOutputStream(new FileOutputStream(f));
Log.e("AudioEncoder", "outputStream initialized");
} catch (Exception e) {
e.printStackTrace();
}


// mediaCodec = MediaCodec.createEncoderByType("audio/mp4a-latm");
mediaCodec = MediaCodec.createByCodecName(mediaType);
final int kSampleRates[] = { 8000, 11025, 22050, 44100, 48000 };
final int kBitRates[] = { 64000,96000,128000 };
MediaFormat mediaFormat = MediaFormat.createAudioFormat(
"audio/mp4a-latm", kSampleRates[3], 2);
mediaFormat.setInteger(MediaFormat.KEY_AAC_PROFILE,
MediaCodecInfo.CodecProfileLevel.AACObjectLC);
mediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, kBitRates[1]);
mediaFormat.setInteger(MediaFormat.KEY_MAX_INPUT_SIZE, 8192);// It will
// increase
// capacity
// of
// inputBuffers
mediaCodec.configure(mediaFormat, null, null,
MediaCodec.CONFIGURE_FLAG_ENCODE);
mediaCodec.start();


inputBuffers = mediaCodec.getInputBuffers();
outputBuffers = mediaCodec.getOutputBuffers();
}


public void close() {
try {
mediaCodec.stop();
mediaCodec.release();
outputStream.flush();
outputStream.close();
} catch (Exception e) {
e.printStackTrace();
}
}


// called AudioRecord's read
public synchronized void offerEncoder(byte[] input) {
Log.e("AudioEncoder", input.length + " is coming");


int inputBufferIndex = mediaCodec.dequeueInputBuffer(-1);
if (inputBufferIndex >= 0) {
ByteBuffer inputBuffer = inputBuffers[inputBufferIndex];
inputBuffer.clear();


inputBuffer.put(input);


mediaCodec
.queueInputBuffer(inputBufferIndex, 0, input.length, 0, 0);
}


MediaCodec.BufferInfo bufferInfo = new MediaCodec.BufferInfo();
int outputBufferIndex = mediaCodec.dequeueOutputBuffer(bufferInfo, 0);


// //trying to add a ADTS
while (outputBufferIndex >= 0) {
int outBitsSize = bufferInfo.size;
int outPacketSize = outBitsSize + 7; // 7 is ADTS size
ByteBuffer outputBuffer = outputBuffers[outputBufferIndex];


outputBuffer.position(bufferInfo.offset);
outputBuffer.limit(bufferInfo.offset + outBitsSize);


byte[] outData = new byte[outPacketSize];
addADTStoPacket(outData, outPacketSize);


outputBuffer.get(outData, 7, outBitsSize);
outputBuffer.position(bufferInfo.offset);


// byte[] outData = new byte[bufferInfo.size];
try {
outputStream.write(outData, 0, outData.length);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
Log.e("AudioEncoder", outData.length + " bytes written");


mediaCodec.releaseOutputBuffer(outputBufferIndex, false);
outputBufferIndex = mediaCodec.dequeueOutputBuffer(bufferInfo, 0);


}


// Without ADTS header
/*
* while (outputBufferIndex >= 0) { ByteBuffer outputBuffer =
* outputBuffers[outputBufferIndex]; byte[] outData = new
* byte[bufferInfo.size];

* outputBuffer.get(outData); try { outputStream.write(outData, 0,
* outData.length); } catch (IOException e) { // TODO Auto-generated
* catch block e.printStackTrace(); } Log.e("AudioEncoder",
* outData.length + " bytes written");

* mediaCodec.releaseOutputBuffer(outputBufferIndex, false);
* outputBufferIndex = mediaCodec.dequeueOutputBuffer(bufferInfo, 0);

* }
*/
}


/**
* Add ADTS header at the beginning of each and every AAC packet. This is
* needed as MediaCodec encoder generates a packet of raw AAC data.

* Note the packetLen must count in the ADTS header itself.
**/
public void addADTStoPacket(byte[] packet, int packetLen) {
int profile = 2; // AAC LC
// 39=MediaCodecInfo.CodecProfileLevel.AACObjectELD;
int freqIdx = 4; // 44.1KHz
int chanCfg = 2; // CPE


// fill in ADTS data
packet[0] = (byte) 0xFF;
packet[1] = (byte) 0xF9;
packet[2] = (byte) (((profile - 1) << 6) + (freqIdx << 2) + (chanCfg >> 2));
packet[3] = (byte) (((chanCfg & 3) << 6) + (packetLen >> 11));
packet[4] = (byte) ((packetLen & 0x7FF) >> 3);
packet[5] = (byte) (((packetLen & 7) << 5) + 0x1F);
packet[6] = (byte) 0xFC;
}


public void touch(File f) {
try {
if (!f.exists())
f.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
}

}


这篇关于MediaCodec编译pcm为aac的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

解决IDEA使用springBoot创建项目,lombok标注实体类后编译无报错,但是运行时报错问题

《解决IDEA使用springBoot创建项目,lombok标注实体类后编译无报错,但是运行时报错问题》文章详细描述了在使用lombok的@Data注解标注实体类时遇到编译无误但运行时报错的问题,分析... 目录问题分析问题解决方案步骤一步骤二步骤三总结问题使用lombok注解@Data标注实体类,编译时

maven 编译构建可以执行的jar包

💝💝💝欢迎莅临我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:「stormsha的主页」👈,「stormsha的知识库」👈持续学习,不断总结,共同进步,为了踏实,做好当下事儿~ 专栏导航 Python系列: Python面试题合集,剑指大厂Git系列: Git操作技巧GO

Windows环境利用VS2022编译 libvpx 源码教程

libvpx libvpx 是一个开源的视频编码库,由 WebM 项目开发和维护,专门用于 VP8 和 VP9 视频编码格式的编解码处理。它支持高质量的视频压缩,广泛应用于视频会议、在线教育、视频直播服务等多种场景中。libvpx 的特点包括跨平台兼容性、硬件加速支持以及灵活的接口设计,使其可以轻松集成到各种应用程序中。 libvpx 的安装和配置过程相对简单,用户可以从官方网站下载源代码

Golang test编译使用

创建文件my_test.go package testsimport "testing"func TestMy(t *testing.T) {t.Log("TestMy")} 通常用法: $ go test -v -run TestMy my_test.go=== RUN TestMyTestMy: my_test.go:6: TestMy--- PASS: TestMy (0.

C++/《C/C++程序编译流程》

程序的基本流程如图:   1.预处理        预处理相当于根据预处理指令组装新的C/C++程序。经过预处理,会产生一个没有宏定义,没有条件编译指令,没有特殊符号的输出文件,这个文件的含义同原本的文件无异,只是内容上有所不同。 读取C/C++源程序,对其中的伪指令(以#开头的指令)进行处理将所有的“#define”删除,并且展开所有的宏定义处理所有的条件编译指令,如:“#if”、“

编译linux内核出现 arm-eabi-gcc: error: : No such file or directory

external/e2fsprogs/lib/ext2fs/tdb.c:673:29: warning: comparison between : In function 'max2165_set_params': -。。。。。。。。。。。。。。。。。。 。。。。。。。。。。。。。 。。。。。。。。 host asm: libdvm <= dalvik/vm/mterp/out/Inte

QT 编译报错:C3861: ‘tr‘ identifier not found

问题: QT 编译报错:C3861: ‘tr’ identifier not found 原因 使用tr的地方所在的类没有继承自 QObject 类 或者在不在某一类中, 解决方案 就直接用类名引用 :QObject::tr( )

hector_quadrotor编译总结 | ubuntu 16.04 ros-kinetic版本

hector_quadrotor编译总结 | ubuntu 16.04 ros-kinetic版本 基于Ubuntu 16.04 LTS系统所用ROS版本为 Kinetic hector_quadrotor ROS包主要用于四旋翼无人机的建模、控制和仿真。 1.安装依赖库 所需系统及依赖库 Ubuntu 16.04|ros-kinetic|Gazebo|gazebo_ros_pkgs|ge

hector_quadrotor编译总结 | ubuntu 14.04 ros-indigo版本

hector_quadrotor编译总结 | ubuntu 14.04 ros-indigo版本 基于Ubuntu 14.04 LTS系统所用ROS版本为 Indigo hector_quadrotor ROS包主要用于四旋翼无人机的建模、控制和仿真。 备注:两种安装方式可选:install the binary packages | install the source files