Android MediaCodec dequeueInputBuffer总是返回-1

6
我试图使用MediaMuxer和MediaCodec从AudioRecord对象中获取原始数据并将其保存到文件中。我启动了编解码器和muxer,将数据加载到输入缓冲区中,但却没有这样的好运气。通过调试调查,我发现问题出现在dequeueInputBuffer()的调用中。似乎前几个数据块是成功的,但最终dequeueInputBuffer()只是不断地返回-1。我是否遗漏了什么明显的东西?这似乎是发生的情况是我正在填充输入缓冲区,但它们从未被编解码器释放。以下是相关代码片段:
int numChunks = input.length / CHUNKSIZE;
mAudioEncoder.start();
    for (int chunk = 0; chunk <= numChunks; chunk++) {
        byte[] passMe = new byte[CHUNKSIZE];
        int inputBufferIndex = -1;
        Log.d("offerAudioEncoder","printing chunk #" + chunk + "of " + numChunks);
        //Copy the data into the chunk array
        if (chunk < input.length / CHUNKSIZE)
            for (int i = 0; i < CHUNKSIZE; i++)
                passMe[i] = input[chunk * CHUNKSIZE + i];
        else {
            eosReceived = true;
            for (int i = 0; chunk * CHUNKSIZE + i < input.length; i++)
                passMe[i] = input[chunk * CHUNKSIZE + i];
        }

        //Get the input buffer
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR2) {
            while(inputBufferIndex < 0)//justk keep trying.
                inputBufferIndex = mAudioEncoder.dequeueInputBuffer(100);
            inputBuffer = mAudioEncoder.getInputBuffer(inputBufferIndex);
        } else {
            //backwards compatibility.
            ByteBuffer[] inputBuffers = mAudioEncoder.getInputBuffers();
            inputBufferIndex = mAudioEncoder.dequeueInputBuffer(-1);
            if (inputBufferIndex >= 0)
                inputBuffer = inputBuffers[inputBufferIndex];
        }

        //Plop the data into the input buffer
        if (inputBuffer != null) {
            inputBuffer.clear();
            inputBuffer.put(passMe);
        }
        long presentationTimeUs = chunk * 10000000; //each encoded chunk represents one second of audio
        //this is what the frame should be labeled as
        mAudioEncoder.queueInputBuffer(inputBufferIndex, 0, passMe.length, presentationTimeUs, 0);

        //Pull the output buffer.
        int encoderStatus = -1;
        while(encoderStatus < 0) //Like, seriously, WAIT forever.
            encoderStatus = mAudioEncoder.dequeueOutputBuffer(mAudioBufferInfo, -1);//wait forever, why not?
        if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR2)
            outputBuffer = mAudioEncoder.getOutputBuffer(encoderStatus);
        else {
            ByteBuffer[] encoderOutputBuffers = mAudioEncoder.getOutputBuffers();
            outputBuffer = encoderOutputBuffers[encoderStatus];
        }

        if(encoderStatus >= 0) {
            if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.JELLY_BEAN_MR2)
                mMuxer.writeSampleData(audioTrackIndex, outputBuffer, mAudioBufferInfo);

            //Done with the output buffer, release it.
            mAudioEncoder.releaseOutputBuffer(encoderStatus, false);
        }//TODO: Add cases for what to do when the output format changes
1个回答

3

好的,我找到解决方法了。最终我放弃了分块逻辑,并且通过设置

audioFormat.setInteger(MediaFormat.KEY_MAX_INPUT_SIZE, 14000000);

来增加输入缓冲区的大小,该设置是针对传递给MediaCodec的configure方法的MediaFormat对象。

另外一个好提示:确保使用16位音频编码,并使用AudioRecord.read方法返回shorts。字节似乎会产生扭曲的音频(可能是因为AudioRecord希望在16位操作)。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接