diff mbox

[FFmpeg-devel,PATCHv2,1/2] avcodec: add execute3() api to utilize the main function of avpriv_slicethread_create().

Message ID 20170905235725.1240-1-zakne0ne@gmail.com
State New
Headers show

Commit Message

Ilia Sept. 5, 2017, 11:57 p.m. UTC
Signed-off-by: Ilia Valiakhmetov <zakne0ne@gmail.com>
---
 libavcodec/avcodec.h       |  7 ++++++-
 libavcodec/options.c       |  1 +
 libavcodec/pthread_slice.c | 26 ++++++++++++++++++++++++--
 libavcodec/utils.c         | 14 ++++++++++++++
 4 files changed, 45 insertions(+), 3 deletions(-)

Comments

James Almer Sept. 6, 2017, 3:23 a.m. UTC | #1
On 9/5/2017 8:57 PM, Ilia Valiakhmetov wrote:
> Signed-off-by: Ilia Valiakhmetov <zakne0ne@gmail.com>
> ---
>  libavcodec/avcodec.h       |  7 ++++++-
>  libavcodec/options.c       |  1 +
>  libavcodec/pthread_slice.c | 26 ++++++++++++++++++++++++--
>  libavcodec/utils.c         | 14 ++++++++++++++
>  4 files changed, 45 insertions(+), 3 deletions(-)

Missing APIChanges entry and version bump for the new API.

> 
> diff --git a/libavcodec/avcodec.h b/libavcodec/avcodec.h
> index 6555550..712f40c 100644
> --- a/libavcodec/avcodec.h
> +++ b/libavcodec/avcodec.h
> @@ -1089,6 +1089,10 @@ typedef struct RcOverride{
>   */
>  #define AV_CODEC_CAP_AVOID_PROBING       (1 << 17)
>  /**
> + * Codec initializes slice-based threading with a main function
> + */
> +#define AV_CODEC_SLICE_THREAD_HAS_MF     (1 << 18)

Prefix should be AV_CODEC_CAP_ like other flags.
Does this need to be a public capability flag, for that matter? Maybe it
could be instead an internal FF_CODEC_CAP_ one.

See AVCodec->caps_internal.
wm4 Sept. 6, 2017, 9:35 a.m. UTC | #2
On Wed,  6 Sep 2017 06:57:25 +0700
Ilia Valiakhmetov <zakne0ne@gmail.com> wrote:

> Signed-off-by: Ilia Valiakhmetov <zakne0ne@gmail.com>
> ---
>  libavcodec/avcodec.h       |  7 ++++++-
>  libavcodec/options.c       |  1 +
>  libavcodec/pthread_slice.c | 26 ++++++++++++++++++++++++--
>  libavcodec/utils.c         | 14 ++++++++++++++
>  4 files changed, 45 insertions(+), 3 deletions(-)
> 
> diff --git a/libavcodec/avcodec.h b/libavcodec/avcodec.h
> index 6555550..712f40c 100644
> --- a/libavcodec/avcodec.h
> +++ b/libavcodec/avcodec.h
> @@ -1089,6 +1089,10 @@ typedef struct RcOverride{
>   */
>  #define AV_CODEC_CAP_AVOID_PROBING       (1 << 17)
>  /**
> + * Codec initializes slice-based threading with a main function
> + */
> +#define AV_CODEC_SLICE_THREAD_HAS_MF     (1 << 18)
> +/**
>   * Codec is intra only.
>   */
>  #define AV_CODEC_CAP_INTRA_ONLY       0x40000000
> @@ -3233,7 +3237,7 @@ typedef struct AVCodecContext {
>       * - decoding: Set by libavcodec, user can override.
>       */
>      int (*execute2)(struct AVCodecContext *c, int (*func)(struct AVCodecContext *c2, void *arg, int jobnr, int threadnr), void *arg2, int *ret, int count);
> -
> +    int (*execute3)(struct AVCodecContext *c, int (*func)(struct AVCodecContext *c2, void *arg, int jobnr, int threadnr), int (*m_func)(struct AVCodecContext *c3), void *arg2, int *ret, int count);
>      /**
>       * noise vs. sse weight for the nsse comparison function
>       * - encoding: Set by user.
> @@ -5774,6 +5778,7 @@ const char *avcodec_profile_name(enum AVCodecID codec_id, int profile);
>  
>  int avcodec_default_execute(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2),void *arg, int *ret, int count, int size);
>  int avcodec_default_execute2(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2, int, int),void *arg, int *ret, int count);
> +int avcodec_default_execute3(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2, int jobnr, int threadnr), int (*m_func)(struct AVCodecContext *c3), void *arg, int *ret, int count);
>  //FIXME func typedef
>  
>  /**
> diff --git a/libavcodec/options.c b/libavcodec/options.c
> index 82e1217..6d63bdb 100644
> --- a/libavcodec/options.c
> +++ b/libavcodec/options.c
> @@ -117,6 +117,7 @@ static int init_context_defaults(AVCodecContext *s, const AVCodec *codec)
>      s->get_format          = avcodec_default_get_format;
>      s->execute             = avcodec_default_execute;
>      s->execute2            = avcodec_default_execute2;
> +    s->execute3            = avcodec_default_execute3;
>      s->sample_aspect_ratio = (AVRational){0,1};
>      s->pix_fmt             = AV_PIX_FMT_NONE;
>      s->sw_pix_fmt          = AV_PIX_FMT_NONE;
> diff --git a/libavcodec/pthread_slice.c b/libavcodec/pthread_slice.c
> index c781d35..3aff816 100644
> --- a/libavcodec/pthread_slice.c
> +++ b/libavcodec/pthread_slice.c
> @@ -38,11 +38,13 @@
>  
>  typedef int (action_func)(AVCodecContext *c, void *arg);
>  typedef int (action_func2)(AVCodecContext *c, void *arg, int jobnr, int threadnr);
> +typedef int (main_func)(AVCodecContext *c);
>  
>  typedef struct SliceThreadContext {
>      AVSliceThread *thread;
>      action_func *func;
>      action_func2 *func2;
> +    main_func *m_func;
>      void *args;
>      int *rets;
>      int job_size;
> @@ -54,6 +56,12 @@ typedef struct SliceThreadContext {
>      pthread_mutex_t *progress_mutex;
>  } SliceThreadContext;
>  
> +static void main_function(void *priv) {
> +    AVCodecContext *avctx = priv;
> +    SliceThreadContext *c = avctx->internal->thread_ctx;
> +    c->m_func(avctx);
> +}
> +
>  static void worker_func(void *priv, int jobnr, int threadnr, int nb_jobs, int nb_threads)
>  {
>      AVCodecContext *avctx = priv;
> @@ -99,7 +107,8 @@ static int thread_execute(AVCodecContext *avctx, action_func* func, void *arg, i
>      c->func = func;
>      c->rets = ret;
>  
> -    avpriv_slicethread_execute(c->thread, job_count, 0);
> +    avpriv_slicethread_execute(c->thread, job_count, !!c->m_func);
> +
>      return 0;
>  }
>  
> @@ -110,10 +119,20 @@ static int thread_execute2(AVCodecContext *avctx, action_func2* func2, void *arg
>      return thread_execute(avctx, NULL, arg, ret, job_count, 0);
>  }
>  
> +static int thread_execute3(AVCodecContext *avctx, action_func2* func2, main_func* m_func, void *arg, int *ret, int job_count)
> +{
> +    SliceThreadContext *c = avctx->internal->thread_ctx;
> +    c->func2 = func2;
> +    c->m_func = m_func;
> +    return thread_execute(avctx, NULL, arg, ret, job_count, 0);
> +}
> +
> +
>  int ff_slice_thread_init(AVCodecContext *avctx)
>  {
>      SliceThreadContext *c;
>      int thread_count = avctx->thread_count;
> +    static void (*m_f)(void *);
>  
>  #if HAVE_W32THREADS
>      w32thread_init();
> @@ -142,7 +161,9 @@ int ff_slice_thread_init(AVCodecContext *avctx)
>      }
>  
>      avctx->internal->thread_ctx = c = av_mallocz(sizeof(*c));
> -    if (!c || (thread_count = avpriv_slicethread_create(&c->thread, avctx, worker_func, NULL, thread_count)) <= 1) {
> +    m_f = avctx->codec->capabilities & AV_CODEC_SLICE_THREAD_HAS_MF ? &main_function : NULL;
> +
> +    if (!c || (thread_count = avpriv_slicethread_create(&c->thread, avctx, worker_func, m_f, thread_count)) <= 1) {
>          if (c)
>              avpriv_slicethread_free(&c->thread);
>          av_freep(&avctx->internal->thread_ctx);
> @@ -154,6 +175,7 @@ int ff_slice_thread_init(AVCodecContext *avctx)
>  
>      avctx->execute = thread_execute;
>      avctx->execute2 = thread_execute2;
> +    avctx->execute3 = thread_execute3;
>      return 0;
>  }
>  
> diff --git a/libavcodec/utils.c b/libavcodec/utils.c
> index baf4992..150e145 100644
> --- a/libavcodec/utils.c
> +++ b/libavcodec/utils.c
> @@ -558,6 +558,20 @@ int avcodec_default_execute2(AVCodecContext *c, int (*func)(AVCodecContext *c2,
>      return 0;
>  }
>  
> +int avcodec_default_execute3(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2, int jobnr, int threadnr), int (*m_func)(struct AVCodecContext *c3), void *arg, int *ret, int count)
> +{
> +    int i;
> +
> +    for (i = 0; i < count; i++) {
> +        int r = func(c, arg, i, 0);
> +        if (ret)
> +            ret[i] = r;
> +    }
> +    m_func(c);
> +    emms_c();
> +    return 0;
> +}
> +
>  enum AVPixelFormat avpriv_find_pix_fmt(const PixelFormatTag *tags,
>                                         unsigned int fourcc)
>  {

Shouldn't all of this be strictly internal? Also I could think of
better names than just attaching a "3".
diff mbox

Patch

diff --git a/libavcodec/avcodec.h b/libavcodec/avcodec.h
index 6555550..712f40c 100644
--- a/libavcodec/avcodec.h
+++ b/libavcodec/avcodec.h
@@ -1089,6 +1089,10 @@  typedef struct RcOverride{
  */
 #define AV_CODEC_CAP_AVOID_PROBING       (1 << 17)
 /**
+ * Codec initializes slice-based threading with a main function
+ */
+#define AV_CODEC_SLICE_THREAD_HAS_MF     (1 << 18)
+/**
  * Codec is intra only.
  */
 #define AV_CODEC_CAP_INTRA_ONLY       0x40000000
@@ -3233,7 +3237,7 @@  typedef struct AVCodecContext {
      * - decoding: Set by libavcodec, user can override.
      */
     int (*execute2)(struct AVCodecContext *c, int (*func)(struct AVCodecContext *c2, void *arg, int jobnr, int threadnr), void *arg2, int *ret, int count);
-
+    int (*execute3)(struct AVCodecContext *c, int (*func)(struct AVCodecContext *c2, void *arg, int jobnr, int threadnr), int (*m_func)(struct AVCodecContext *c3), void *arg2, int *ret, int count);
     /**
      * noise vs. sse weight for the nsse comparison function
      * - encoding: Set by user.
@@ -5774,6 +5778,7 @@  const char *avcodec_profile_name(enum AVCodecID codec_id, int profile);
 
 int avcodec_default_execute(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2),void *arg, int *ret, int count, int size);
 int avcodec_default_execute2(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2, int, int),void *arg, int *ret, int count);
+int avcodec_default_execute3(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2, int jobnr, int threadnr), int (*m_func)(struct AVCodecContext *c3), void *arg, int *ret, int count);
 //FIXME func typedef
 
 /**
diff --git a/libavcodec/options.c b/libavcodec/options.c
index 82e1217..6d63bdb 100644
--- a/libavcodec/options.c
+++ b/libavcodec/options.c
@@ -117,6 +117,7 @@  static int init_context_defaults(AVCodecContext *s, const AVCodec *codec)
     s->get_format          = avcodec_default_get_format;
     s->execute             = avcodec_default_execute;
     s->execute2            = avcodec_default_execute2;
+    s->execute3            = avcodec_default_execute3;
     s->sample_aspect_ratio = (AVRational){0,1};
     s->pix_fmt             = AV_PIX_FMT_NONE;
     s->sw_pix_fmt          = AV_PIX_FMT_NONE;
diff --git a/libavcodec/pthread_slice.c b/libavcodec/pthread_slice.c
index c781d35..3aff816 100644
--- a/libavcodec/pthread_slice.c
+++ b/libavcodec/pthread_slice.c
@@ -38,11 +38,13 @@ 
 
 typedef int (action_func)(AVCodecContext *c, void *arg);
 typedef int (action_func2)(AVCodecContext *c, void *arg, int jobnr, int threadnr);
+typedef int (main_func)(AVCodecContext *c);
 
 typedef struct SliceThreadContext {
     AVSliceThread *thread;
     action_func *func;
     action_func2 *func2;
+    main_func *m_func;
     void *args;
     int *rets;
     int job_size;
@@ -54,6 +56,12 @@  typedef struct SliceThreadContext {
     pthread_mutex_t *progress_mutex;
 } SliceThreadContext;
 
+static void main_function(void *priv) {
+    AVCodecContext *avctx = priv;
+    SliceThreadContext *c = avctx->internal->thread_ctx;
+    c->m_func(avctx);
+}
+
 static void worker_func(void *priv, int jobnr, int threadnr, int nb_jobs, int nb_threads)
 {
     AVCodecContext *avctx = priv;
@@ -99,7 +107,8 @@  static int thread_execute(AVCodecContext *avctx, action_func* func, void *arg, i
     c->func = func;
     c->rets = ret;
 
-    avpriv_slicethread_execute(c->thread, job_count, 0);
+    avpriv_slicethread_execute(c->thread, job_count, !!c->m_func);
+
     return 0;
 }
 
@@ -110,10 +119,20 @@  static int thread_execute2(AVCodecContext *avctx, action_func2* func2, void *arg
     return thread_execute(avctx, NULL, arg, ret, job_count, 0);
 }
 
+static int thread_execute3(AVCodecContext *avctx, action_func2* func2, main_func* m_func, void *arg, int *ret, int job_count)
+{
+    SliceThreadContext *c = avctx->internal->thread_ctx;
+    c->func2 = func2;
+    c->m_func = m_func;
+    return thread_execute(avctx, NULL, arg, ret, job_count, 0);
+}
+
+
 int ff_slice_thread_init(AVCodecContext *avctx)
 {
     SliceThreadContext *c;
     int thread_count = avctx->thread_count;
+    static void (*m_f)(void *);
 
 #if HAVE_W32THREADS
     w32thread_init();
@@ -142,7 +161,9 @@  int ff_slice_thread_init(AVCodecContext *avctx)
     }
 
     avctx->internal->thread_ctx = c = av_mallocz(sizeof(*c));
-    if (!c || (thread_count = avpriv_slicethread_create(&c->thread, avctx, worker_func, NULL, thread_count)) <= 1) {
+    m_f = avctx->codec->capabilities & AV_CODEC_SLICE_THREAD_HAS_MF ? &main_function : NULL;
+
+    if (!c || (thread_count = avpriv_slicethread_create(&c->thread, avctx, worker_func, m_f, thread_count)) <= 1) {
         if (c)
             avpriv_slicethread_free(&c->thread);
         av_freep(&avctx->internal->thread_ctx);
@@ -154,6 +175,7 @@  int ff_slice_thread_init(AVCodecContext *avctx)
 
     avctx->execute = thread_execute;
     avctx->execute2 = thread_execute2;
+    avctx->execute3 = thread_execute3;
     return 0;
 }
 
diff --git a/libavcodec/utils.c b/libavcodec/utils.c
index baf4992..150e145 100644
--- a/libavcodec/utils.c
+++ b/libavcodec/utils.c
@@ -558,6 +558,20 @@  int avcodec_default_execute2(AVCodecContext *c, int (*func)(AVCodecContext *c2,
     return 0;
 }
 
+int avcodec_default_execute3(AVCodecContext *c, int (*func)(AVCodecContext *c2, void *arg2, int jobnr, int threadnr), int (*m_func)(struct AVCodecContext *c3), void *arg, int *ret, int count)
+{
+    int i;
+
+    for (i = 0; i < count; i++) {
+        int r = func(c, arg, i, 0);
+        if (ret)
+            ret[i] = r;
+    }
+    m_func(c);
+    emms_c();
+    return 0;
+}
+
 enum AVPixelFormat avpriv_find_pix_fmt(const PixelFormatTag *tags,
                                        unsigned int fourcc)
 {