3

我尝试为“audio/3gpp”和我的应用程序崩溃创建编码器。

我配置 MediaCodec...

    String mMime = "audio/3gpp";
    mMediaCodec = MediaCodec.createEncoderByType(mMime);
    MediaFormat mMediaFormat = MediaFormat.createAudioFormat(mMime, 44100, 1);
    mMediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, 12000);
    mMediaFormat.setInteger(MediaFormat.KEY_SAMPLE_RATE, 44100);
    mMediaFormat.setInteger(MediaFormat.KEY_CHANNEL_COUNT, 1);
    mMediaCodec.configure(mMediaFormat, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE);
    mMediaCodec.start();

然后我尝试对数据进行编码......

private byte[] EncodeDataTo3gp(byte[] input)
{
    byte[] outData = null;

    try 
    {
        ByteBuffer[] inputBuffers = mMediaCodec.getInputBuffers();

        ByteBuffer[] outputBuffers = mMediaCodec.getOutputBuffers();
        int inputBufferIndex = mMediaCodec.dequeueInputBuffer(-1);
        if (inputBufferIndex >= 0) 
        {
            ByteBuffer inputBuffer = inputBuffers[inputBufferIndex];
            inputBuffer.clear();
            inputBuffer.put(input);
            mMediaCodec.queueInputBuffer(inputBufferIndex, 0, input.length, 0, 0);
        }

        MediaCodec.BufferInfo bufferInfo = new MediaCodec.BufferInfo();
        int outputBufferIndex = mMediaCodec.dequeueOutputBuffer(bufferInfo, 0);
        while (outputBufferIndex >= 0) 
        {
            ByteBuffer outputBuffer = outputBuffers[outputBufferIndex];
            outData = new byte[bufferInfo.size];
            outputBuffer.get(outData);

            mMediaCodec.releaseOutputBuffer(outputBufferIndex, false);
            outputBufferIndex = mMediaCodec.dequeueOutputBuffer(bufferInfo, 0);
        }
    } catch (Throwable t) {
        t.printStackTrace();
    }

    return outData;
}

但是这个函数抛出了一个异常。什么时候"inputBuffer.put(input);": java.nio.BufferOverflowException

4

3 回答 3

4

尝试根据大小添加MediaFormat.KEY_MAX_INPUT_SIZE参数。它将增加容量。mMediaFormatinputinputBuffers

于 2013-07-17T12:36:49.147 回答
2

输入缓冲区处理看起来没问题——你clear()用来重置位置和限制——所以我假设你提供的输入只是比缓冲区可以容纳的大。如果input.length大于inputBuffer.limit(),则需要以较小的块提供它。

输出缓冲区管理看起来有点奇怪——你outData在每次循环迭代时重新分配,所以如果你得到多个缓冲区,你最终会丢弃除最后一个之外的所有缓冲区。

对于输入和输出,您都没有处理负返回值。特别是INFO_*_BUFFERS_CHANGED需要您重新获取ByteBuffer阵列。

于 2013-07-10T00:23:01.933 回答
1

放在inputBuffer.rewind();前面inputBuffer.put(input);试试

于 2013-07-05T14:00:12.090 回答