Commit 7ab1842e authored by Thomas Guillem's avatar Thomas Guillem Committed by Jean-Baptiste Kempf

mediacodec: process input buffers in only one place

(behavior not changed)
Signed-off-by: default avatarJean-Baptiste Kempf <jb@videolan.org>
parent 039c69d6
...@@ -781,6 +781,58 @@ static int InsertInflightPicture(decoder_t *p_dec, picture_t *p_pic, ...@@ -781,6 +781,58 @@ static int InsertInflightPicture(decoder_t *p_dec, picture_t *p_pic,
return 0; return 0;
} }
static void PutInput(decoder_t *p_dec, JNIEnv *env, block_t **pp_block, jlong timeout)
{
decoder_sys_t *p_sys = p_dec->p_sys;
block_t *p_block = *pp_block;
int index;
jobject buf;
jsize size;
uint8_t *bufptr;
struct H264ConvertState convert_state = { 0, 0 };
index = (*env)->CallIntMethod(env, p_sys->codec,
p_sys->dequeue_input_buffer, timeout);
if (CHECK_EXCEPTION()) {
msg_Err(p_dec, "Exception occurred in MediaCodec.dequeueInputBuffer");
p_sys->error_state = true;
return;
}
if (index < 0)
return;
if (p_sys->get_input_buffers)
buf = (*env)->GetObjectArrayElement(env, p_sys->input_buffers, index);
else
buf = (*env)->CallObjectMethod(env, p_sys->codec, p_sys->get_input_buffer, index);
size = (*env)->GetDirectBufferCapacity(env, buf);
bufptr = (*env)->GetDirectBufferAddress(env, buf);
if (size < 0) {
msg_Err(p_dec, "Java buffer has invalid size");
p_sys->error_state = true;
return;
}
if ((size_t) size > p_block->i_buffer)
size = p_block->i_buffer;
memcpy(bufptr, p_block->p_buffer, size);
convert_h264_to_annexb(bufptr, size, p_sys->nal_size, &convert_state);
int64_t ts = p_block->i_pts;
if (!ts && p_block->i_dts)
ts = p_block->i_dts;
timestamp_FifoPut(p_sys->timestamp_fifo, p_block->i_pts ? VLC_TS_INVALID : p_block->i_dts);
(*env)->CallVoidMethod(env, p_sys->codec, p_sys->queue_input_buffer, index, 0, size, ts, 0);
(*env)->DeleteLocalRef(env, buf);
if (CHECK_EXCEPTION()) {
msg_Err(p_dec, "Exception in MediaCodec.queueInputBuffer");
return;
}
block_Release(p_block);
*pp_block = NULL;
p_sys->decoded = true;
}
static void GetOutput(decoder_t *p_dec, JNIEnv *env, picture_t **pp_pic, jlong timeout) static void GetOutput(decoder_t *p_dec, JNIEnv *env, picture_t **pp_pic, jlong timeout)
{ {
decoder_sys_t *p_sys = p_dec->p_sys; decoder_sys_t *p_sys = p_dec->p_sys;
...@@ -981,27 +1033,20 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block) ...@@ -981,27 +1033,20 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block)
decoder_sys_t *p_sys = p_dec->p_sys; decoder_sys_t *p_sys = p_dec->p_sys;
picture_t *p_pic = NULL; picture_t *p_pic = NULL;
JNIEnv *env = NULL; JNIEnv *env = NULL;
struct H264ConvertState convert_state = { 0, 0 };
if (!pp_block || !*pp_block) if (!pp_block || !*pp_block)
return NULL; return NULL;
block_t *p_block = *pp_block; if (p_sys->error_state)
goto endclean;
if (p_sys->error_state) {
block_Release(p_block);
if (!p_sys->error_event_sent) {
/* Signal the error to the Java. */
jni_EventHardwareAccelerationError();
p_sys->error_event_sent = true;
}
return NULL;
}
jni_attach_thread(&env, THREAD_NAME); jni_attach_thread(&env, THREAD_NAME);
if (!env)
goto endclean;
if (p_block->i_flags & (BLOCK_FLAG_DISCONTINUITY|BLOCK_FLAG_CORRUPTED)) { if ((*pp_block)->i_flags & (BLOCK_FLAG_DISCONTINUITY|BLOCK_FLAG_CORRUPTED)) {
block_Release(p_block); block_Release(*pp_block);
*pp_block = NULL;
timestamp_FifoEmpty(p_sys->timestamp_fifo); timestamp_FifoEmpty(p_sys->timestamp_fifo);
if (p_sys->decoded) { if (p_sys->decoded) {
/* Invalidate all pictures that are currently in flight /* Invalidate all pictures that are currently in flight
...@@ -1017,8 +1062,7 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block) ...@@ -1017,8 +1062,7 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block)
} }
} }
p_sys->decoded = false; p_sys->decoded = false;
jni_detach_thread(); goto endclean;
return NULL;
} }
/* Use the aspect ratio provided by the input (ie read from packetizer). /* Use the aspect ratio provided by the input (ie read from packetizer).
...@@ -1032,37 +1076,30 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block) ...@@ -1032,37 +1076,30 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block)
jlong timeout = 0; jlong timeout = 0;
const int max_polling_attempts = 50; const int max_polling_attempts = 50;
int attempts = 0; int attempts = 0;
while (true) { /* return when pp_block is processed */
int index = (*env)->CallIntMethod(env, p_sys->codec, p_sys->dequeue_input_buffer, (jlong) 0); while (*pp_block != NULL) {
if (CHECK_EXCEPTION()) { if (*pp_block != NULL)
msg_Err(p_dec, "Exception occurred in MediaCodec.dequeueInputBuffer"); PutInput(p_dec, env, pp_block, (jlong) 0);
p_sys->error_state = true; if (p_sys->error_state)
break; break;
}
if (index < 0) { if (p_pic == NULL) {
GetOutput(p_dec, env, &p_pic, timeout); GetOutput(p_dec, env, &p_pic, timeout);
if (p_sys->error_state) if (p_sys->error_state)
break; break;
if (p_pic) { }
/* If we couldn't get an available input buffer but a
* decoded frame is available, we return the frame if (p_pic == NULL && *pp_block != NULL) {
* without assigning NULL to *pp_block. The next call
* to DecodeVideo will try to send the input packet again.
*/
jni_detach_thread();
return p_pic;
}
timeout = 30 * 1000; timeout = 30 * 1000;
++attempts; ++attempts;
/* With opaque DR the output buffers are released by the /* With opaque DR the output buffers are released by the
vout therefore we implement a timeout for polling in vout therefore we implement a timeout for polling in
order to avoid being indefinitely stalled in this loop. */ order to avoid being indefinitely stalled in this loop. */
if (p_sys->direct_rendering && attempts == max_polling_attempts) { if (p_sys->direct_rendering && attempts == max_polling_attempts) {
picture_t *invalid_picture = decoder_NewPicture(p_dec); p_pic = decoder_NewPicture(p_dec);
if (invalid_picture) { if (p_pic) {
invalid_picture->date = VLC_TS_INVALID; p_pic->date = VLC_TS_INVALID;
picture_sys_t *p_picsys = invalid_picture->p_sys; picture_sys_t *p_picsys = p_pic->p_sys;
p_picsys->pf_lock_pic = NULL; p_picsys->pf_lock_pic = NULL;
p_picsys->pf_unlock_pic = NULL; p_picsys->pf_unlock_pic = NULL;
p_picsys->priv.hw.p_dec = NULL; p_picsys->priv.hw.p_dec = NULL;
...@@ -1073,59 +1110,32 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block) ...@@ -1073,59 +1110,32 @@ static picture_t *DecodeVideo(decoder_t *p_dec, block_t **pp_block)
/* If we cannot return a picture we must free the /* If we cannot return a picture we must free the
block since the decoder will proceed with the block since the decoder will proceed with the
next block. */ next block. */
block_Release(p_block); block_Release(*pp_block);
*pp_block = NULL; *pp_block = NULL;
} }
jni_detach_thread();
return invalid_picture;
} }
continue;
}
jobject buf;
if (p_sys->get_input_buffers)
buf = (*env)->GetObjectArrayElement(env, p_sys->input_buffers, index);
else
buf = (*env)->CallObjectMethod(env, p_sys->codec, p_sys->get_input_buffer, index);
jsize size = (*env)->GetDirectBufferCapacity(env, buf);
uint8_t *bufptr = (*env)->GetDirectBufferAddress(env, buf);
if (size < 0) {
msg_Err(p_dec, "Java buffer has invalid size");
p_sys->error_state = true;
break;
}
if ((size_t) size > p_block->i_buffer)
size = p_block->i_buffer;
memcpy(bufptr, p_block->p_buffer, size);
convert_h264_to_annexb(bufptr, size, p_sys->nal_size, &convert_state);
int64_t ts = p_block->i_pts;
if (!ts && p_block->i_dts)
ts = p_block->i_dts;
timestamp_FifoPut(p_sys->timestamp_fifo, p_block->i_pts ? VLC_TS_INVALID : p_block->i_dts);
(*env)->CallVoidMethod(env, p_sys->codec, p_sys->queue_input_buffer, index, 0, size, ts, 0);
(*env)->DeleteLocalRef(env, buf);
if (CHECK_EXCEPTION()) {
msg_Err(p_dec, "Exception in MediaCodec.queueInputBuffer");
p_sys->error_state = true;
break;
} }
p_sys->decoded = true;
break;
} }
endclean:
if (p_sys->error_state) { if (p_sys->error_state) {
if( pp_block && *pp_block )
{
block_Release(*pp_block);
*pp_block = NULL;
}
if (p_pic) if (p_pic)
picture_Release(p_pic); picture_Release(p_pic);
jni_detach_thread(); p_pic = NULL;
return NULL;
}
if (!p_pic)
GetOutput(p_dec, env, &p_pic, 0);
jni_detach_thread();
block_Release(p_block); if (!p_sys->error_event_sent) {
*pp_block = NULL; /* Signal the error to the Java. */
jni_EventHardwareAccelerationError();
p_sys->error_event_sent = true;
}
}
if (env != NULL)
jni_detach_thread();
return p_pic; return p_pic;
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment