Blame gst-libs/gst/video/video-color.c

Packit 0652a1
/* GStreamer
Packit 0652a1
 * Copyright (C) <1999> Erik Walthinsen <omega@cse.ogi.edu>
Packit 0652a1
 * Library       <2002> Ronald Bultje <rbultje@ronald.bitfreak.net>
Packit 0652a1
 * Copyright (C) 2007 David A. Schleef <ds@schleef.org>
Packit 0652a1
 *
Packit 0652a1
 * This library is free software; you can redistribute it and/or
Packit 0652a1
 * modify it under the terms of the GNU Library General Public
Packit 0652a1
 * License as published by the Free Software Foundation; either
Packit 0652a1
 * version 2 of the License, or (at your option) any later version.
Packit 0652a1
 *
Packit 0652a1
 * This library is distributed in the hope that it will be useful,
Packit 0652a1
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 0652a1
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 0652a1
 * Library General Public License for more details.
Packit 0652a1
 *
Packit 0652a1
 * You should have received a copy of the GNU Library General Public
Packit 0652a1
 * License along with this library; if not, write to the
Packit 0652a1
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
Packit 0652a1
 * Boston, MA 02110-1301, USA.
Packit 0652a1
 */
Packit 0652a1
Packit 0652a1
#ifdef HAVE_CONFIG_H
Packit 0652a1
#  include "config.h"
Packit 0652a1
#endif
Packit 0652a1
Packit 0652a1
#include <string.h>
Packit 0652a1
#include <stdio.h>
Packit 0652a1
#include <math.h>
Packit 0652a1
Packit 0652a1
#include "video-color.h"
Packit 0652a1
Packit 0652a1
#ifndef GST_DISABLE_GST_DEBUG
Packit 0652a1
#define GST_CAT_DEFAULT ensure_debug_category()
Packit 0652a1
static GstDebugCategory *
Packit 0652a1
ensure_debug_category (void)
Packit 0652a1
{
Packit 0652a1
  static gsize cat_gonce = 0;
Packit 0652a1
Packit 0652a1
  if (g_once_init_enter (&cat_gonce)) {
Packit 0652a1
    gsize cat_done;
Packit 0652a1
Packit 0652a1
    cat_done = (gsize) _gst_debug_category_new ("video-color", 0,
Packit 0652a1
        "video-color object");
Packit 0652a1
Packit 0652a1
    g_once_init_leave (&cat_gonce, cat_done);
Packit 0652a1
  }
Packit 0652a1
Packit 0652a1
  return (GstDebugCategory *) cat_gonce;
Packit 0652a1
}
Packit 0652a1
#else
Packit 0652a1
#define ensure_debug_category() /* NOOP */
Packit 0652a1
#endif /* GST_DISABLE_GST_DEBUG */
Packit 0652a1
Packit 0652a1
typedef struct
Packit 0652a1
{
Packit 0652a1
  const gchar *name;
Packit 0652a1
  GstVideoColorimetry color;
Packit 0652a1
} ColorimetryInfo;
Packit 0652a1
Packit 0652a1
#define MAKE_COLORIMETRY(n,r,m,t,p) { GST_VIDEO_COLORIMETRY_ ##n, \
Packit 0652a1
  { GST_VIDEO_COLOR_RANGE ##r, GST_VIDEO_COLOR_MATRIX_ ##m, \
Packit 0652a1
  GST_VIDEO_TRANSFER_ ##t, GST_VIDEO_COLOR_PRIMARIES_ ##p } }
Packit 0652a1
Packit 0652a1
#define GST_VIDEO_COLORIMETRY_NONAME  NULL
Packit 0652a1
Packit 0652a1
static const ColorimetryInfo colorimetry[] = {
Packit 0652a1
  MAKE_COLORIMETRY (BT601, _16_235, BT601, BT709, SMPTE170M),
Packit 0652a1
  MAKE_COLORIMETRY (BT709, _16_235, BT709, BT709, BT709),
Packit 0652a1
  MAKE_COLORIMETRY (SMPTE240M, _16_235, SMPTE240M, SMPTE240M, SMPTE240M),
Packit 0652a1
  MAKE_COLORIMETRY (SRGB, _0_255, RGB, SRGB, BT709),
Packit 0652a1
  MAKE_COLORIMETRY (BT2020, _16_235, BT2020, BT2020_12, BT2020),
Packit 0652a1
  MAKE_COLORIMETRY (NONAME, _0_255, BT601, UNKNOWN, UNKNOWN),
Packit 0652a1
  MAKE_COLORIMETRY (NONAME, _UNKNOWN, UNKNOWN, UNKNOWN, UNKNOWN),       /* Keep last! */
Packit 0652a1
};
Packit 0652a1
Packit 0652a1
#define DEFAULT_UNKNOWN (G_N_ELEMENTS(colorimetry)-1)
Packit 0652a1
Packit 0652a1
static const ColorimetryInfo *
Packit 0652a1
gst_video_get_colorimetry (const gchar * s)
Packit 0652a1
{
Packit 0652a1
  gint i;
Packit 0652a1
Packit 0652a1
  for (i = 0; colorimetry[i].name; i++) {
Packit 0652a1
    if (g_str_equal (colorimetry[i].name, s))
Packit 0652a1
      return &colorimetry[i];
Packit 0652a1
  }
Packit 0652a1
  return NULL;
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
#define CI_IS_EQUAL(ci,i) (((ci)->range == (i)->range) && \
Packit 0652a1
                        ((ci)->matrix == (i)->matrix) && \
Packit 0652a1
                        ((ci)->transfer == (i)->transfer) && \
Packit 0652a1
                        ((ci)->primaries == (i)->primaries))
Packit 0652a1
Packit 0652a1
#define IS_EQUAL(ci,i) CI_IS_EQUAL(&(ci)->color, (i))
Packit 0652a1
Packit 0652a1
#define IS_UNKNOWN(ci) (IS_EQUAL (&colorimetry[DEFAULT_UNKNOWN], ci))
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_colorimetry_from_string:
Packit 0652a1
 * @cinfo: a #GstVideoColorimetry
Packit 0652a1
 * @color: a colorimetry string
Packit 0652a1
 *
Packit 0652a1
 * Parse the colorimetry string and update @cinfo with the parsed
Packit 0652a1
 * values.
Packit 0652a1
 *
Packit 0652a1
 * Returns: %TRUE if @color points to valid colorimetry info.
Packit 0652a1
 */
Packit 0652a1
gboolean
Packit 0652a1
gst_video_colorimetry_from_string (GstVideoColorimetry * cinfo,
Packit 0652a1
    const gchar * color)
Packit 0652a1
{
Packit 0652a1
  const ColorimetryInfo *ci;
Packit 0652a1
  gboolean res = FALSE;
Packit 0652a1
Packit 0652a1
  if (!color) {
Packit 0652a1
    *cinfo = colorimetry[DEFAULT_UNKNOWN].color;
Packit 0652a1
    res = TRUE;
Packit 0652a1
  } else if ((ci = gst_video_get_colorimetry (color))) {
Packit 0652a1
    *cinfo = ci->color;
Packit 0652a1
    res = TRUE;
Packit 0652a1
  } else {
Packit 0652a1
    gint r, m, t, p;
Packit 0652a1
Packit 0652a1
    if (sscanf (color, "%d:%d:%d:%d", &r, &m, &t, &p) == 4) {
Packit 0652a1
      cinfo->range = r;
Packit 0652a1
      cinfo->matrix = m;
Packit 0652a1
      cinfo->transfer = t;
Packit 0652a1
      cinfo->primaries = p;
Packit 0652a1
      res = TRUE;
Packit 0652a1
    }
Packit 0652a1
  }
Packit 0652a1
  return res;
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_colorimetry_to_string:
Packit 0652a1
 * @cinfo: a #GstVideoColorimetry
Packit 0652a1
 *
Packit 0652a1
 * Make a string representation of @cinfo.
Packit 0652a1
 *
Packit 0652a1
 * Returns: a string representation of @cinfo.
Packit 0652a1
 */
Packit 0652a1
gchar *
Packit 0652a1
gst_video_colorimetry_to_string (const GstVideoColorimetry * cinfo)
Packit 0652a1
{
Packit 0652a1
  gint i;
Packit 0652a1
Packit 0652a1
  for (i = 0; colorimetry[i].name; i++) {
Packit 0652a1
    if (IS_EQUAL (&colorimetry[i], cinfo)) {
Packit 0652a1
      return g_strdup (colorimetry[i].name);
Packit 0652a1
    }
Packit 0652a1
  }
Packit 0652a1
  if (!IS_UNKNOWN (cinfo)) {
Packit 0652a1
    return g_strdup_printf ("%d:%d:%d:%d", cinfo->range, cinfo->matrix,
Packit 0652a1
        cinfo->transfer, cinfo->primaries);
Packit 0652a1
  }
Packit 0652a1
  return NULL;
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_colorimetry_matches:
Packit 0652a1
 * @cinfo: a #GstVideoInfo
Packit 0652a1
 * @color: a colorimetry string
Packit 0652a1
 *
Packit 0652a1
 * Check if the colorimetry information in @info matches that of the
Packit 0652a1
 * string @color.
Packit 0652a1
 *
Packit 0652a1
 * Returns: %TRUE if @color conveys the same colorimetry info as the color
Packit 0652a1
 * information in @info.
Packit 0652a1
 */
Packit 0652a1
gboolean
Packit 0652a1
gst_video_colorimetry_matches (const GstVideoColorimetry * cinfo,
Packit 0652a1
    const gchar * color)
Packit 0652a1
{
Packit 0652a1
  const ColorimetryInfo *ci;
Packit 0652a1
Packit 0652a1
  if ((ci = gst_video_get_colorimetry (color)))
Packit 0652a1
    return IS_EQUAL (ci, cinfo);
Packit 0652a1
Packit 0652a1
  return FALSE;
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_color_range_offsets:
Packit 0652a1
 * @range: a #GstVideoColorRange
Packit 0652a1
 * @info: a #GstVideoFormatInfo
Packit 0652a1
 * @offset: (out caller-allocates) (array fixed-size=4): output offsets
Packit 0652a1
 * @scale: (out caller-allocates) (array fixed-size=4): output scale
Packit 0652a1
 *
Packit 0652a1
 * Compute the offset and scale values for each component of @info. For each
Packit 0652a1
 * component, (c[i] - offset[i]) / scale[i] will scale the component c[i] to the
Packit 0652a1
 * range [0.0 .. 1.0].
Packit 0652a1
 *
Packit 0652a1
 * The reverse operation (c[i] * scale[i]) + offset[i] can be used to convert
Packit 0652a1
 * the component values in range [0.0 .. 1.0] back to their representation in
Packit 0652a1
 * @info and @range.
Packit 0652a1
 */
Packit 0652a1
void
Packit 0652a1
gst_video_color_range_offsets (GstVideoColorRange range,
Packit 0652a1
    const GstVideoFormatInfo * info, gint offset[GST_VIDEO_MAX_COMPONENTS],
Packit 0652a1
    gint scale[GST_VIDEO_MAX_COMPONENTS])
Packit 0652a1
{
Packit 0652a1
  gboolean yuv;
Packit 0652a1
Packit 0652a1
  yuv = GST_VIDEO_FORMAT_INFO_IS_YUV (info);
Packit 0652a1
Packit 0652a1
  switch (range) {
Packit 0652a1
    default:
Packit 0652a1
    case GST_VIDEO_COLOR_RANGE_0_255:
Packit 0652a1
      offset[0] = 0;
Packit 0652a1
      if (yuv) {
Packit 0652a1
        offset[1] = 1 << (info->depth[1] - 1);
Packit 0652a1
        offset[2] = 1 << (info->depth[2] - 1);
Packit 0652a1
      } else {
Packit 0652a1
        offset[1] = 0;
Packit 0652a1
        offset[2] = 0;
Packit 0652a1
      }
Packit 0652a1
      scale[0] = (1 << info->depth[0]) - 1;
Packit 0652a1
      scale[1] = (1 << info->depth[1]) - 1;
Packit 0652a1
      scale[2] = (1 << info->depth[2]) - 1;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_COLOR_RANGE_16_235:
Packit 0652a1
      offset[0] = 1 << (info->depth[0] - 4);
Packit 0652a1
      scale[0] = 219 << (info->depth[0] - 8);
Packit 0652a1
      if (yuv) {
Packit 0652a1
        offset[1] = 1 << (info->depth[1] - 1);
Packit 0652a1
        offset[2] = 1 << (info->depth[2] - 1);
Packit 0652a1
        scale[1] = 224 << (info->depth[1] - 8);
Packit 0652a1
        scale[2] = 224 << (info->depth[2] - 8);
Packit 0652a1
      } else {
Packit 0652a1
        offset[1] = 1 << (info->depth[1] - 4);
Packit 0652a1
        offset[2] = 1 << (info->depth[2] - 4);
Packit 0652a1
        scale[1] = 219 << (info->depth[1] - 8);
Packit 0652a1
        scale[2] = 219 << (info->depth[2] - 8);
Packit 0652a1
      }
Packit 0652a1
      break;
Packit 0652a1
  }
Packit 0652a1
  /* alpha channel is always full range */
Packit 0652a1
  offset[3] = 0;
Packit 0652a1
  scale[3] = (1 << info->depth[3]) - 1;
Packit 0652a1
Packit 0652a1
  GST_DEBUG ("scale: %d %d %d %d", scale[0], scale[1], scale[2], scale[3]);
Packit 0652a1
  GST_DEBUG ("offset: %d %d %d %d", offset[0], offset[1], offset[2], offset[3]);
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_colorimetry_is_equal:
Packit 0652a1
 * @cinfo: a #GstVideoColorimetry
Packit 0652a1
 * @other: another #GstVideoColorimetry
Packit 0652a1
 *
Packit 0652a1
 * Compare the 2 colorimetry sets for equality
Packit 0652a1
 *
Packit 0652a1
 * Returns: %TRUE if @cinfo and @other are equal.
Packit 0652a1
 *
Packit 0652a1
 * Since: 1.6
Packit 0652a1
 */
Packit 0652a1
gboolean
Packit 0652a1
gst_video_colorimetry_is_equal (const GstVideoColorimetry * cinfo,
Packit 0652a1
    const GstVideoColorimetry * other)
Packit 0652a1
{
Packit 0652a1
  g_return_val_if_fail (cinfo != NULL, FALSE);
Packit 0652a1
  g_return_val_if_fail (other != NULL, FALSE);
Packit 0652a1
Packit 0652a1
  return CI_IS_EQUAL (cinfo, other);
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
#define WP_C    0.31006, 0.31616
Packit 0652a1
#define WP_D65  0.31271, 0.32902
Packit 0652a1
#define WP_CENTRE (1/3), (1/3)
Packit 0652a1
#define WP_WHITE 0.314, 0.351
Packit 0652a1
Packit 0652a1
static const GstVideoColorPrimariesInfo color_primaries[] = {
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_UNKNOWN, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0, 0.0},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_BT709, WP_D65, 0.64, 0.33, 0.30, 0.60, 0.15, 0.06},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_BT470M, WP_C, 0.67, 0.33, 0.21, 0.71, 0.14, 0.08},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_BT470BG, WP_D65, 0.64, 0.33, 0.29, 0.60, 0.15,
Packit 0652a1
      0.06},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_SMPTE170M, WP_D65, 0.63, 0.34, 0.31, 0.595, 0.155,
Packit 0652a1
      0.07},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_SMPTE240M, WP_D65, 0.63, 0.34, 0.31, 0.595, 0.155,
Packit 0652a1
      0.07},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_FILM, WP_C, 0.681, 0.319, 0.243, 0.692, 0.145,
Packit 0652a1
      0.049},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_BT2020, WP_D65, 0.708, 0.292, 0.170, 0.797, 0.131,
Packit 0652a1
      0.046},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_ADOBERGB, WP_D65, 0.64, 0.33, 0.21, 0.71, 0.15,
Packit 0652a1
      0.06},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_SMPTEST428, WP_CENTRE, 1.0, 0.0, 0.0, 1.0, 0.0,
Packit 0652a1
      0.0},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_SMPTERP431, WP_WHITE, 0.68, 0.32, 0.265, 0.69,
Packit 0652a1
      0.15, 0.06},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_SMPTEEG432, WP_D65, 0.68, 0.32, 0.265, 0.69, 0.15,
Packit 0652a1
      0.06},
Packit 0652a1
  {GST_VIDEO_COLOR_PRIMARIES_EBU3213, WP_D65, 0.63, 0.34, 0.295, 0.605, 0.155,
Packit 0652a1
      0.077},
Packit 0652a1
};
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_color_primaries_get_info:
Packit 0652a1
 * @primaries: a #GstVideoColorPrimaries
Packit 0652a1
 *
Packit 0652a1
 * Get information about the chromaticity coordinates of @primaries.
Packit 0652a1
 *
Packit 0652a1
 * Returns: a #GstVideoColorPrimariesInfo for @primaries.
Packit 0652a1
 *
Packit 0652a1
 * Since: 1.6
Packit 0652a1
 */
Packit 0652a1
const GstVideoColorPrimariesInfo *
Packit 0652a1
gst_video_color_primaries_get_info (GstVideoColorPrimaries primaries)
Packit 0652a1
{
Packit 0652a1
  g_return_val_if_fail ((gint) primaries <
Packit 0652a1
      G_N_ELEMENTS (color_primaries), NULL);
Packit 0652a1
Packit 0652a1
  return &color_primaries[primaries];
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_color_matrix_get_Kr_Kb:
Packit 0652a1
 * @matrix: a #GstVideoColorMatrix
Packit 0652a1
 * @Kr: (out): result red channel coefficient
Packit 0652a1
 * @Kb: (out): result blue channel coefficient
Packit 0652a1
 *
Packit 0652a1
 * Get the coefficients used to convert between Y'PbPr and R'G'B' using @matrix.
Packit 0652a1
 *
Packit 0652a1
 * When:
Packit 0652a1
 *
Packit 0652a1
 * |[
Packit 0652a1
 *   0.0 <= [Y',R',G',B'] <= 1.0)
Packit 0652a1
 *   (-0.5 <= [Pb,Pr] <= 0.5)
Packit 0652a1
 * ]|
Packit 0652a1
 *
Packit 0652a1
 * the general conversion is given by:
Packit 0652a1
 *
Packit 0652a1
 * |[
Packit 0652a1
 *   Y' = Kr*R' + (1-Kr-Kb)*G' + Kb*B'
Packit 0652a1
 *   Pb = (B'-Y')/(2*(1-Kb))
Packit 0652a1
 *   Pr = (R'-Y')/(2*(1-Kr))
Packit 0652a1
 * ]|
Packit 0652a1
 *
Packit 0652a1
 * and the other way around:
Packit 0652a1
 *
Packit 0652a1
 * |[
Packit 0652a1
 *   R' = Y' + Cr*2*(1-Kr)
Packit 0652a1
 *   G' = Y' - Cb*2*(1-Kb)*Kb/(1-Kr-Kb) - Cr*2*(1-Kr)*Kr/(1-Kr-Kb)
Packit 0652a1
 *   B' = Y' + Cb*2*(1-Kb)
Packit 0652a1
 * ]|
Packit 0652a1
 *
Packit 0652a1
 * Returns: TRUE if @matrix was a YUV color format and @Kr and @Kb contain valid
Packit 0652a1
 *    values.
Packit 0652a1
 *
Packit 0652a1
 * Since: 1.6
Packit 0652a1
 */
Packit 0652a1
gboolean
Packit 0652a1
gst_video_color_matrix_get_Kr_Kb (GstVideoColorMatrix matrix, gdouble * Kr,
Packit 0652a1
    gdouble * Kb)
Packit 0652a1
{
Packit 0652a1
  gboolean res = TRUE;
Packit 0652a1
Packit 0652a1
  switch (matrix) {
Packit 0652a1
      /* RGB */
Packit 0652a1
    default:
Packit 0652a1
    case GST_VIDEO_COLOR_MATRIX_RGB:
Packit 0652a1
      res = FALSE;
Packit 0652a1
      break;
Packit 0652a1
      /* YUV */
Packit 0652a1
    case GST_VIDEO_COLOR_MATRIX_FCC:
Packit 0652a1
      *Kr = 0.30;
Packit 0652a1
      *Kb = 0.11;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_COLOR_MATRIX_BT709:
Packit 0652a1
      *Kr = 0.2126;
Packit 0652a1
      *Kb = 0.0722;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_COLOR_MATRIX_BT601:
Packit 0652a1
      *Kr = 0.2990;
Packit 0652a1
      *Kb = 0.1140;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_COLOR_MATRIX_SMPTE240M:
Packit 0652a1
      *Kr = 0.212;
Packit 0652a1
      *Kb = 0.087;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_COLOR_MATRIX_BT2020:
Packit 0652a1
      *Kr = 0.2627;
Packit 0652a1
      *Kb = 0.0593;
Packit 0652a1
      break;
Packit 0652a1
  }
Packit 0652a1
  GST_DEBUG ("matrix: %d, Kr %f, Kb %f", matrix, *Kr, *Kb);
Packit 0652a1
Packit 0652a1
  return res;
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_color_transfer_encode:
Packit 0652a1
 * @func: a #GstVideoTransferFunction
Packit 0652a1
 * @val: a value
Packit 0652a1
 *
Packit 0652a1
 * Convert @val to its gamma encoded value.
Packit 0652a1
 *
Packit 0652a1
 * For a linear value L in the range [0..1], conversion to the non-linear
Packit 0652a1
 * (gamma encoded) L' is in general performed with a power function like:
Packit 0652a1
 *
Packit 0652a1
 * |[
Packit 0652a1
 *    L' = L ^ (1 / gamma)
Packit 0652a1
 * ]|
Packit 0652a1
 *
Packit 0652a1
 * Depending on @func, different formulas might be applied. Some formulas
Packit 0652a1
 * encode a linear segment in the lower range.
Packit 0652a1
 *
Packit 0652a1
 * Returns: the gamme encoded value of @val
Packit 0652a1
 *
Packit 0652a1
 * Since: 1.6
Packit 0652a1
 */
Packit 0652a1
gdouble
Packit 0652a1
gst_video_color_transfer_encode (GstVideoTransferFunction func, gdouble val)
Packit 0652a1
{
Packit 0652a1
  gdouble res;
Packit 0652a1
Packit 0652a1
  switch (func) {
Packit 0652a1
    case GST_VIDEO_TRANSFER_UNKNOWN:
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA10:
Packit 0652a1
    default:
Packit 0652a1
      res = val;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA18:
Packit 0652a1
      res = pow (val, 1.0 / 1.8);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA20:
Packit 0652a1
      res = pow (val, 1.0 / 2.0);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA22:
Packit 0652a1
      res = pow (val, 1.0 / 2.2);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_BT709:
Packit 0652a1
      if (val < 0.018)
Packit 0652a1
        res = 4.5 * val;
Packit 0652a1
      else
Packit 0652a1
        res = 1.099 * pow (val, 0.45) - 0.099;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_SMPTE240M:
Packit 0652a1
      if (val < 0.0228)
Packit 0652a1
        res = val * 4.0;
Packit 0652a1
      else
Packit 0652a1
        res = 1.1115 * pow (val, 0.45) - 0.1115;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_SRGB:
Packit 0652a1
      if (val <= 0.0031308)
Packit 0652a1
        res = 12.92 * val;
Packit 0652a1
      else
Packit 0652a1
        res = 1.055 * pow (val, 1.0 / 2.4) - 0.055;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA28:
Packit 0652a1
      res = pow (val, 1 / 2.8);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_LOG100:
Packit 0652a1
      if (val < 0.01)
Packit 0652a1
        res = 0.0;
Packit 0652a1
      else
Packit 0652a1
        res = 1.0 + log10 (val) / 2.0;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_LOG316:
Packit 0652a1
      if (val < 0.0031622777)
Packit 0652a1
        res = 0.0;
Packit 0652a1
      else
Packit 0652a1
        res = 1.0 + log10 (val) / 2.5;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_BT2020_12:
Packit 0652a1
      if (val < 0.0181)
Packit 0652a1
        res = 4.5 * val;
Packit 0652a1
      else
Packit 0652a1
        res = 1.0993 * pow (val, 0.45) - 0.0993;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_ADOBERGB:
Packit 0652a1
      res = pow (val, 1.0 / 2.19921875);
Packit 0652a1
      break;
Packit 0652a1
  }
Packit 0652a1
  return res;
Packit 0652a1
}
Packit 0652a1
Packit 0652a1
/**
Packit 0652a1
 * gst_video_color_transfer_decode:
Packit 0652a1
 * @func: a #GstVideoTransferFunction
Packit 0652a1
 * @val: a value
Packit 0652a1
 *
Packit 0652a1
 * Convert @val to its gamma decoded value. This is the inverse operation of
Packit 0652a1
 * @gst_video_color_transfer_encode().
Packit 0652a1
 *
Packit 0652a1
 * For a non-linear value L' in the range [0..1], conversion to the linear
Packit 0652a1
 * L is in general performed with a power function like:
Packit 0652a1
 *
Packit 0652a1
 * |[
Packit 0652a1
 *    L = L' ^ gamma
Packit 0652a1
 * ]|
Packit 0652a1
 *
Packit 0652a1
 * Depending on @func, different formulas might be applied. Some formulas
Packit 0652a1
 * encode a linear segment in the lower range.
Packit 0652a1
 *
Packit 0652a1
 * Returns: the gamme decoded value of @val
Packit 0652a1
 *
Packit 0652a1
 * Since: 1.6
Packit 0652a1
 */
Packit 0652a1
gdouble
Packit 0652a1
gst_video_color_transfer_decode (GstVideoTransferFunction func, gdouble val)
Packit 0652a1
{
Packit 0652a1
  gdouble res;
Packit 0652a1
Packit 0652a1
  switch (func) {
Packit 0652a1
    case GST_VIDEO_TRANSFER_UNKNOWN:
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA10:
Packit 0652a1
    default:
Packit 0652a1
      res = val;
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA18:
Packit 0652a1
      res = pow (val, 1.8);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA20:
Packit 0652a1
      res = pow (val, 2.0);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA22:
Packit 0652a1
      res = pow (val, 2.2);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_BT709:
Packit 0652a1
      if (val < 0.081)
Packit 0652a1
        res = val / 4.5;
Packit 0652a1
      else
Packit 0652a1
        res = pow ((val + 0.099) / 1.099, 1.0 / 0.45);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_SMPTE240M:
Packit 0652a1
      if (val < 0.0913)
Packit 0652a1
        res = val / 4.0;
Packit 0652a1
      else
Packit 0652a1
        res = pow ((val + 0.1115) / 1.1115, 1.0 / 0.45);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_SRGB:
Packit 0652a1
      if (val <= 0.04045)
Packit 0652a1
        res = val / 12.92;
Packit 0652a1
      else
Packit 0652a1
        res = pow ((val + 0.055) / 1.055, 2.4);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_GAMMA28:
Packit 0652a1
      res = pow (val, 2.8);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_LOG100:
Packit 0652a1
      if (val == 0.0)
Packit 0652a1
        res = 0.0;
Packit 0652a1
      else
Packit 0652a1
        res = pow (10.0, 2.0 * (val - 1.0));
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_LOG316:
Packit 0652a1
      if (val == 0.0)
Packit 0652a1
        res = 0.0;
Packit 0652a1
      else
Packit 0652a1
        res = pow (10.0, 2.5 * (val - 1.0));
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_BT2020_12:
Packit 0652a1
      if (val < 0.08145)
Packit 0652a1
        res = val / 4.5;
Packit 0652a1
      else
Packit 0652a1
        res = pow ((val + 0.0993) / 1.0993, 1.0 / 0.45);
Packit 0652a1
      break;
Packit 0652a1
    case GST_VIDEO_TRANSFER_ADOBERGB:
Packit 0652a1
      res = pow (val, 2.19921875);
Packit 0652a1
      break;
Packit 0652a1
  }
Packit 0652a1
  return res;
Packit 0652a1
}