shithub: libvpx

Download patch

ref: 2fa9e9e2276b080c5357bcdbf6a129eb1fec864a
parent: 1cfbc86e3809c87b2397487baa01e3046bb0531d
author: hkuang <[email protected]>
date: Mon Feb 9 07:14:00 EST 2015

Fix the frame parallel invalid file test failure on ARM.

There is a corner case that when a frame is corrupted, the following
inter frame decode worker will miss the previous failure. To solve
this problem, a need_resync flag needs to be added to master thread
to keep control of that.

Change-Id: Iea9309b2562e7b59a83dd6b720607410286c90a6

--- a/test/vp9_frame_parallel_test.cc
+++ b/test/vp9_frame_parallel_test.cc
@@ -184,7 +184,7 @@
   }
 }
 
-TEST(VP9MultiThreadedFrameParallel, DISABLED_InvalidFileTest) {
+TEST(VP9MultiThreadedFrameParallel, InvalidFileTest) {
   static const InvalidFileList files[] = {
     // invalid-vp90-2-07-frame_parallel-1.webm is a 40 frame video file with
     // one key frame for every ten frames. The 11th frame has corrupted data.
--- a/vp9/vp9_dx_iface.c
+++ b/vp9/vp9_dx_iface.c
@@ -54,10 +54,10 @@
   int                     flushed;
   int                     invert_tile_order;
   int                     last_show_frame;  // Index of last output frame.
+  int                     byte_alignment;
 
   // Frame parallel related.
   int                     frame_parallel_decode;  // frame-based threading.
-  int                     byte_alignment;
   VP9Worker               *frame_workers;
   int                     num_frame_workers;
   int                     next_submit_worker_id;
@@ -68,7 +68,7 @@
   int                     frame_cache_write;
   int                     frame_cache_read;
   int                     num_cache_frames;
-
+  int                     need_resync;      // wait for key/intra-only frame
   // BufferPool that holds all reference frames. Shared by all the FrameWorkers.
   BufferPool              *buffer_pool;
 
@@ -361,6 +361,7 @@
   ctx->frame_cache_read = 0;
   ctx->frame_cache_write = 0;
   ctx->num_cache_frames = 0;
+  ctx->need_resync = 1;
   ctx->num_frame_workers =
       (ctx->frame_parallel_decode == 1) ? ctx->cfg.threads: 1;
   if (ctx->num_frame_workers > MAX_DECODE_THREADS)
@@ -445,6 +446,14 @@
   return VPX_CODEC_OK;
 }
 
+static INLINE void check_resync(vpx_codec_alg_priv_t *const ctx,
+                                const VP9Decoder *const pbi) {
+  // Clear resync flag if worker got a key frame or intra only frame.
+  if (ctx->need_resync == 1 && pbi->need_resync == 0 &&
+      (pbi->common.intra_only || pbi->common.frame_type == KEY_FRAME))
+    ctx->need_resync = 0;
+}
+
 static vpx_codec_err_t decode_one(vpx_codec_alg_priv_t *ctx,
                                   const uint8_t **data, unsigned int data_sz,
                                   void *user_priv, int64_t deadline) {
@@ -473,6 +482,7 @@
     frame_worker_data->data = *data;
     frame_worker_data->data_size = data_sz;
     frame_worker_data->user_priv = user_priv;
+    frame_worker_data->received_frame = 1;
 
     // Set these even if already initialized.  The caller may have changed the
     // decrypt config between frames.
@@ -487,6 +497,8 @@
 
     if (worker->had_error)
       return update_error_state(ctx, &frame_worker_data->pbi->common.error);
+
+    check_resync(ctx, frame_worker_data->pbi);
   } else {
     const VP9WorkerInterface *const winterface = vp9_get_worker_interface();
     VP9Worker *const worker = &ctx->frame_workers[ctx->next_submit_worker_id];
@@ -549,6 +561,9 @@
   winterface->sync(worker);
   frame_worker_data->received_frame = 0;
   ++ctx->available_threads;
+
+  check_resync(ctx, frame_worker_data->pbi);
+
   if (vp9_get_raw_frame(frame_worker_data->pbi, &sd, &flags) == 0) {
     VP9_COMMON *const cm = &frame_worker_data->pbi->common;
     RefCntBuffer *const frame_bufs = cm->buffer_pool->frame_bufs;
@@ -708,7 +723,7 @@
   // application fluhsed the decoder in frame parallel decode.
   if (ctx->frame_parallel_decode && ctx->available_threads > 0 &&
       !ctx->flushed) {
-    return img;
+    return NULL;
   }
 
   // Output the frames in the cache first.
@@ -715,6 +730,8 @@
   if (ctx->num_cache_frames > 0) {
     release_last_output_frame(ctx);
     ctx->last_show_frame  = ctx->frame_cache[ctx->frame_cache_read].fb_idx;
+    if (ctx->need_resync)
+      return NULL;
     img = &ctx->frame_cache[ctx->frame_cache_read].img;
     ctx->frame_cache_read = (ctx->frame_cache_read + 1) % FRAME_CACHE_SIZE;
     --ctx->num_cache_frames;
@@ -737,14 +754,18 @@
       // Wait for the frame from worker thread.
       if (winterface->sync(worker)) {
         // Check if worker has received any frames.
-        if (frame_worker_data->received_frame == 1)
+        if (frame_worker_data->received_frame == 1) {
           ++ctx->available_threads;
-        frame_worker_data->received_frame = 0;
+          frame_worker_data->received_frame = 0;
+          check_resync(ctx, frame_worker_data->pbi);
+        }
         if (vp9_get_raw_frame(frame_worker_data->pbi, &sd, &flags) == 0) {
           VP9_COMMON *const cm = &frame_worker_data->pbi->common;
           RefCntBuffer *const frame_bufs = cm->buffer_pool->frame_bufs;
           release_last_output_frame(ctx);
           ctx->last_show_frame = frame_worker_data->pbi->common.new_fb_idx;
+          if (ctx->need_resync)
+            return NULL;
           yuvconfig2image(&ctx->img, &sd, frame_worker_data->user_priv);
           ctx->img.fb_priv = frame_bufs[cm->new_fb_idx].raw_frame_buffer.priv;
           img = &ctx->img;
@@ -754,12 +775,13 @@
         // Decoding failed. Release the worker thread.
         frame_worker_data->received_frame = 0;
         ++ctx->available_threads;
+        ctx->need_resync = 1;
         if (ctx->flushed != 1)
-          return img;
+          return NULL;
       }
     } while (ctx->next_output_worker_id != ctx->next_submit_worker_id);
   }
-  return img;
+  return NULL;
 }
 
 static vpx_codec_err_t decoder_set_fb_fn(