android - MediaCodec 没有任何可用的输入缓冲区

标签 android android-mediacodec audiorecord

我正在使用 MediaCodec API 将视频和音频编码为 mp4 文件。在单独的线程中编码的数据。有时在某些设备上,音频编码器会停止返回任何可用的输入缓冲区,因此 MediaMuxer 在尝试停止它时会崩溃。这是我的代码:

配置媒体编解码器:

public static final String MIME_TYPE_AUDIO = "audio/mp4a-latm";
public static final int SAMPLE_RATE = 44100;
public static final int CHANNEL_COUNT = 1;
public static final int CHANNEL_CONFIG = AudioFormat.CHANNEL_IN_MONO;
public static final int BIT_RATE_AUDIO = 128000;
public static final int SAMPLES_PER_FRAME = 1024 * 2;
public static final int FRAMES_PER_BUFFER = 24;
public static final int AUDIO_FORMAT = AudioFormat.ENCODING_PCM_16BIT;
public static final int AUDIO_SOURCE = MediaRecorder.AudioSource.MIC;
public static final int MAX_INPUT_SIZE = 16384 * 4;
public static final int MAX_SAMPLE_SIZE = 256 * 1024;

private AudioRecord audioRecord;
private ByteBuffer[] inputBuffers;
private ByteBuffer inputBuffer;
private MediaExtractor mediaExtractor;

private boolean audioSended = false;
private boolean completed = false;
private int sampleCount;
private int iBufferSize;

public AudioEncoderCore(MovieMuxer muxer) throws IOException {
    this.muxer = muxer;
    bufferInfo = new MediaCodec.BufferInfo();

    MediaFormat mediaFormat = null;

        mediaFormat = MediaFormat.createAudioFormat(MIME_TYPE_AUDIO, SAMPLE_RATE, CHANNEL_COUNT);
        mediaFormat.setInteger(MediaFormat.KEY_AAC_PROFILE, MediaCodecInfo.CodecProfileLevel.AACObjectLC);
        mediaFormat.setInteger(MediaFormat.KEY_MAX_INPUT_SIZE, MAX_INPUT_SIZE);
        mediaFormat.setInteger(MediaFormat.KEY_BIT_RATE, BIT_RATE_AUDIO);

        encoder = MediaCodec.createEncoderByType(MIME_TYPE_AUDIO);
        encoder.configure(mediaFormat, null, null, MediaCodec.CONFIGURE_FLAG_ENCODE);
        encoder.start();
        iBufferSize = SAMPLES_PER_FRAME * FRAMES_PER_BUFFER;

        // Ensure buffer is adequately sized for the AudioRecord
        // object to initialize
        int iMinBufferSize = AudioRecord.getMinBufferSize(SAMPLE_RATE, CHANNEL_CONFIG, AUDIO_FORMAT);
        if (iBufferSize < iMinBufferSize)
            iBufferSize = ((iMinBufferSize / SAMPLES_PER_FRAME) + 1) * SAMPLES_PER_FRAME * 2;

        audioRecord = new AudioRecord(AUDIO_SOURCE, SAMPLE_RATE, CHANNEL_CONFIG, AUDIO_FORMAT, iBufferSize);
        audioRecord.startRecording();

}

向编码器发送数据:

public void sendDataFromMic(boolean endOfStream) {
    if (endOfStream)
        Log.d(TAG, "sendDataFromMic end of stream");

    long audioPresentationTimeNs;

    byte[] mTempBuffer = new byte[SAMPLES_PER_FRAME];
    audioPresentationTimeNs = System.nanoTime();

    int iReadResult = audioRecord.read(mTempBuffer, 0, mTempBuffer.length);

    if (iReadResult == AudioRecord.ERROR_BAD_VALUE || iReadResult == AudioRecord.ERROR_INVALID_OPERATION || iReadResult == 0) {
        Log.e(TAG, "audio buffer read error");
    } else {
        // send current frame data to encoder
        try {
            if (inputBuffers == null)
                inputBuffers = encoder.getInputBuffers();

            //Sometimes can't get any available input buffer
            int inputBufferIndex = encoder.dequeueInputBuffer(100000);
            Log.d(TAG, "inputBufferIndex = " + inputBufferIndex);
            if (inputBufferIndex >= 0) {
                inputBuffer = inputBuffers[inputBufferIndex];
                inputBuffer.clear();
                inputBuffer.put(mTempBuffer, 0, iReadResult);

                Log.d(TAG, "sending frame to audio encoder " + iReadResult + " bytes");
                encoder.queueInputBuffer(inputBufferIndex, 0, iReadResult, audioPresentationTimeNs / 1000,
                        endOfStream ? MediaCodec.BUFFER_FLAG_END_OF_STREAM : 0);
            }


        } catch (Throwable t) {
            Log.e(TAG, "sendFrameToAudioEncoder exception");
            t.printStackTrace();
        }
    }
}

排水编码器:

    public void drainEncoder() {
    ByteBuffer[] encoderOutputBuffers = encoder.getOutputBuffers();
    while (true) {
        int encoderStatus = encoder.dequeueOutputBuffer(bufferInfo, TIMEOUT_NSECS);
        if (encoderStatus == MediaCodec.INFO_TRY_AGAIN_LATER) {
            Log.d(TAG, "no output available, spinning to await EOS");
            break;
        } else if (encoderStatus == MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED)
            encoderOutputBuffers = encoder.getOutputBuffers();
        else if (encoderStatus == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED) {
            MediaFormat newFormat = encoder.getOutputFormat();
            Log.d(TAG, "encoder format changed: " + newFormat);
            trackIndex = muxer.addTrack(newFormat);
        } else if (muxer.isMuxerStarted()) {
            ByteBuffer encodedData = encoderOutputBuffers[encoderStatus];
            if (encodedData == null)
                throw new RuntimeException("encoded buffer " + encoderStatus + " was null");

            if ((bufferInfo.flags & MediaCodec.BUFFER_FLAG_CODEC_CONFIG) != 0) {
                Log.d(TAG, "ignoring BUFFER_FLAG_CODEC_CONFIG");
                bufferInfo.size = 0;
            }

            if (bufferInfo.size != 0) {
                encodedData.position(bufferInfo.offset);
                encodedData.limit(bufferInfo.offset + bufferInfo.size);

                muxer.writeSampleData(trackIndex, encodedData, bufferInfo);

                Log.d(TAG, "sent " + bufferInfo.size + " bytes to muxer, ts=" +
                        bufferInfo.presentationTimeUs + " track index=" + trackIndex);
            }

            encoder.releaseOutputBuffer(encoderStatus, false);

            if ((bufferInfo.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0) {
                Log.d(TAG, "end of stream reached");
                completed = true;
                break;      // out of while
            }
        }
    }
}

错误在 HTC One、Galaxy S3 上稳定重现,但在华为荣耀 3C 上一切正常。

最佳答案

在调查源代码后,我找到了解决方案。当我出列输出缓冲区时,复用器可能尚未启动,在这种情况下缓冲区未释放。所以这里是 drain 编码器的工作代码:

    public void drainEncoder() {
    ByteBuffer[] encoderOutputBuffers = encoder.getOutputBuffers();
    while (true) {
        int encoderStatus = encoder.dequeueOutputBuffer(bufferInfo, TIMEOUT_NSECS);
        if (encoderStatus == MediaCodec.INFO_TRY_AGAIN_LATER) {
            Log.d(TAG, "no output available, spinning to await EOS");
            break;
        } else if (encoderStatus == MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED)
            encoderOutputBuffers = encoder.getOutputBuffers();
        else if (encoderStatus == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED) {
            MediaFormat newFormat = encoder.getOutputFormat();
            Log.d(TAG, "encoder format changed: " + newFormat);
            trackIndex = muxer.addTrack(newFormat);
        } else if (muxer.isMuxerStarted()) {
            ByteBuffer encodedData = encoderOutputBuffers[encoderStatus];
            if (encodedData == null)
                throw new RuntimeException("encoded buffer " + encoderStatus + " was null");

            if ((bufferInfo.flags & MediaCodec.BUFFER_FLAG_CODEC_CONFIG) != 0) {
                Log.d(TAG, "ignoring BUFFER_FLAG_CODEC_CONFIG");
                bufferInfo.size = 0;
            }

            if (bufferInfo.size != 0) {
                encodedData.position(bufferInfo.offset);
                encodedData.limit(bufferInfo.offset + bufferInfo.size);

                muxer.writeSampleData(trackIndex, encodedData, bufferInfo);
                Log.d(TAG, "sent " + bufferInfo.size + " bytes to muxer, ts=" +
                        bufferInfo.presentationTimeUs + " track index=" + trackIndex);
            }

            encoder.releaseOutputBuffer(encoderStatus, false);

            if ((bufferInfo.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0) {
                Log.d(TAG, "end of stream reached");
                completed = true;
                break;      // out of while
            }
        }
        else{
            //Muxer not ready, release buffer
            encoder.releaseOutputBuffer(encoderStatus, false);
            Log.d(TAG, "muxer not ready, skip data");
        }
    }
}

关于android - MediaCodec 没有任何可用的输入缓冲区,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28826065/

相关文章:

android - 以编程方式创建 LayerDrawable 对象

Android MediaPlayer 更改直播流的 URL

三星手机中的 Android AudioTrack 出现奇怪的噪音

android - MediaRecorder 捕获的音频文件在使用 Retrofit 2 发送到服务器后损坏

android - Android 上的硬件加速 H.264/HEVC 视频解码为 OpenGL FBO 或纹理

iphone - 我想将元数据设置为在 iOS 中以 .m4a 格式完成的录音

Android - 更改通知图标

android - 无法从 int 转换为 Android.Graphics.Color

android - AudioRecord 和 MediaRecorder Android 并行

java - 语音通话期间麦克风录制的音频被静音