Кодировка MediaCodec - dequeueInputBuffer возвращает INFO_TRY_AGAIN_LATER?

Я работаю над проектом, где я использую AudioRecord класс для записи аудио и MediaMuxer записать закодированные данные в выходной файл. Кажется, все работает нормально, как только я начинаю запись, но после нескольких звонков на writeRecordingAudioFileSampleData метод ниже, dequeueInputBuffer метод непрерывно возвращает INFO_TRY_AGAIN_LATER (-1),

Я пытался очищать кодировщик каждый раз, когда вызываю метод, и проблема больше не возникает. Но затем происходит сбой приложения, когда я вызываю метод stop на мультимедийном мультиплексоре (он говорит "Не удалось остановить мультиплексор"). Мне не нужно вызывать Flush MediaCodec, но без него я продолжаю получать INFO_TRY_AGAIN_LATER,

Вот метод, который запускает процесс записи. Для краткости я не включил методы, которые получают время представления, или методы, которые инициализируют экземпляр AudioRecord, MediaMuxerи MediaCodec кодировщик. Я не верю, что проблема связана с ними, но если они могут быть полезны, дайте мне знать, и я опубликую их.

    public ActionResponse startRecording()
    {
        ActionResponse response = new ActionResponse();
        try
        {
            File recordDir = new File(RECORD_TEMP_DIR);
            if(!recordDir.exists())
            {
                if(!recordDir.mkdirs())
                    throw new IOException("Could not create file");
            }

            String fileName = generateRecordingName();
            String path = RECORD_TEMP_DIR + "/" + fileName + "." + recordingOutputOptions.recordAudioFormat.name;
            currentTrack = new AudioFile();
            currentTrack.setName(fileName);
            currentTrack.setPath(path);
            currentTrack.setFormat(recordingOutputOptions.recordAudioFormat);

            this.initializeAudioRecord();
            if(audioRecord.getState() != AudioRecord.STATE_INITIALIZED)
                throw new IOException("Could not initialize audio record instance");

            this.initializeRecordingAudioEncoder();
            this.initializeRecordingMediaMuxer();

            encoder.start();

            audioRecord.startRecording();

            recordReadThread = new Thread(readAudioRecordRunnable);
            recordReadThread.start();

            recorderState = MEDIA_RECORDER_STATE.RECORDING;
            response.setStatus(ActionResponse.ResponseStatus.SUCCESS);
        }
        catch (IOException e)
        {
            response.setStatus(ActionResponse.ResponseStatus.FAILURE);
            response.setError(e);
        }

        return response;
    }

И вот исполняемый файл, где я читаю данные из экземпляра AudioRecord:

        private Runnable readAudioRecordRunnable = new Runnable()
    {
        @Override
        public void run()
        {
            ActionResponse response = new ActionResponse(ActionResponse.ResponseStatus.SUCCESS);
            File outFile = new File(currentTrack.path);
            try
            {
                if(outFile.exists())
                    outFile.delete();

                // Parent directory should have already been created before we started recording
                if(!outFile.createNewFile())
                {
                    response.setStatusMessage("Could not create file");
                    response.setStatus(ActionResponse.ResponseStatus.FAILURE);
                }
                else
                {
                    ByteBuffer buffer = ByteBuffer.allocateDirect(recordingOutputOptions.bufferSize);
                    int numReads = 0;
                    while (recorderState == MEDIA_RECORDER_STATE.RECORDING)
                    {
                        int result = audioRecord.read(buffer, recordingOutputOptions.bufferSize);

                        if (result >= 0)
                        {
                            audioAbsolutePtsUs = (System.nanoTime()) / 1000L;
                            int samplesPerBuffer = recordingOutputOptions.bufferSize / recordingOutputOptions.bytesPerSample;
                            long presentationTime = getPresentationTimeUs(audioAbsolutePtsUs,samplesPerBuffer);

                            Log.e("TAG","Writing " + result + " bytes of recorded data with presentation time: " + String.valueOf(presentationTime));
                            ActionResponse encodeResponse = writeRecordingAudioFileSampleData(buffer,false,presentationTime);
                            if(!encodeResponse.isSuccess())
                            {
                                response = encodeResponse;
                                break;
                            }

                            buffer.clear();
                            numReads++;
                            //encoder.flush();
                        }
                        else
                        {
                            response.setStatus(ActionResponse.ResponseStatus.FAILURE);
                            response.setStatusMessage("Audio record return result: " + result);
                            break;
                        }
                    }
                }

                if(response.isSuccess())
                    writeRecordingAudioFileSampleData(null,true,0);
            }
            catch (Exception e)
            {
                response.setStatus(ActionResponse.ResponseStatus.FAILURE);
                response.setStatusMessage("Error occurred while recording");
                response.setError(e);
            }
            finally
            {
                try
                {
                    encoder.stop();
                    encoder.release();
                    encoder = null;

                    muxer.stop();
                    muxer.release();
                    muxer = null;
                    isMuxerStarted = false;
                }
                catch (Exception ex)
                {
                    Log.e("Recording Error","Error stopping and cleaning up encoder and muxer",ex);
                }
            }

            if(!response.isSuccess())
            {
                stopRecording();

                outFile.delete();

                Log.e("Recording Error",response.statusMessage,response.error);
                if(recordEventHandler!=null)
                    recordEventHandler.onRecordError(response);
            }
        }
    };

И вот метод writeRecordingAudioFileSampleData:

        public ActionResponse writeRecordingAudioFileSampleData(final ByteBuffer data,boolean isEndOfStream, long presentationTime)
    {
        ActionResponse response = new ActionResponse(ActionResponse.ResponseStatus.SUCCESS);
        try
        {
            boolean doneSubmittingInput = false;
            int index;
            int numBytesSubmitted = 0;
            int numBytesEncoded = 0;
            int numRetriesDequeueOutputBuffer = 0;
            ByteBuffer[] inputBuffers = encoder.getInputBuffers();
            ByteBuffer[] outputBuffers = encoder.getOutputBuffers();

            while (true)
            {
                if (!doneSubmittingInput)
                {
                    index = encoder.dequeueInputBuffer(ENCODER_BUFFER_TIMEOUT_US);
                    if (index != MediaCodec.INFO_TRY_AGAIN_LATER)
                    {
                        Log.e("TAG","Dequeued input buffer: " + index);
                        if(isEndOfStream)
                        {
                            doneSubmittingInput = true;
                            encoder.queueInputBuffer(index, 0, 0, 0, BUFFER_FLAG_END_OF_STREAM);
                        }
                        else
                        {
                            int dataSize = data.capacity();
                            if (numBytesSubmitted >= dataSize)
                            {
                                doneSubmittingInput = true;
                            }
                            else
                            {
                                ByteBuffer buffer = inputBuffers[index];
                                buffer.clear();
                                buffer.put(data);
                                encoder.queueInputBuffer(index, 0, buffer.remaining(), presentationTime, 0);
                                Log.e("TAG","Queued input buffer: " + index);
                                numBytesSubmitted += buffer.remaining();
                            }
                        }
                    }
                    else
                    {                            response.setStatus(ActionResponse.ResponseStatus.FAILURE);
                        response.setStatusMessage("Dequeue input buffer returned: " + index);
                        break;
                    }
                }

                MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
                index = encoder.dequeueOutputBuffer(info, ENCODER_BUFFER_TIMEOUT_US);

                if (index == MediaCodec.INFO_TRY_AGAIN_LATER)
                {
                    Log.e("TAG","No output buffer available");

                    if (++numRetriesDequeueOutputBuffer > MAX_NUM_RETRIES_DEQUEUE_OUTPUT_BUFFER)
                    {
                        if(!isEndOfStream)
                        {
                            // All data encoded
                            Log.e("TAG","Max Retries Exceeded. All data has been encoded.");
                            break;
                        }
                        else
                        {
                            Log.e("TAG","Max Retries Exceeded. End of stream input has been queued.");
                            // EOS Input Queued, Continue Loop to Receive output
                        }
                    }
                }
                else if (index == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED)
                {
                    // should only happen once before receiving buffers
                    if (isMuxerStarted)
                    {
                        // should not occur
                        response.setStatus(ActionResponse.ResponseStatus.FAILURE);
                        response.setStatusMessage("Output format changed after muxer started");
                        break;
                    }

                    MediaFormat newFormat = encoder.getOutputFormat();
                    muxerTrackIndex = muxer.addTrack(newFormat);
                    muxer.start();
                    isMuxerStarted = true;
                    Log.e("TAG","Started Muxer");
                }
                else if (index == MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED)
                {
                    // should not occur
                    Log.e("TAG","Output buffers changed");
                    outputBuffers = encoder.getOutputBuffers();
                }
                else if (index < 0)
                {
                    Log.e("TAG","Unexpected index " + index + " received from dequeueOutputBuffer");
                    // Unexpected result - should not occur. Ignore it
                }
                else
                {
                    Log.e("TAG","Dequeued output buffer: " + index);
                    ByteBuffer encodedData = outputBuffers[index];
                    if ((info.flags & MediaCodec.BUFFER_FLAG_CODEC_CONFIG) != 0)
                    {
                        // The codec config data was pulled out and fed to the muxer when we got
                        // the INFO_OUTPUT_FORMAT_CHANGED status.  Ignore it.
                        Log.e("TAG","Ignoring codec config data");
                        info.size = 0;
                    }

                    if (info.size != 0)
                    {
                        if (!isMuxerStarted)
                        {
                            // should not occur
                            response.setStatus(ActionResponse.ResponseStatus.FAILURE);
                            response.setStatusMessage("Attempting to write data but muxer was not started");
                            break;
                        }


                        info.presentationTimeUs = presentationTime;
                        muxer.writeSampleData(muxerTrackIndex,encodedData,info);
                        Log.e("TAG","Wrote " + info.size + " bytes to muxer");
                        numBytesEncoded += info.size;
                    }

                    encoder.releaseOutputBuffer(index, false);
                    Log.e("TAG","Releasing output buffer: " + index);

                    if ((info.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0)
                    {
                        // EOS Submitted
                        if(!isEndOfStream)
                        {
                            // Unexpected EOS
                            response.setStatus(ActionResponse.ResponseStatus.FAILURE);
                            response.setStatusMessage("Unexpected buffer end of stream flag");
                        }
                        else
                            Log.e("TAG","Dequeued end of stream reached. All data has been encoded.");

                        break;
                    }
                }
            }
        }
        catch (Exception exc)
        {
            response.setStatus(ActionResponse.ResponseStatus.FAILURE);
            response.setStatusMessage(exc.getMessage());
            response.setError(exc);
        }
        finally
        {

        }

        return response;
    }

Наконец, вот пример вывода, который я получаю при запуске приложения, который показывает, где останавливается запись:

W/ExtendedACodec: Failed to get extension for extradata parameter
E/TAG: Writing 28288 bytes of recorded data with presentation time: 41035517230
E/TAG: Dequeued input buffer: 0
E/TAG: Queued input buffer: 0
I/MPEG4Writer: limits: 4294967295/0 bytes/us, bit rate: -1 bps and the estimated moov size 3195 bytes
D/MPEG4Writer: Audio track starting
E/TAG: Started Muxer
E/TAG: Dequeued input buffer: 1
E/TAG: Dequeued output buffer: 0
       Ignoring codec config data
E/TAG: Releasing output buffer: 0
E/TAG: Dequeued output buffer: 1
I/MPEG4Writer: setStartTimestampUs: 41035517230
               Earliest track starting time: 41035517230
E/TAG: Wrote 371 bytes to muxer
       Releasing output buffer: 1
E/TAG: Dequeued output buffer: 2
E/TAG: Wrote 372 bytes to muxer
E/TAG: Releasing output buffer: 2
E/TAG: Dequeued output buffer: 3
W/MPEG4Writer: 0-duration samples found: 1
E/TAG: Wrote 520 bytes to muxer
       Releasing output buffer: 3
E/TAG: No output buffer available
E/TAG: No output buffer available
I/chatty: uid=10286(com.bandindustries.musicjournal) Thread-4 identical 3 lines
E/TAG: No output buffer available
E/TAG: Max Retries Exceeded. All data has been encoded.
       Writing 28288 bytes of recorded data with presentation time: 41035677592
       Dequeued input buffer: 2
E/TAG: Queued input buffer: 2
E/TAG: Dequeued output buffer: 0
W/MPEG4Writer: 0-duration samples found: 1
E/TAG: Wrote 513 bytes to muxer
E/TAG: Releasing output buffer: 0
E/TAG: Dequeued input buffer: 3
E/TAG: Dequeued output buffer: 1
       Wrote 462 bytes to muxer
E/TAG: Releasing output buffer: 1
E/TAG: Dequeued output buffer: 2
       Wrote 433 bytes to muxer
E/TAG: Releasing output buffer: 2
E/TAG: No output buffer available
I/chatty: uid=10286(com.bandindustries.musicjournal) Thread-4 identical 4 lines
E/TAG: No output buffer available
       Max Retries Exceeded. All data has been encoded.
E/TAG: Writing 28288 bytes of recorded data with presentation time: 41035837955
E/TAG: Dequeued input buffer: 0
E/TAG: Queued input buffer: 0
E/TAG: Dequeued output buffer: 3
W/MPEG4Writer: 0-duration samples found: 2
E/TAG: Wrote 407 bytes to muxer
E/TAG: Releasing output buffer: 3
E/TAG: Dequeued input buffer: 2
E/TAG: Dequeued output buffer: 0
       Wrote 408 bytes to muxer
E/TAG: Releasing output buffer: 0
E/TAG: Dequeued output buffer: 1
E/TAG: Wrote 392 bytes to muxer
E/TAG: Releasing output buffer: 1
E/TAG: Dequeued output buffer: 2
E/TAG: Wrote 390 bytes to muxer
E/TAG: Releasing output buffer: 2
E/TAG: No output buffer available
I/chatty: uid=10286(com.bandindustries.musicjournal) Thread-4 identical 4 lines
E/TAG: No output buffer available
E/TAG: Max Retries Exceeded. All data has been encoded.
E/TAG: Writing 28288 bytes of recorded data with presentation time: 41035998318
E/TAG: Dequeued input buffer: 0
E/TAG: Queued input buffer: 0
E/TAG: Dequeued output buffer: 3
W/MPEG4Writer: 0-duration samples found: 3
E/TAG: Wrote 356 bytes to muxer
E/TAG: Releasing output buffer: 3
I/zygote64: Do partial code cache collection, code=27KB, data=29KB
            After code cache collection, code=27KB, data=29KB
            Increasing code cache capacity to 128KB
D/MPEG4Writer: Audio track stopping. Stop source
               Audio track source stopping
               Audio track source stopped
I/MPEG4Writer: Received total/0-length (11/0) buffers and encoded 11 frames. - Audio
               Audio track drift time: 0 us
D/MPEG4Writer: Audio track stopped. Stop source
               Audio track stopped. Stop source
               Stopping writer thread
D/MPEG4Writer: 0 chunks are written in the last batch
D/MPEG4Writer: Writer thread stopped
I/MPEG4Writer: Ajust the moov start time from 41035517230 us -> 41035517230 us
D/MPEG4Writer: Audio track stopping. Stop source
E/Recording Error: Dequeue input buffer returned: -1

Кто-нибудь когда-либо испытывал это раньше? Я впервые работаю с низкоуровневыми классами мультимедиа в Android, и я не смог найти много хороших примеров в Интернете или в документации. Я ценю любую помощь. Спасибо.

ОБНОВЛЕНИЕ: проблема, из-за которой происходит сбой приложения при остановке мультиплексора мультимедиа, из-за отправки 0 в качестве метки времени представления для последнего вызова, который является концом потока. Но я получаю пустой файл в качестве вывода по некоторым причинам.

0 ответов

Другие вопросы по тегам