diff mbox series

[FFmpeg-devel,1/2] lavc/vdpau_hevc: add function to find exact vdp_profile for REXT

Message ID 20200625112302.21994-2-mbonda@nvidia.com
State Superseded
Headers show
Series VDPAU HEVC 10/12 Bit Decode Support | expand

Checks

Context Check Description
andriy/default pending
andriy/make fail Make failed

Commit Message

ManojGuptaBonda June 25, 2020, 11:23 a.m. UTC
Add vdpau_parse_rext_profile and use profile constraint flags to
determine the exact vdp_profile for HEVC_REXT.

If profile mismatch is allowed, select Main profile by default.

Add build object in Makefile for h265_profile_level dependency.
---
 libavcodec/Makefile     |  2 +-
 libavcodec/vdpau_hevc.c | 83 ++++++++++++++++++++++++++++++++++++++++-
 2 files changed, 83 insertions(+), 2 deletions(-)

Comments

Philip Langdale June 25, 2020, 4:17 p.m. UTC | #1
On Thu, 25 Jun 2020 16:53:01 +0530
ManojGuptaBonda <mbonda@nvidia.com> wrote:

> Add vdpau_parse_rext_profile and use profile constraint flags to
> determine the exact vdp_profile for HEVC_REXT.
> 
> If profile mismatch is allowed, select Main profile by default.
> 
> Add build object in Makefile for h265_profile_level dependency.
> ---
>  libavcodec/Makefile     |  2 +-
>  libavcodec/vdpau_hevc.c | 83
> ++++++++++++++++++++++++++++++++++++++++- 2 files changed, 83
> insertions(+), 2 deletions(-)
> 
> diff --git a/libavcodec/Makefile b/libavcodec/Makefile
> index 5a6ea59715..4f28cb0f2a 100644
> --- a/libavcodec/Makefile
> +++ b/libavcodec/Makefile
> @@ -910,7 +910,7 @@ OBJS-$(CONFIG_HEVC_DXVA2_HWACCEL)         +=
> dxva2_hevc.o OBJS-$(CONFIG_HEVC_NVDEC_HWACCEL)         += nvdec_hevc.o
>  OBJS-$(CONFIG_HEVC_QSV_HWACCEL)           += qsvdec_h2645.o
>  OBJS-$(CONFIG_HEVC_VAAPI_HWACCEL)         += vaapi_hevc.o
> h265_profile_level.o -OBJS-$(CONFIG_HEVC_VDPAU_HWACCEL)         +=
> vdpau_hevc.o +OBJS-$(CONFIG_HEVC_VDPAU_HWACCEL)         +=
> vdpau_hevc.o h265_profile_level.o OBJS-$(CONFIG_MJPEG_NVDEC_HWACCEL)
>       += nvdec_mjpeg.o OBJS-$(CONFIG_MJPEG_VAAPI_HWACCEL)        +=
> vaapi_mjpeg.o OBJS-$(CONFIG_MPEG1_NVDEC_HWACCEL)        +=
> nvdec_mpeg12.o diff --git a/libavcodec/vdpau_hevc.c
> b/libavcodec/vdpau_hevc.c index 29cb2da078..1ee781c685 100644
> --- a/libavcodec/vdpau_hevc.c
> +++ b/libavcodec/vdpau_hevc.c
> @@ -29,6 +29,8 @@
>  #include "hwconfig.h"
>  #include "vdpau.h"
>  #include "vdpau_internal.h"
> +#include "h265_profile_level.h"
> +
>  
>  static int vdpau_hevc_start_frame(AVCodecContext *avctx,
>                                    const uint8_t *buffer, uint32_t
> size) @@ -429,10 +431,87 @@ static int
> vdpau_hevc_end_frame(AVCodecContext *avctx) return 0;
>  }
>  
> +
> +
> +static int ptl_convert(const PTLCommon *general_ptl,
> H265RawProfileTierLevel *h265_raw_ptl) +{
> +    h265_raw_ptl->general_profile_space = general_ptl->profile_space;
> +    h265_raw_ptl->general_tier_flag     = general_ptl->tier_flag;
> +    h265_raw_ptl->general_profile_idc   = general_ptl->profile_idc;
> +
> +    memcpy(h265_raw_ptl->general_profile_compatibility_flag,
> +
> general_ptl->profile_compatibility_flag, 32 * sizeof(uint8_t)); +
> +#define copy_field(name) h265_raw_ptl->general_ ## name =
> general_ptl->name
> +    copy_field(progressive_source_flag);
> +    copy_field(interlaced_source_flag);
> +    copy_field(non_packed_constraint_flag);
> +    copy_field(frame_only_constraint_flag);
> +    copy_field(max_12bit_constraint_flag);
> +    copy_field(max_10bit_constraint_flag);
> +    copy_field(max_8bit_constraint_flag);
> +    copy_field(max_422chroma_constraint_flag);
> +    copy_field(max_420chroma_constraint_flag);
> +    copy_field(max_monochrome_constraint_flag);
> +    copy_field(intra_constraint_flag);
> +    copy_field(one_picture_only_constraint_flag);
> +    copy_field(lower_bit_rate_constraint_flag);
> +    copy_field(max_14bit_constraint_flag);
> +    copy_field(inbld_flag);
> +    copy_field(level_idc);
> +#undef copy_field
> +
> +    return 0;
> +}
> +
> +/*
> + * Find exact vdpau_profile for HEVC Range Extension
> + */
> +static int vdpau_hevc_parse_rext_profile(AVCodecContext *avctx,
> VdpDecoderProfile *vdp_profile) +{
> +    const HEVCContext *h = avctx->priv_data;
> +    const HEVCSPS *sps = h->ps.sps;
> +    const PTL *ptl = &sps->ptl;
> +    const PTLCommon *general_ptl = &ptl->general_ptl;
> +    const H265ProfileDescriptor *profile;
> +    H265RawProfileTierLevel h265_raw_ptl = {0};
> +
> +    /* convert PTLCommon to H265RawProfileTierLevel */
> +    ptl_convert(general_ptl, &h265_raw_ptl);
> +
> +    profile = ff_h265_get_profile(&h265_raw_ptl);
> +    if (!profile) {
> +        av_log(avctx, AV_LOG_WARNING, "HEVC profile is not
> found.\n");
> +        if (avctx->hwaccel_flags &
> AV_HWACCEL_FLAG_ALLOW_PROFILE_MISMATCH) {
> +            // Default to selecting Main profile if profile mismatch
> is allowed
> +            *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN;
> +            return 0;
> +        } else
> +            return AVERROR(ENOTSUP);
> +    }
> +
> +    if (!strcmp(profile->name, "Main 12") ||
> +        !strcmp(profile->name, "Main 12 Intra"))
> +        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_12;
> +    else if (!strcmp(profile->name, "Main 4:4:4") ||
> +             !strcmp(profile->name, "Main 4:4:4 Intra"))
> +        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_444;
> +    else if (!strcmp(profile->name, "Main 4:4:4 10") ||
> +             !strcmp(profile->name, "Main 4:4:4 10 Intra"))
> +        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_444_10;
> +    else if (!strcmp(profile->name, "Main 4:4:4 12") ||
> +             !strcmp(profile->name, "Main 4:4:4 12 Intra"))
> +        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_444_12;
> +
> +    return 0;
> +}
> +
> +
>  static int vdpau_hevc_init(AVCodecContext *avctx)
>  {
>      VdpDecoderProfile profile;
>      uint32_t level = avctx->level;
> +    int ret;
>  
>      switch (avctx->profile) {
>      case FF_PROFILE_HEVC_MAIN:
> @@ -445,7 +524,9 @@ static int vdpau_hevc_init(AVCodecContext *avctx)
>          profile = VDP_DECODER_PROFILE_HEVC_MAIN_STILL;
>          break;
>      case FF_PROFILE_HEVC_REXT:
> -        profile = VDP_DECODER_PROFILE_HEVC_MAIN_444;
> +        ret = vdpau_hevc_parse_rext_profile(avctx, &profile);
> +        if (ret)
> +            return AVERROR(ENOTSUP);
>          break;
>      default:
>          return AVERROR(ENOTSUP);


LGTM.


--phil
Philip Langdale June 25, 2020, 6:12 p.m. UTC | #2
On Thu, 25 Jun 2020 09:17:19 -0700
Philip Langdale <philipl@overt.org> wrote:
> 
> 
> LGTM.
> 

I have been reminded that this only compiles cleanly with the latest
vdpau. We need #ifdef guards to continue compiling with older releases.

--phil
ManojGuptaBonda June 26, 2020, 7:16 a.m. UTC | #3
Thanks Philip, 

Updated the patch with #if gaurds and fixed the typo for AV_PIX_FMT_YUV422P10
Please review. 

Thanks,
ManojGupta.

> -----Original Message-----
> From: VDPAU <vdpau-bounces@lists.freedesktop.org> On Behalf Of Philip
> Langdale
> Sent: Thursday, June 25, 2020 11:43 PM
> To: Manoj Bonda <mbonda@nvidia.com>
> Cc: vdpau@lists.freedesktop.org; FFmpeg development discussions and
> patches <ffmpeg-devel@ffmpeg.org>; Aaron Plattner <aplattner@nvidia.com>
> Subject: Re: [VDPAU] [FFmpeg-devel] [PATCH 1/2] lavc/vdpau_hevc: add
> function to find exact vdp_profile for REXT
> 
> External email: Use caution opening links or attachments
> 
> 
> On Thu, 25 Jun 2020 09:17:19 -0700
> Philip Langdale <philipl@overt.org> wrote:
> >
> >
> > LGTM.
> >
> 
> I have been reminded that this only compiles cleanly with the latest vdpau. We
> need #ifdef guards to continue compiling with older releases.
> 
> --phil
> _______________________________________________
> VDPAU mailing list
> VDPAU@lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/vdpau
diff mbox series

Patch

diff --git a/libavcodec/Makefile b/libavcodec/Makefile
index 5a6ea59715..4f28cb0f2a 100644
--- a/libavcodec/Makefile
+++ b/libavcodec/Makefile
@@ -910,7 +910,7 @@  OBJS-$(CONFIG_HEVC_DXVA2_HWACCEL)         += dxva2_hevc.o
 OBJS-$(CONFIG_HEVC_NVDEC_HWACCEL)         += nvdec_hevc.o
 OBJS-$(CONFIG_HEVC_QSV_HWACCEL)           += qsvdec_h2645.o
 OBJS-$(CONFIG_HEVC_VAAPI_HWACCEL)         += vaapi_hevc.o h265_profile_level.o
-OBJS-$(CONFIG_HEVC_VDPAU_HWACCEL)         += vdpau_hevc.o
+OBJS-$(CONFIG_HEVC_VDPAU_HWACCEL)         += vdpau_hevc.o h265_profile_level.o
 OBJS-$(CONFIG_MJPEG_NVDEC_HWACCEL)        += nvdec_mjpeg.o
 OBJS-$(CONFIG_MJPEG_VAAPI_HWACCEL)        += vaapi_mjpeg.o
 OBJS-$(CONFIG_MPEG1_NVDEC_HWACCEL)        += nvdec_mpeg12.o
diff --git a/libavcodec/vdpau_hevc.c b/libavcodec/vdpau_hevc.c
index 29cb2da078..1ee781c685 100644
--- a/libavcodec/vdpau_hevc.c
+++ b/libavcodec/vdpau_hevc.c
@@ -29,6 +29,8 @@ 
 #include "hwconfig.h"
 #include "vdpau.h"
 #include "vdpau_internal.h"
+#include "h265_profile_level.h"
+
 
 static int vdpau_hevc_start_frame(AVCodecContext *avctx,
                                   const uint8_t *buffer, uint32_t size)
@@ -429,10 +431,87 @@  static int vdpau_hevc_end_frame(AVCodecContext *avctx)
     return 0;
 }
 
+
+
+static int ptl_convert(const PTLCommon *general_ptl, H265RawProfileTierLevel *h265_raw_ptl)
+{
+    h265_raw_ptl->general_profile_space = general_ptl->profile_space;
+    h265_raw_ptl->general_tier_flag     = general_ptl->tier_flag;
+    h265_raw_ptl->general_profile_idc   = general_ptl->profile_idc;
+
+    memcpy(h265_raw_ptl->general_profile_compatibility_flag,
+                                  general_ptl->profile_compatibility_flag, 32 * sizeof(uint8_t));
+
+#define copy_field(name) h265_raw_ptl->general_ ## name = general_ptl->name
+    copy_field(progressive_source_flag);
+    copy_field(interlaced_source_flag);
+    copy_field(non_packed_constraint_flag);
+    copy_field(frame_only_constraint_flag);
+    copy_field(max_12bit_constraint_flag);
+    copy_field(max_10bit_constraint_flag);
+    copy_field(max_8bit_constraint_flag);
+    copy_field(max_422chroma_constraint_flag);
+    copy_field(max_420chroma_constraint_flag);
+    copy_field(max_monochrome_constraint_flag);
+    copy_field(intra_constraint_flag);
+    copy_field(one_picture_only_constraint_flag);
+    copy_field(lower_bit_rate_constraint_flag);
+    copy_field(max_14bit_constraint_flag);
+    copy_field(inbld_flag);
+    copy_field(level_idc);
+#undef copy_field
+
+    return 0;
+}
+
+/*
+ * Find exact vdpau_profile for HEVC Range Extension
+ */
+static int vdpau_hevc_parse_rext_profile(AVCodecContext *avctx, VdpDecoderProfile *vdp_profile)
+{
+    const HEVCContext *h = avctx->priv_data;
+    const HEVCSPS *sps = h->ps.sps;
+    const PTL *ptl = &sps->ptl;
+    const PTLCommon *general_ptl = &ptl->general_ptl;
+    const H265ProfileDescriptor *profile;
+    H265RawProfileTierLevel h265_raw_ptl = {0};
+
+    /* convert PTLCommon to H265RawProfileTierLevel */
+    ptl_convert(general_ptl, &h265_raw_ptl);
+
+    profile = ff_h265_get_profile(&h265_raw_ptl);
+    if (!profile) {
+        av_log(avctx, AV_LOG_WARNING, "HEVC profile is not found.\n");
+        if (avctx->hwaccel_flags & AV_HWACCEL_FLAG_ALLOW_PROFILE_MISMATCH) {
+            // Default to selecting Main profile if profile mismatch is allowed
+            *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN;
+            return 0;
+        } else
+            return AVERROR(ENOTSUP);
+    }
+
+    if (!strcmp(profile->name, "Main 12") ||
+        !strcmp(profile->name, "Main 12 Intra"))
+        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_12;
+    else if (!strcmp(profile->name, "Main 4:4:4") ||
+             !strcmp(profile->name, "Main 4:4:4 Intra"))
+        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_444;
+    else if (!strcmp(profile->name, "Main 4:4:4 10") ||
+             !strcmp(profile->name, "Main 4:4:4 10 Intra"))
+        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_444_10;
+    else if (!strcmp(profile->name, "Main 4:4:4 12") ||
+             !strcmp(profile->name, "Main 4:4:4 12 Intra"))
+        *vdp_profile = VDP_DECODER_PROFILE_HEVC_MAIN_444_12;
+
+    return 0;
+}
+
+
 static int vdpau_hevc_init(AVCodecContext *avctx)
 {
     VdpDecoderProfile profile;
     uint32_t level = avctx->level;
+    int ret;
 
     switch (avctx->profile) {
     case FF_PROFILE_HEVC_MAIN:
@@ -445,7 +524,9 @@  static int vdpau_hevc_init(AVCodecContext *avctx)
         profile = VDP_DECODER_PROFILE_HEVC_MAIN_STILL;
         break;
     case FF_PROFILE_HEVC_REXT:
-        profile = VDP_DECODER_PROFILE_HEVC_MAIN_444;
+        ret = vdpau_hevc_parse_rext_profile(avctx, &profile);
+        if (ret)
+            return AVERROR(ENOTSUP);
         break;
     default:
         return AVERROR(ENOTSUP);