MediaCodec converted file from WAV to AMR does not

2019-03-29 10:20发布

问题:

I am using MediaCodec to convert a .wav file to .amr. I have used following code for input and getting encoded buffer. I get the encoded file but it does not play, my input is proper as i am able to play that file in Audacity. I am using EncodeDecodeTest.java from android API test. Any pointers to potential problem are appreciated.

 MediaCodec codec = MediaCodec.createByCodecName(componentName);

    try {
        codec.configure(
                format,
                null /* surface */,
                null /* crypto */,
                MediaCodec.CONFIGURE_FLAG_ENCODE);
    } catch (IllegalStateException e) {
        AppLog.logString("codec '" + componentName + "' failed configuration.");

       // assertTrue("codec '" + componentName + "' failed configuration.", false);
    }

    codec.start();
    ByteBuffer[] codecInputBuffers = codec.getInputBuffers();
    ByteBuffer[] codecOutputBuffers = codec.getOutputBuffers();

    int numBytesSubmitted = 0;
    boolean doneSubmittingInput = false;
    int numBytesDequeued = 0;

    while (true) {
        int index =0 ;

        if (!doneSubmittingInput) {
            index = codec.dequeueInputBuffer(kTimeoutUs /* timeoutUs */);

            if (index != MediaCodec.INFO_TRY_AGAIN_LATER) {
                if (numBytesSubmitted >= filesize ){
                    codec.queueInputBuffer(
                            index,
                            0 /* offset */,
                            0 /* size */,
                            0 /* timeUs */,
                            MediaCodec.BUFFER_FLAG_END_OF_STREAM);

                    if (VERBOSE) {
                        AppLog.logString("queued input EOS.");
                    }

                    doneSubmittingInput = true;
                } else {
                    int inputsize = 0;
                    int size = 0;
                    if ((filesize - numBytesSubmitted) > codecInputBuffers[index]
                                .limit()) {
                            inputsize = codecInputBuffers[index].limit();
                        } else
                            inputsize = (int) (filesize - numBytesSubmitted);
                            size = queueInputBuffer(codec,
                                codecInputBuffers, index, in, inputsize);
                    numBytesSubmitted += size;
                    /*if (VERBOSE) {
                        AppLog.logString("queued " + size + " bytes of input data.");
                    }*/
                }
            }else
                 AppLog.logString("MediaCodec.INFO_TRY_AGAIN_LATER.");
        }

        MediaCodec.BufferInfo info = new MediaCodec.BufferInfo();
        index = codec.dequeueOutputBuffer(info, kTimeoutUs /* timeoutUs */);

        if (index == MediaCodec.INFO_TRY_AGAIN_LATER) {
        } else if (index == MediaCodec.INFO_OUTPUT_FORMAT_CHANGED) {
            AppLog.logString("INFO_OUTPUT_FORMAT_CHANGED");
            MediaFormat fmt = codec.getOutputFormat();
            AppLog.logString(fmt.getString(MediaFormat.KEY_MIME));
        } else if (index == MediaCodec.INFO_OUTPUT_BUFFERS_CHANGED) {
            codecOutputBuffers = codec.getOutputBuffers();
        } else {
            dequeueOutputBuffer(codec, codecOutputBuffers, index, info, fos);

            numBytesDequeued += info.size;

            if ((info.flags & MediaCodec.BUFFER_FLAG_END_OF_STREAM) != 0) {
                if (VERBOSE) {
                    AppLog.logString("dequeued output EOS.");
                }
                break;
            }

            if (VERBOSE) {
                AppLog.logString("dequeued " + info.size + " bytes of output data.");
            }
        }
    }

    if (VERBOSE) {
        AppLog.logString("queued a total of " + numBytesSubmitted + "bytes, "
                + "dequeued " + numBytesDequeued + " bytes.");
    }

    codec.release();
    codec = null;

queueInputBuffer and dequeueOutputBuffer are implemented as follows:

private int queueInputBuffer(
        MediaCodec codec, ByteBuffer[] inputBuffers, int index, FileInputStream in, int size) {
        ByteBuffer buffer = inputBuffers[index];
        buffer.clear();

       // int size = buffer.limit();
        AppLog.logString("Size is :" + size);
        byte[] data = new byte[size];
        try {
            in.read(data,0,size);
        } catch (IOException e) {
            e.printStackTrace();
        }
        buffer.put(data);
        buffer.flip();
        codec.queueInputBuffer(index, 0 /* offset */, size, 0 /* timeUs */, MediaCodec.BUFFER_FLAG_SYNC_FRAME);

        return size;
}

private void dequeueOutputBuffer(
        MediaCodec codec, ByteBuffer[] outputBuffers,
        int index, MediaCodec.BufferInfo info, FileOutputStream fos) {
    byte[] data= new byte[info.size];
    ByteBuffer out = outputBuffers[index];
    out.get(data);
    try {
        fos.write(data);
    } catch (IOException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    codec.releaseOutputBuffer(index, false /* render */);
}