FFmpeg is a very good open source audio and video codec library, its codec covers almost all formats of audio and video. However, it uses CPU to encode and decode, and its decoding ability can meet the demand on PC and other devices, but it is very awkward when decoding 720p and above video on mobile devices. The insufficient decoding speed leads to slow decoding of video frames, resulting in playback lag and fast power consumption. If you can decode video frames using a GPU on a mobile device, the efficiency will be increased many times, which requires the use of a hard decoder called MediaCodec.
FFmpeg can decode AVpacket fast enough, so we thought that if we could decode the raw compressed data of AVpacket video with MediaCodec, we could play hd video without consuming too much power. Fortunately, this approach worked perfectly when tested.
So start our MediaCodec decoding AVpacket journey. First look at the effect: all 720p video (source download wlplayer)
First, Mediacodec decoding process
1.1, the first configuration of MediaFormat, to tell MediaCodec how to decode the video, what information, the following code:
public void mediacodecInit(int mimetype, int width, int height, byte[] csd0, byte[] csd1) { if(surface ! = null) { try { wlGlSurfaceView.setCodecType(1); String mtype = getMimeType(mimetype); mediaFormat = MediaFormat.createVideoFormat(mtype, width, height); mediaFormat.setInteger(MediaFormat.KEY_WIDTH, width); mediaFormat.setInteger(MediaFormat.KEY_HEIGHT, height); mediaFormat.setLong(MediaFormat.KEY_MAX_INPUT_SIZE, width * height); mediaFormat.setByteBuffer("csd-0", ByteBuffer.wrap(csd0)); mediaFormat.setByteBuffer("csd-1", ByteBuffer.wrap(csd1)); Log.d("ywl5320", mediaFormat.toString()); mediaCodec = MediaCodec.createDecoderByType(mtype); if(surface ! = null) { mediaCodec.configure(mediaFormat, surface, null, 0); mediaCodec.start(); } } catch (Exception e) { e.printStackTrace(); } } else { if(wlOnErrorListener ! = null) { wlOnErrorListener.onError(WlStatus.WL_STATUS_SURFACE_NULL, "surface is null"); }}}Copy the code
Width and height: indicates the width and length of the video.
Csd0 and CSd1: both correspond to the Extradata field in AVCodecContext.
MediaFormat is now configured.
1.2 MediaCodec decoding AVpacket:
MediaCodec decodes video with two Buffer queues. One is an InputBuffer queue, which sends compressed video data to the MediaCodec decoder for decoding, then emptying the data and completing the loop. The other is the OutputBuffer queue, which renders the decoded MediaCodec data to the surface, then emptying the data and terminating the loop. Basically, one feeds the MediaCodec data, and one feeds the MediaCodec data to the Surface for rendering, and then loops through the process to play the video.
Once we understand the decoding process of MediaCodec, we know where to start. We take the InputBuffer of MediaCodec and add the compressed video data from AVpacket to it. QueueInputBuffer is sent to MediaCodec, so that MediaCodec has the raw data to decode.
public void mediacodecDecode(byte[] bytes, int size, int pts) { if(bytes ! = null && mediaCodec ! = null && info ! = null) { try { int inputBufferIndex = mediaCodec.dequeueInputBuffer(10000); if(inputBufferIndex >= 0) { ByteBuffer byteBuffer = mediaCodec.getInputBuffers()[inputBufferIndex]; byteBuffer.clear(); byteBuffer.put(bytes); mediaCodec.queueInputBuffer(inputBufferIndex, 0, size, pts, 0); } int index = mediaCodec.dequeueOutputBuffer(info, 10000); if (index >= 0) { ByteBuffer buffer = mediaCodec.getOutputBuffers()[index]; buffer.position(info.offset); buffer.limit(info.offset + info.size); mediaCodec.releaseOutputBuffer(index, true); } }catch (Exception e) { e.printStackTrace(); }}}Copy the code
The decoding process is unchanged, just like the official process.
C++ provides AVpacket data:
2.1 encapsulate the method to call Java:
void WlJavaCall::onDecMediacodec(int type, int size, uint8_t *packet_data, int pts) {
if(type == WL_THREAD_CHILD)
{
JNIEnv *jniEnv;
if(javaVM->AttachCurrentThread(&jniEnv, 0) != JNI_OK)
{
// LOGE("%s: AttachCurrentThread() failed", __FUNCTION__);
return;
}
jbyteArray data = jniEnv->NewByteArray(size);
jniEnv->SetByteArrayRegion(data, 0, size, (jbyte*)packet_data);
jniEnv->CallVoidMethod(jobj, jmid_dec_mediacodec, data, size, pts);
jniEnv->DeleteLocalRef(data);
javaVM->DetachCurrentThread();
}
else
{
jbyteArray data = jniEnv->NewByteArray(size);
jniEnv->SetByteArrayRegion(data, 0, size, (jbyte*)data);
jniEnv->CallVoidMethod(jobj, jmid_dec_mediacodec, data, size, pts);
jniEnv->DeleteLocalRef(data);
}
}Copy the code
There is a main thread and a child thread, but the decoding is in the child thread, so the main thread is not used.
2.2 add data header
Because the compressed data in AVpacket is very pure, such data cannot be decoded or played by MediaCodec. Therefore, it is necessary to add the corresponding data header to AVpacket, which requires the AV_bitSTREAM_filter_filter method of FFmpeg, such as:
mimType = av_bitstream_filter_init("h264_mp4toannexb"); if(mimType ! = NULL && ! isavi) { uint8_t *data; av_bitstream_filter_filter(mimType, pFormatCtx->streams[wlVideo->streamIndex]->codec, NULL, &data, &packet->size, packet->data, packet->size, 0); uint8_t *tdata = NULL; tdata = packet->data; packet->data = data; if(tdata ! = NULL) { av_free(tdata); }}Copy the code
2.3. Transfer AVpacket data to MediaCodec
wljavaCall->onDecMediacodec(WL_THREAD_CHILD, packet->size, packet->data, clock);Copy the code
2.4. Release AVpacket
Since we operate AVpacket’s data during demultiplexing, if we directly release AV_packet_free, it will report the error of releasing the address, so we just release the pointer in AVpacket separately:
av_free(packet->data);
av_free(packet->buf);
av_free(packet->side_data);
packet = NULL;Copy the code
For a complete example, see: wlPlayer
OK, that’s it: tinkering always works!