[v9 04/13] drm: Enable HDR infoframe support

Ville Syrjälä ville.syrjala at linux.intel.com
Tue May 14 12:44:39 UTC 2019


On Tue, May 14, 2019 at 12:36:25PM +0000, Kazlauskas, Nicholas wrote:
> On 5/8/19 2:38 PM, Uma Shankar wrote:
> > [CAUTION: External Email]
> > 
> > Enable Dynamic Range and Mastering Infoframe for HDR
> > content, which is defined in CEA 861.3 spec.
> > 
> > The metadata will be computed based on blending
> > policy in userspace compositors and passed as a connector
> > property blob to driver. The same will be sent as infoframe
> > to panel which support HDR.
> > 
> > Added the const version of infoframe for DRM metadata
> > for HDR.
> > 
> > v2: Rebase and added Ville's POC changes.
> > 
> > v3: No Change
> > 
> > v4: Addressed Shashank's review comments and merged the
> > patch making drm infoframe function arguments as constant.
> > 
> > v5: Rebase
> > 
> > v6: Fixed checkpatch warnings with --strict option. Addressed
> > Shashank's review comments and added his RB.
> > 
> > v7: Addressed Brian Starkey's review comments. Merged 2 patches
> > into one.
> > 
> > v8: Addressed Jonas Karlman review comments.
> > 
> > Signed-off-by: Uma Shankar <uma.shankar at intel.com>
> > Signed-off-by: Ville Syrjälä <ville.syrjala at linux.intel.com>
> > Reviewed-by: Shashank Sharma <shashank.sharma at intel.com>
> > ---
> >   drivers/gpu/drm/drm_edid.c |  48 ++++++++++++
> >   drivers/video/hdmi.c       | 186 +++++++++++++++++++++++++++++++++++++++++++++
> >   include/drm/drm_edid.h     |   5 ++
> >   include/linux/hdmi.h       |  27 +++++++
> >   4 files changed, 266 insertions(+)
> > 
> > diff --git a/drivers/gpu/drm/drm_edid.c b/drivers/gpu/drm/drm_edid.c
> > index fe2c29b..5f48965 100644
> > --- a/drivers/gpu/drm/drm_edid.c
> > +++ b/drivers/gpu/drm/drm_edid.c
> > @@ -4906,6 +4906,54 @@ static bool is_hdmi2_sink(struct drm_connector *connector)
> >   }
> > 
> >   /**
> > + * drm_hdmi_infoframe_set_hdr_metadata() - fill an HDMI AVI infoframe with
> > + *                                         HDR metadata from userspace
> > + * @frame: HDMI AVI infoframe
> > + * @hdr_source_metadata: hdr_source_metadata info from userspace
> > + *
> > + * Return: 0 on success or a negative error code on failure.
> > + */
> > +int
> > +drm_hdmi_infoframe_set_hdr_metadata(struct hdmi_drm_infoframe *frame,
> > +                                   struct hdr_output_metadata *hdr_metadata)

hdr_metadata can be const.

> > +{
> > +       int err;
> > +
> > +       if (!frame || !hdr_metadata)
> > +               return true;

true?

> > +
> > +       err = hdmi_drm_infoframe_init(frame);
> > +       if (err < 0)
> > +               return err;
> > +
> > +       DRM_DEBUG_KMS("type = %x\n", frame->type);
> > +
> > +       frame->length = sizeof(struct hdr_metadata_infoframe);
> > +
> > +       frame->eotf = hdr_metadata->hdmi_metadata_type1.eotf;
> > +       frame->metadata_type = hdr_metadata->hdmi_metadata_type1.metadata_type;
> > +
> > +       memcpy(&frame->display_primaries,
> > +              &hdr_metadata->hdmi_metadata_type1.display_primaries, 12);
> > +
> > +       memcpy(&frame->white_point,
> > +              &hdr_metadata->hdmi_metadata_type1.white_point, 4);
> > +
> > +       frame->max_display_mastering_luminance =
> > +               hdr_metadata->hdmi_metadata_type1.max_display_mastering_luminance;
> > +       frame->min_display_mastering_luminance =
> > +               hdr_metadata->hdmi_metadata_type1.min_display_mastering_luminance;
> > +       frame->max_fall = hdr_metadata->hdmi_metadata_type1.max_fall;
> > +       frame->max_cll = hdr_metadata->hdmi_metadata_type1.max_cll;
> > +
> > +       hdmi_infoframe_log(KERN_CRIT, NULL,
> > +                          (union hdmi_infoframe *)frame);
> 
> This shouldn't really be KERN_CRIT for debug output. Maybe KERN_INFO or 
> KERN_DEBUG or just drop the log altogether since it can't actually tag 
> it to DRM or the driver itself in this function.

Yeah, this shouldn't be here. It's up to the driver to dump it when it
wants to.

> 
> Nicholas Kazlauskas
> 
> > +
> > +       return 0;
> > +}
> > +EXPORT_SYMBOL(drm_hdmi_infoframe_set_hdr_metadata);
> > +
> > +/**
> >    * drm_hdmi_avi_infoframe_from_display_mode() - fill an HDMI AVI infoframe with
> >    *                                              data from a DRM display mode
> >    * @frame: HDMI AVI infoframe
> > diff --git a/drivers/video/hdmi.c b/drivers/video/hdmi.c
> > index 799ae49..717ed7fb 100644
> > --- a/drivers/video/hdmi.c
> > +++ b/drivers/video/hdmi.c
> > @@ -650,6 +650,146 @@ ssize_t hdmi_vendor_infoframe_pack(struct hdmi_vendor_infoframe *frame,
> >          return 0;
> >   }
> > 
> > +/**
> > + * hdmi_drm_infoframe_init() - initialize an HDMI Dynaminc Range and
> > + * mastering infoframe
> > + * @frame: HDMI DRM infoframe
> > + *
> > + * Returns 0 on success or a negative error code on failure.
> > + */
> > +int hdmi_drm_infoframe_init(struct hdmi_drm_infoframe *frame)
> > +{
> > +       memset(frame, 0, sizeof(*frame));
> > +
> > +       frame->type = HDMI_INFOFRAME_TYPE_DRM;
> > +       frame->version = 1;
> > +
> > +       return 0;
> > +}
> > +EXPORT_SYMBOL(hdmi_drm_infoframe_init);
> > +
> > +static int hdmi_drm_infoframe_check_only(const struct hdmi_drm_infoframe *frame)
> > +{
> > +       if (frame->type != HDMI_INFOFRAME_TYPE_DRM ||
> > +           frame->version != 1)
> > +               return -EINVAL;
> > +
> > +       return 0;
> > +}
> > +
> > +/**
> > + * hdmi_drm_infoframe_check() - check a HDMI DRM infoframe
> > + * @frame: HDMI DRM infoframe
> > + *
> > + * Validates that the infoframe is consistent.
> > + * Returns 0 on success or a negative error code on failure.
> > + */
> > +int hdmi_drm_infoframe_check(struct hdmi_drm_infoframe *frame)
> > +{
> > +       return hdmi_drm_infoframe_check_only(frame);
> > +}
> > +EXPORT_SYMBOL(hdmi_drm_infoframe_check);
> > +
> > +/**
> > + * hdmi_drm_infoframe_pack() - write HDMI DRM infoframe to binary buffer
> > + * @frame: HDMI DRM infoframe
> > + * @buffer: destination buffer
> > + * @size: size of buffer
> > + *
> > + * Packs the information contained in the @frame structure into a binary
> > + * representation that can be written into the corresponding controller
> > + * registers. Also computes the checksum as required by section 5.3.5 of
> > + * the HDMI 1.4 specification.
> > + *
> > + * Returns the number of bytes packed into the binary buffer or a negative
> > + * error code on failure.
> > + */
> > +ssize_t hdmi_drm_infoframe_pack_only(const struct hdmi_drm_infoframe *frame,
> > +                                    void *buffer, size_t size)
> > +{
> > +       u8 *ptr = buffer;
> > +       size_t length;
> > +       int i;
> > +
> > +       length = HDMI_INFOFRAME_HEADER_SIZE + frame->length;
> > +
> > +       if (size < length)
> > +               return -ENOSPC;
> > +
> > +       memset(buffer, 0, size);
> > +
> > +       ptr[0] = frame->type;
> > +       ptr[1] = frame->version;
> > +       ptr[2] = frame->length;
> > +       ptr[3] = 0; /* checksum */
> > +
> > +       /* start infoframe payload */
> > +       ptr += HDMI_INFOFRAME_HEADER_SIZE;
> > +
> > +       *ptr++ = frame->eotf;
> > +       *ptr++ = frame->metadata_type;
> > +
> > +       for (i = 0; i < 3; i++) {
> > +               *ptr++ = frame->display_primaries[i].x;
> > +               *ptr++ = frame->display_primaries[i].x >> 8;
> > +               *ptr++ = frame->display_primaries[i].y;
> > +               *ptr++ = frame->display_primaries[i].y >> 8;
> > +       }
> > +
> > +       *ptr++ = frame->white_point.x;
> > +       *ptr++ = frame->white_point.x >> 8;
> > +
> > +       *ptr++ = frame->white_point.y;
> > +       *ptr++ = frame->white_point.y >> 8;
> > +
> > +       *ptr++ = frame->max_display_mastering_luminance;
> > +       *ptr++ = frame->max_display_mastering_luminance >> 8;
> > +
> > +       *ptr++ = frame->min_display_mastering_luminance;
> > +       *ptr++ = frame->min_display_mastering_luminance >> 8;
> > +
> > +       *ptr++ = frame->max_cll;
> > +       *ptr++ = frame->max_cll >> 8;
> > +
> > +       *ptr++ = frame->max_fall;
> > +       *ptr++ = frame->max_fall >> 8;
> > +
> > +       hdmi_infoframe_set_checksum(buffer, length);
> > +
> > +       return length;
> > +}
> > +EXPORT_SYMBOL(hdmi_drm_infoframe_pack_only);
> > +
> > +/**
> > + * hdmi_drm_infoframe_pack() - check a HDMI DRM infoframe,
> > + *                             and write it to binary buffer
> > + * @frame: HDMI DRM infoframe
> > + * @buffer: destination buffer
> > + * @size: size of buffer
> > + *
> > + * Validates that the infoframe is consistent and updates derived fields
> > + * (eg. length) based on other fields, after which it packs the information
> > + * contained in the @frame structure into a binary representation that
> > + * can be written into the corresponding controller registers. This function
> > + * also computes the checksum as required by section 5.3.5 of the HDMI 1.4
> > + * specification.
> > + *
> > + * Returns the number of bytes packed into the binary buffer or a negative
> > + * error code on failure.
> > + */
> > +ssize_t hdmi_drm_infoframe_pack(struct hdmi_drm_infoframe *frame,
> > +                               void *buffer, size_t size)
> > +{
> > +       int ret;
> > +
> > +       ret = hdmi_drm_infoframe_check(frame);
> > +       if (ret)
> > +               return ret;
> > +
> > +       return hdmi_drm_infoframe_pack_only(frame, buffer, size);
> > +}
> > +EXPORT_SYMBOL(hdmi_drm_infoframe_pack);
> > +
> >   /*
> >    * hdmi_vendor_any_infoframe_check() - check a vendor infoframe
> >    */
> > @@ -758,6 +898,10 @@ ssize_t hdmi_vendor_infoframe_pack(struct hdmi_vendor_infoframe *frame,
> >                  length = hdmi_avi_infoframe_pack_only(&frame->avi,
> >                                                        buffer, size);
> >                  break;
> > +       case HDMI_INFOFRAME_TYPE_DRM:
> > +               length = hdmi_drm_infoframe_pack_only(&frame->drm,
> > +                                                     buffer, size);
> > +               break;
> >          case HDMI_INFOFRAME_TYPE_SPD:
> >                  length = hdmi_spd_infoframe_pack_only(&frame->spd,
> >                                                        buffer, size);
> > @@ -806,6 +950,9 @@ ssize_t hdmi_vendor_infoframe_pack(struct hdmi_vendor_infoframe *frame,
> >          case HDMI_INFOFRAME_TYPE_AVI:
> >                  length = hdmi_avi_infoframe_pack(&frame->avi, buffer, size);
> >                  break;
> > +       case HDMI_INFOFRAME_TYPE_DRM:
> > +               length = hdmi_drm_infoframe_pack(&frame->drm, buffer, size);
> > +               break;
> >          case HDMI_INFOFRAME_TYPE_SPD:
> >                  length = hdmi_spd_infoframe_pack(&frame->spd, buffer, size);
> >                  break;
> > @@ -838,6 +985,8 @@ static const char *hdmi_infoframe_type_get_name(enum hdmi_infoframe_type type)
> >                  return "Source Product Description (SPD)";
> >          case HDMI_INFOFRAME_TYPE_AUDIO:
> >                  return "Audio";
> > +       case HDMI_INFOFRAME_TYPE_DRM:
> > +               return "Dynamic Range and Mastering";
> >          }
> >          return "Reserved";
> >   }
> > @@ -1284,6 +1433,40 @@ static void hdmi_audio_infoframe_log(const char *level,
> >                          frame->downmix_inhibit ? "Yes" : "No");
> >   }
> > 
> > +/**
> > + * hdmi_drm_infoframe_log() - log info of HDMI DRM infoframe
> > + * @level: logging level
> > + * @dev: device
> > + * @frame: HDMI DRM infoframe
> > + */
> > +static void hdmi_drm_infoframe_log(const char *level,
> > +                                  struct device *dev,
> > +                                  const struct hdmi_drm_infoframe *frame)
> > +{
> > +       int i;
> > +
> > +       hdmi_infoframe_log_header(level, dev,
> > +                                 (struct hdmi_any_infoframe *)frame);
> > +       hdmi_log("length: %d\n", frame->length);
> > +       hdmi_log("metadata type: %d\n", frame->metadata_type);
> > +       hdmi_log("eotf: %d\n", frame->eotf);
> > +       for (i = 0; i < 3; i++) {
> > +               hdmi_log("x[%d]: %d\n", i, frame->display_primaries[i].x);
> > +               hdmi_log("y[%d]: %d\n", i, frame->display_primaries[i].y);
> > +       }
> > +
> > +       hdmi_log("white point x: %d\n", frame->white_point.x);
> > +       hdmi_log("white point y: %d\n", frame->white_point.y);
> > +
> > +       hdmi_log("max_display_mastering_luminance: %d\n",
> > +                frame->max_display_mastering_luminance);
> > +       hdmi_log("min_display_mastering_luminance: %d\n",
> > +                frame->min_display_mastering_luminance);
> > +
> > +       hdmi_log("max_cll: %d\n", frame->max_cll);
> > +       hdmi_log("max_fall: %d\n", frame->max_fall);
> > +}
> > +
> >   static const char *
> >   hdmi_3d_structure_get_name(enum hdmi_3d_structure s3d_struct)
> >   {
> > @@ -1372,6 +1555,9 @@ void hdmi_infoframe_log(const char *level,
> >          case HDMI_INFOFRAME_TYPE_VENDOR:
> >                  hdmi_vendor_any_infoframe_log(level, dev, &frame->vendor);
> >                  break;
> > +       case HDMI_INFOFRAME_TYPE_DRM:
> > +               hdmi_drm_infoframe_log(level, dev, &frame->drm);
> > +               break;
> >          }
> >   }
> >   EXPORT_SYMBOL(hdmi_infoframe_log);
> > diff --git a/include/drm/drm_edid.h b/include/drm/drm_edid.h
> > index 9d3b5b9..d81c5c2 100644
> > --- a/include/drm/drm_edid.h
> > +++ b/include/drm/drm_edid.h
> > @@ -25,6 +25,7 @@
> > 
> >   #include <linux/types.h>
> >   #include <linux/hdmi.h>
> > +#include <drm/drm_mode.h>
> > 
> >   struct drm_device;
> >   struct i2c_adapter;
> > @@ -370,6 +371,10 @@ int drm_av_sync_delay(struct drm_connector *connector,
> >                                     const struct drm_display_mode *mode,
> >                                     enum hdmi_quantization_range rgb_quant_range);
> > 
> > +int
> > +drm_hdmi_infoframe_set_hdr_metadata(struct hdmi_drm_infoframe *frame,
> > +                                   struct hdr_output_metadata *hdr_metadata);
> > +
> >   /**
> >    * drm_eld_mnl - Get ELD monitor name length in bytes.
> >    * @eld: pointer to an eld memory structure with mnl set
> > diff --git a/include/linux/hdmi.h b/include/linux/hdmi.h
> > index 6780476..7edafcf 100644
> > --- a/include/linux/hdmi.h
> > +++ b/include/linux/hdmi.h
> > @@ -47,6 +47,7 @@ enum hdmi_infoframe_type {
> >          HDMI_INFOFRAME_TYPE_AVI = 0x82,
> >          HDMI_INFOFRAME_TYPE_SPD = 0x83,
> >          HDMI_INFOFRAME_TYPE_AUDIO = 0x84,
> > +       HDMI_INFOFRAME_TYPE_DRM = 0x87,
> >   };
> > 
> >   #define HDMI_IEEE_OUI 0x000c03
> > @@ -185,12 +186,37 @@ struct hdmi_avi_infoframe {
> >          unsigned short right_bar;
> >   };
> > 
> > +/* DRM Infoframe as per CTA 861.G spec */
> > +struct hdmi_drm_infoframe {
> > +       enum hdmi_infoframe_type type;
> > +       unsigned char version;
> > +       unsigned char length;
> > +       enum hdmi_eotf eotf;
> > +       enum hdmi_metadata_type metadata_type;
> > +       struct {
> > +               u16 x, y;
> > +       } display_primaries[3];
> > +       struct {
> > +               u16 x, y;
> > +       } white_point;
> > +       u16 max_display_mastering_luminance;
> > +       u16 min_display_mastering_luminance;
> > +       u16 max_cll;
> > +       u16 max_fall;
> > +};
> > +
> >   int hdmi_avi_infoframe_init(struct hdmi_avi_infoframe *frame);
> >   ssize_t hdmi_avi_infoframe_pack(struct hdmi_avi_infoframe *frame, void *buffer,
> >                                  size_t size);
> >   ssize_t hdmi_avi_infoframe_pack_only(const struct hdmi_avi_infoframe *frame,
> >                                       void *buffer, size_t size);
> >   int hdmi_avi_infoframe_check(struct hdmi_avi_infoframe *frame);
> > +int hdmi_drm_infoframe_init(struct hdmi_drm_infoframe *frame);
> > +ssize_t hdmi_drm_infoframe_pack(struct hdmi_drm_infoframe *frame, void *buffer,
> > +                               size_t size);
> > +ssize_t hdmi_drm_infoframe_pack_only(const struct hdmi_drm_infoframe *frame,
> > +                                    void *buffer, size_t size);
> > +int hdmi_drm_infoframe_check(struct hdmi_drm_infoframe *frame);
> > 
> >   enum hdmi_spd_sdi {
> >          HDMI_SPD_SDI_UNKNOWN,
> > @@ -381,6 +407,7 @@ ssize_t hdmi_vendor_infoframe_pack_only(const struct hdmi_vendor_infoframe *fram
> >          struct hdmi_spd_infoframe spd;
> >          union hdmi_vendor_any_infoframe vendor;
> >          struct hdmi_audio_infoframe audio;
> > +       struct hdmi_drm_infoframe drm;
> >   };
> > 
> >   ssize_t hdmi_infoframe_pack(union hdmi_infoframe *frame, void *buffer,
> > --
> > 1.9.1
> > 
> > _______________________________________________
> > dri-devel mailing list
> > dri-devel at lists.freedesktop.org
> > https://lists.freedesktop.org/mailman/listinfo/dri-devel
> > 
> 
> _______________________________________________
> dri-devel mailing list
> dri-devel at lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/dri-devel

-- 
Ville Syrjälä
Intel


More information about the dri-devel mailing list