weston/libweston/color-operations.c

237 lines
6.6 KiB
C
Raw Normal View History

/*
* Copyright 2025 Collabora, Ltd.
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice (including the
* next paragraph) shall be included in all copies or substantial
* portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#include "config.h"
#include "color.h"
#include "color-properties.h"
#include "color-operations.h"
#include "shared/helpers.h"
#include "shared/weston-assert.h"
static float
linpow(float x, const union weston_color_curve_parametric_chan_data *p)
{
/* See WESTON_COLOR_CURVE_PARAMETRIC_TYPE_LINPOW for details about LINPOW. */
/* LINPOW uses mirroring for negative input values. */
float abs_x = fabsf(x);
if (abs_x >= p->d)
return copysign(pow((p->a * abs_x) + p->b, p->g), x);
return p->c * x;
}
static void
sample_linpow(const union weston_color_curve_parametric_data *p,
bool clamp_input, const struct weston_vec3f *in,
struct weston_vec3f *out, size_t len)
{
struct weston_vec3f tmp;
size_t i;
for (i = 0; i < len; i++) {
tmp = in[i];
if (clamp_input)
tmp = weston_v3f_clamp(tmp, 0.0f, 1.0f);
out[i] = WESTON_VEC3F(linpow(tmp.x, &p->chan[0]),
linpow(tmp.y, &p->chan[1]),
linpow(tmp.z, &p->chan[2]));
}
}
static float
powlin(float x, const union weston_color_curve_parametric_chan_data *p)
{
/* See WESTON_COLOR_CURVE_PARAMETRIC_TYPE_POWLIN for details about POWLIN. */
/* POWLIN uses mirroring for negative input values. */
float abs_x = fabsf(x);
if (abs_x >= p->d)
return copysign(p->a * pow(abs_x, p->g) + p->b, x);
return p->c * x;
}
static void
sample_powlin(const union weston_color_curve_parametric_data *p,
bool clamp_input, const struct weston_vec3f *in,
struct weston_vec3f *out, size_t len)
{
struct weston_vec3f tmp;
size_t i;
for (i = 0; i < len; i++) {
tmp = in[i];
if (clamp_input)
tmp = weston_v3f_clamp(tmp, 0.0f, 1.0f);
out[i] = WESTON_VEC3F(powlin(tmp.x, &p->chan[0]),
powlin(tmp.y, &p->chan[1]),
powlin(tmp.z, &p->chan[2]));
}
}
static void
sample_parametric(struct weston_compositor *compositor,
const struct weston_color_curve_parametric *param,
const struct weston_vec3f *in,
struct weston_vec3f *out,
size_t len)
{
switch (param->type) {
case WESTON_COLOR_CURVE_PARAMETRIC_TYPE_LINPOW:
sample_linpow(&param->params, param->clamped_input, in, out, len);
return;
case WESTON_COLOR_CURVE_PARAMETRIC_TYPE_POWLIN:
sample_powlin(&param->params, param->clamped_input, in, out, len);
return;
}
weston_assert_not_reached(compositor, "unknown parametric color curve");
}
static float
perceptual_quantizer(float x)
{
float aux, c1, c2, c3, m1_inv, m2_inv;
m1_inv = 1.0 / 0.1593017578125;
m2_inv = 1.0 / 78.84375;
c1 = 0.8359375;
c2 = 18.8515625;
c3 = 18.6875;
aux = pow(x, m2_inv);
/* Normalized result. We don't take into consideration the luminance
* levels, as we don't receive the input as nits, but normalized in the
* [0, 1] range. */
return pow(MAX(aux - c1, 0.0) / (c2 - c3 * aux), m1_inv);
}
static float
perceptual_quantizer_inverse(float x)
{
float aux, c1, c2, c3, m1, m2;
m1 = 0.1593017578125;
m2 = 78.84375;
c1 = 0.8359375;
c2 = 18.8515625;
c3 = 18.6875;
aux = pow(x, m1);
/* Normalized result. We don't take into consideration the luminance
* levels, as we don't receive the input as nits, but normalized in the
* [0, 1] range. */
return pow((c1 + c2 * aux) / (1.0 + c3 * aux), m2);
}
static void
sample_pq(enum weston_tf_direction tf_direction,
const struct weston_vec3f *in,
struct weston_vec3f *out,
size_t len)
{
struct weston_vec3f tmp;
size_t i;
switch (tf_direction) {
case WESTON_FORWARD_TF:
for (i = 0; i < len; i++) {
tmp = weston_v3f_clamp(in[i], 0.0f, 1.0f);
out[i] = WESTON_VEC3F(perceptual_quantizer(tmp.x),
perceptual_quantizer(tmp.y),
perceptual_quantizer(tmp.z));
}
break;
case WESTON_INVERSE_TF:
for (i = 0; i < len; i++) {
tmp = weston_v3f_clamp(in[i], 0.0f, 1.0f);
out[i] = WESTON_VEC3F(perceptual_quantizer_inverse(tmp.x),
perceptual_quantizer_inverse(tmp.y),
perceptual_quantizer_inverse(tmp.z));
}
break;
}
}
/**
* Evaluate a color curve on an array
*
* This handles the parametric curves (LINPOW, POWLIN, etc) and enumerated color
* curves. Others result in failure.
*
* @param compositor The Weston compositor
* @param curve The color curve to evaluate
* @param in The input array of length @c len .
* @param out The output array of length @c len .
* @param len The in and out arrays' length.
*/
void
weston_color_curve_sample(struct weston_compositor *compositor,
struct weston_color_curve *curve,
const struct weston_vec3f *in,
struct weston_vec3f *out,
size_t len)
{
struct weston_color_curve_parametric parametric;
bool ret;
switch(curve->type) {
case WESTON_COLOR_CURVE_TYPE_ENUM:
/**
* If the TF of the enum curve is implemented, sample from that.
* Otherwise, fallback to a parametric curve and we'll handle
* that below.
*/
switch (curve->u.enumerated.tf.info->tf) {
case WESTON_TF_ST2084_PQ:
sample_pq(curve->u.enumerated.tf_direction, in, out, len);
return;
default:
ret = weston_color_curve_enum_get_parametric(compositor,
&curve->u.enumerated,
&parametric);
weston_assert_true(compositor, ret);
sample_parametric(compositor, &parametric, in, out, len);
return;
}
case WESTON_COLOR_CURVE_TYPE_PARAMETRIC:
sample_parametric(compositor, &curve->u.parametric, in, out, len);
return;
case WESTON_COLOR_CURVE_TYPE_IDENTITY:
weston_assert_not_reached(compositor,
"no need to sample identity");
case WESTON_COLOR_CURVE_TYPE_LUT_3x1D:
weston_assert_not_reached(compositor,
"function does not handle LUT 3x1D");
}
weston_assert_not_reached(compositor, "unknown color curve");
}