mirror of
https://gitlab.freedesktop.org/wayland/weston.git
synced 2025-12-20 04:40:07 +01:00
Future development will need to evaluate pipelines with curves and matrices. Such pipelines naturally operate on vec3, as matrices cannot be operated one channel at a time. Make weston_color_curve_sample() operate on arrays of vec3. Its currently only caller, weston_color_curve_to_3x1D_LUT(), is modified to employ a temporary array for the API impedance mismatch. This workaround will be removed later as weston_color_curve_to_3x1D_LUT() itself will be converted to operate on vec3 arrays. weston_v3f_array_to_planar() documentation was generated with AI. weston_color_curve_sample() is restructured a little bit, attempting to make it simpler to read. color-operations.h gets the #includes needed to make it self-standing. Assisted-by: Github Copilot (Claude Sonnet 3.5) Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.com>
1114 lines
30 KiB
C
1114 lines
30 KiB
C
/*
|
|
* Copyright 2019 Sebastian Wick
|
|
* Copyright 2021-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 <assert.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
#include <unistd.h>
|
|
#include <sys/mman.h>
|
|
#include <errno.h>
|
|
#include <string.h>
|
|
|
|
#include <libweston/libweston.h>
|
|
#include <libweston/weston-log.h>
|
|
#include <libweston/linalg-3.h>
|
|
|
|
#include "color.h"
|
|
#include "color-operations.h"
|
|
#include "color-properties.h"
|
|
#include "id-number-allocator.h"
|
|
#include "libweston-internal.h"
|
|
#include "shared/string-helpers.h"
|
|
#include "shared/helpers.h"
|
|
#include "shared/weston-assert.h"
|
|
#include "shared/xalloc.h"
|
|
#include "shared/weston-assert.h"
|
|
|
|
/**
|
|
* Increase reference count of the color profile object
|
|
*
|
|
* \param cprof The color profile. NULL is accepted too.
|
|
* \return cprof.
|
|
*/
|
|
WL_EXPORT struct weston_color_profile *
|
|
weston_color_profile_ref(struct weston_color_profile *cprof)
|
|
{
|
|
if (!cprof)
|
|
return NULL;
|
|
|
|
assert(cprof->ref_count > 0);
|
|
cprof->ref_count++;
|
|
return cprof;
|
|
}
|
|
|
|
/**
|
|
* Decrease reference count and potentially destroy the color profile object
|
|
*
|
|
* \param cprof The color profile. NULL is accepted too.
|
|
*/
|
|
WL_EXPORT void
|
|
weston_color_profile_unref(struct weston_color_profile *cprof)
|
|
{
|
|
if (!cprof)
|
|
return;
|
|
|
|
assert(cprof->ref_count > 0);
|
|
if (--cprof->ref_count > 0)
|
|
return;
|
|
|
|
weston_idalloc_put_id(cprof->cm->compositor->color_profile_id_generator,
|
|
cprof->id);
|
|
|
|
cprof->cm->destroy_color_profile(cprof);
|
|
}
|
|
|
|
/**
|
|
* Get color profile description
|
|
*
|
|
* A description of the profile is meant for human readable logs.
|
|
*
|
|
* \param cprof The color profile, NULL is accepted too.
|
|
* \returns The color profile description, valid as long as the
|
|
* color profile itself is.
|
|
*/
|
|
WL_EXPORT const char *
|
|
weston_color_profile_get_description(struct weston_color_profile *cprof)
|
|
{
|
|
if (cprof)
|
|
return cprof->description;
|
|
else
|
|
return "(untagged)";
|
|
}
|
|
|
|
/**
|
|
* Get color profile detailed description
|
|
*
|
|
* A detailed, multi-line description of the profile is meant for
|
|
* human readable logs.
|
|
*
|
|
* \param cprof The color profile, NULL is accepted too.
|
|
* \returns The detailed description. Must be free()'d.
|
|
*/
|
|
WL_EXPORT char *
|
|
weston_color_profile_get_details(struct weston_color_profile *cprof)
|
|
{
|
|
if (!cprof)
|
|
return xstrdup("NULL profile.\n");
|
|
|
|
if (cprof->cm->print_color_profile_details)
|
|
return cprof->cm->print_color_profile_details(cprof);
|
|
else
|
|
return xstrdup("No details.\n");
|
|
}
|
|
|
|
/**
|
|
* Initializes a newly allocated color profile object
|
|
*
|
|
* This is used only by color managers. They sub-class weston_color_profile.
|
|
*
|
|
* The reference count starts at 1.
|
|
*
|
|
* To destroy a weston_color_profile, use weston_color_profile_unref().
|
|
*/
|
|
WL_EXPORT void
|
|
weston_color_profile_init(struct weston_color_profile *cprof,
|
|
struct weston_color_manager *cm)
|
|
{
|
|
cprof->cm = cm;
|
|
cprof->ref_count = 1;
|
|
cprof->id = weston_idalloc_get_id(cm->compositor->color_profile_id_generator);
|
|
}
|
|
|
|
static void
|
|
weston_color_gamut_fprint(FILE *fp,
|
|
const char *indent,
|
|
const struct weston_color_gamut *g)
|
|
{
|
|
static const char *chan[] = { "R", "G", "B" };
|
|
unsigned i;
|
|
|
|
for (i = 0; i < 3; i++) {
|
|
fprintf(fp, "%s %s = (%.4f, %.4f)\n",
|
|
indent, chan[i], g->primary[i].x, g->primary[i].y);
|
|
}
|
|
|
|
fprintf(fp, "%s WP = (%.4f, %.4f)\n",
|
|
indent, g->white_point.x, g->white_point.y);
|
|
}
|
|
|
|
/**
|
|
* Print color profile parameters to string.
|
|
*
|
|
* \param params The parameters of the color profile.
|
|
* \param ident Indentation to add before each line of the return'ed string.
|
|
* \returns The color profile parameters as string. Callers must free() it.
|
|
*/
|
|
WL_EXPORT char *
|
|
weston_color_profile_params_to_str(const struct weston_color_profile_params *params,
|
|
const char *ident)
|
|
{
|
|
FILE *fp;
|
|
char *str;
|
|
size_t size;
|
|
unsigned int i;
|
|
|
|
fp = open_memstream(&str, &size);
|
|
abort_oom_if_null(fp);
|
|
|
|
fprintf(fp, "%sprimaries (CIE xy):\n", ident);
|
|
weston_color_gamut_fprint(fp, ident, ¶ms->primaries);
|
|
|
|
if (params->primaries_info)
|
|
fprintf(fp, "%sprimaries named: %s\n", ident, params->primaries_info->desc);
|
|
|
|
fprintf(fp, "%stransfer function: %s\n", ident, params->tf.info->desc);
|
|
|
|
if (params->tf.info->count_parameters > 0) {
|
|
fprintf(fp, "%s params:", ident);
|
|
for (i = 0; i < params->tf.info->count_parameters; i++)
|
|
fprintf(fp, " %.4f", params->tf.params[i]);
|
|
fprintf(fp, "\n");
|
|
}
|
|
|
|
fprintf(fp, "%sluminance: [%.3f, %.2f], ref white %.2f (cd/m²)\n", ident, params->min_luminance,
|
|
params->max_luminance,
|
|
params->reference_white_luminance);
|
|
|
|
fprintf(fp, "%starget primaries (CIE xy):\n", ident);
|
|
weston_color_gamut_fprint(fp, ident, ¶ms->target_primaries);
|
|
|
|
if (params->target_min_luminance >= 0.0f && params->target_max_luminance >= 0.0f)
|
|
fprintf(fp, "%starget luminance: [%.3f, %.2f] (cd/m²)\n", ident, params->target_min_luminance,
|
|
params->target_max_luminance);
|
|
|
|
if (params->maxCLL >= 0.0f)
|
|
fprintf(fp, "%smax cll: %.2f (cd/m²)\n", ident, params->maxCLL);
|
|
|
|
if (params->maxFALL >= 0.0f)
|
|
fprintf(fp, "%smax fall: %.2f (cd/m²)\n", ident, params->maxFALL);
|
|
|
|
fclose(fp);
|
|
return str;
|
|
}
|
|
|
|
/**
|
|
* Given an enumerated color curve, returns an equivalent parametric curve.
|
|
*
|
|
* \param compositor The compositor instance.
|
|
* \param curve The enumerated color curve.
|
|
* \param out Where this stores the parametric curve.
|
|
* \return True on success, false otherwise.
|
|
*/
|
|
WL_EXPORT bool
|
|
weston_color_curve_enum_get_parametric(struct weston_compositor *compositor,
|
|
const struct weston_color_curve_enum *curve,
|
|
struct weston_color_curve_parametric *out)
|
|
{
|
|
unsigned int i;
|
|
|
|
memset(out, 0, sizeof(*out));
|
|
|
|
/* This one is special, the only parametric TF we currently have. */
|
|
if (curve->tf.info->tf == WESTON_TF_POWER) {
|
|
out->type = WESTON_COLOR_CURVE_PARAMETRIC_TYPE_LINPOW;
|
|
out->clamped_input = false;
|
|
for (i = 0; i < 3; i++) {
|
|
float exp = curve->tf.params[0];
|
|
/* LINPOW with such params matches pure power-law */
|
|
out->params.chan[i].g = (curve->tf_direction == WESTON_FORWARD_TF) ?
|
|
exp : 1.0f / exp;
|
|
out->params.chan[i].a = 1.0;
|
|
out->params.chan[i].b = 0.0;
|
|
out->params.chan[i].c = 1.0;
|
|
out->params.chan[i].d = 0.0;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/* No other TF's have params. */
|
|
weston_assert_uint_eq(compositor, curve->tf.info->count_parameters, 0);
|
|
|
|
if (!curve->tf.info->curve_params_valid)
|
|
return false;
|
|
|
|
if (curve->tf_direction == WESTON_FORWARD_TF)
|
|
*out = curve->tf.info->curve;
|
|
else
|
|
*out = curve->tf.info->inverse_curve;
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool
|
|
curve_to_lut_has_good_precision(struct weston_color_curve *curve)
|
|
{
|
|
struct weston_color_curve_enum *e = &curve->u.enumerated;
|
|
struct weston_color_curve_parametric *p = &curve->u.parametric;
|
|
float g;
|
|
unsigned int i;
|
|
|
|
if (curve->type == WESTON_COLOR_CURVE_TYPE_ENUM) {
|
|
if (e->tf_direction == WESTON_INVERSE_TF) {
|
|
if (e->tf.info->tf == WESTON_TF_ST2084_PQ ||
|
|
e->tf.info->tf == WESTON_TF_GAMMA22 ||
|
|
e->tf.info->tf == WESTON_TF_GAMMA28) {
|
|
/**
|
|
* These have bad precision in the indirect
|
|
* direction.
|
|
*/
|
|
return false;
|
|
}
|
|
|
|
if (e->tf.info->tf == WESTON_TF_POWER) {
|
|
/**
|
|
* Same as the above, but for parametric
|
|
* power-law transfer function. If g > 1.0
|
|
* it would result in bad precision.
|
|
*/
|
|
g = e->tf.params[0];
|
|
if (g > 1.0f)
|
|
return false;
|
|
}
|
|
} else {
|
|
if (e->tf.info->tf == WESTON_TF_POWER) {
|
|
/**
|
|
* For parametric power-law transfer function
|
|
* in the forward direction, g < 1.0 would
|
|
* result in bad precision.
|
|
*/
|
|
g = e->tf.params[0];
|
|
if (g < 1.0f)
|
|
return false;
|
|
}
|
|
}
|
|
} else if (curve->type == WESTON_COLOR_CURVE_TYPE_PARAMETRIC) {
|
|
switch(p->type) {
|
|
case WESTON_COLOR_CURVE_PARAMETRIC_TYPE_LINPOW:
|
|
case WESTON_COLOR_CURVE_PARAMETRIC_TYPE_POWLIN:
|
|
/**
|
|
* Both LINPOW and POWLIN have bad precision if g < 1.0.
|
|
*/
|
|
for (i = 0; i < 3; i++) {
|
|
if (p->params.chan[i].g < 1.0f)
|
|
return false;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/**
|
|
* Given a xform and an enum corresponding to one of its curves (pre or post),
|
|
* returns a 3x1D LUT that corresponds to such curve. This only works for
|
|
* transformations such that xform->steps_valid.
|
|
*
|
|
* The 3x1D LUT returned looks like this: the first lut_size elements compose
|
|
* the LUT for the R channel, the next lut_size elements compose the LUT for the
|
|
* G channel and the last lut_size elements compose the LUT for the B channel.
|
|
*
|
|
* @param compositor The Weston compositor.
|
|
* @param xform The color transformation that owns the curve.
|
|
* @param step The curve step (pre or post) from the xform.
|
|
* @param precision_mode If WESTON_COLOR_PRECISION_CAREFUL, this fails if we
|
|
* detect that we can't create a LUT from the curve without resulting in
|
|
* precision issues. If WESTON_COLOR_PRECISION_CARELESS, we simply log a warning.
|
|
* @param lut_size The size of each LUT.
|
|
* @param err_msg Set on failure, untouched otherwise. Must be free()'d by caller.
|
|
* @return NULL on failure, the 3x1D LUT on success.
|
|
*/
|
|
WL_EXPORT float *
|
|
weston_color_curve_to_3x1D_LUT(struct weston_compositor *compositor,
|
|
struct weston_color_transform *xform,
|
|
enum weston_color_curve_step step,
|
|
enum weston_color_precision precision_mode,
|
|
uint32_t lut_size, char **err_msg)
|
|
{
|
|
struct weston_color_curve *curve;
|
|
float divider = lut_size - 1;
|
|
const char *step_str;
|
|
float *lut;
|
|
struct weston_vec3f *tmp;
|
|
unsigned int i;
|
|
|
|
switch(step) {
|
|
case WESTON_COLOR_CURVE_STEP_PRE:
|
|
curve = &xform->pre_curve;
|
|
step_str = "pre";
|
|
break;
|
|
case WESTON_COLOR_CURVE_STEP_POST:
|
|
curve = &xform->post_curve;
|
|
step_str = "post";
|
|
break;
|
|
default:
|
|
weston_assert_not_reached(compositor, "unknown curve step");
|
|
}
|
|
|
|
if (!xform->steps_valid) {
|
|
str_printf(err_msg, "can't create LUT from xform (id %u) %s-curve, as the " \
|
|
"xform don't have valid steps",
|
|
xform->id, step_str);
|
|
return NULL;
|
|
}
|
|
|
|
if (!curve_to_lut_has_good_precision(curve)) {
|
|
if (precision_mode == WESTON_COLOR_PRECISION_CAREFUL) {
|
|
str_printf(err_msg, "can't create color LUT from xform (id %u) " \
|
|
"%s-curve, it would result in bad precision",
|
|
xform->id, step_str);
|
|
return NULL;
|
|
}
|
|
weston_log("WARNING: converting xform (id %u) %s-curve to 3x1D LUT should probably " \
|
|
"result in bad precision\n", xform->id, step_str);
|
|
}
|
|
|
|
lut = calloc(lut_size, 3 * sizeof *lut);
|
|
tmp = calloc(lut_size, sizeof *tmp);
|
|
if (!lut || !tmp) {
|
|
/* lut_size could be big. */
|
|
free(lut);
|
|
free(tmp);
|
|
str_printf(err_msg, "Out of memory");
|
|
return NULL;
|
|
}
|
|
|
|
switch(curve->type) {
|
|
case WESTON_COLOR_CURVE_TYPE_LUT_3x1D:
|
|
curve->u.lut_3x1d.fill_in(xform, lut, lut_size);
|
|
free(tmp);
|
|
return lut;
|
|
case WESTON_COLOR_CURVE_TYPE_ENUM:
|
|
case WESTON_COLOR_CURVE_TYPE_PARAMETRIC:
|
|
for (i = 0; i < lut_size; i++) {
|
|
float x = (float)i / divider;
|
|
tmp[i] = WESTON_VEC3F(x, x, x);
|
|
}
|
|
weston_color_curve_sample(compositor, curve, tmp, tmp, lut_size);
|
|
weston_v3f_array_to_planar(lut, tmp, lut_size);
|
|
free(tmp);
|
|
return lut;
|
|
case WESTON_COLOR_CURVE_TYPE_IDENTITY:
|
|
weston_assert_not_reached(compositor,
|
|
"no reason to create LUT for identity curve");
|
|
}
|
|
|
|
weston_assert_not_reached(compositor, "unkown color curve");
|
|
}
|
|
|
|
static float
|
|
linear_interpolation(float x, float x0, float y0, float x1, float y1)
|
|
{
|
|
float delta = x1 - x0;
|
|
|
|
/* x0 == x1, 5 digits precision. */
|
|
if (fabs(delta) < 1e-5)
|
|
return (y0 + y1) / 2.0f;
|
|
|
|
return y0 * ((x1 - x) / delta) + y1 * ((x - x0) / delta);
|
|
}
|
|
|
|
WESTON_EXPORT_FOR_TESTS void
|
|
find_neighbors(struct weston_compositor *compositor, uint32_t len, float *array,
|
|
float val, uint32_t *neigh_A_index, uint32_t *neigh_B_index)
|
|
{
|
|
bool ascendent = (array[0] <= array[len - 1]);
|
|
int32_t left = 0;
|
|
int32_t right = len - 1;
|
|
int32_t mid;
|
|
|
|
/* We need at least 2 elements in the array. */
|
|
weston_assert_u32_gt(compositor, len, 1);
|
|
|
|
while (right - left > 1) {
|
|
mid = left + ((right - left) / 2);
|
|
|
|
if ((ascendent && array[mid] < val) ||
|
|
(!ascendent && array[mid] > val))
|
|
left = mid;
|
|
else
|
|
right = mid;
|
|
}
|
|
|
|
*neigh_A_index = left;
|
|
*neigh_B_index = right;
|
|
}
|
|
|
|
/**
|
|
* Given a 1D LUT, this evaluates a given input using the inverse of the LUT.
|
|
*
|
|
* If the input is out of the LUT range, this extrapolates using the two closest
|
|
* elements present in the LUT.
|
|
*
|
|
* \param compositor The compositor instance.
|
|
* \param len_lut The size of the 1D LUT.
|
|
* \param lut The 1D lut.
|
|
* \param input The input to evaluate
|
|
* \return The evaluation result.
|
|
*/
|
|
WL_EXPORT float
|
|
weston_inverse_evaluate_lut1d(struct weston_compositor *compositor,
|
|
uint32_t len_lut, float *lut, float input)
|
|
{
|
|
float divider = len_lut - 1;
|
|
uint32_t neighbor_A_index, neighbor_B_index;
|
|
|
|
find_neighbors(compositor, len_lut, lut, input,
|
|
&neighbor_A_index, &neighbor_B_index);
|
|
|
|
return linear_interpolation(input,
|
|
lut[neighbor_A_index],
|
|
(float)neighbor_A_index / divider,
|
|
lut[neighbor_B_index],
|
|
(float)neighbor_B_index / divider);
|
|
}
|
|
|
|
/**
|
|
* Increase reference count of the color transform object
|
|
*
|
|
* \param xform The color transform. NULL is accepted too.
|
|
* \return xform.
|
|
*/
|
|
WL_EXPORT struct weston_color_transform *
|
|
weston_color_transform_ref(struct weston_color_transform *xform)
|
|
{
|
|
/* NULL is a valid color transform: identity */
|
|
if (!xform)
|
|
return NULL;
|
|
|
|
assert(xform->ref_count > 0);
|
|
xform->ref_count++;
|
|
return xform;
|
|
}
|
|
|
|
/**
|
|
* Decrease and potentially destroy the color transform object
|
|
*
|
|
* \param xform The color transform. NULL is accepted too.
|
|
*/
|
|
WL_EXPORT void
|
|
weston_color_transform_unref(struct weston_color_transform *xform)
|
|
{
|
|
if (!xform)
|
|
return;
|
|
|
|
assert(xform->ref_count > 0);
|
|
if (--xform->ref_count > 0)
|
|
return;
|
|
|
|
wl_signal_emit(&xform->destroy_signal, xform);
|
|
weston_idalloc_put_id(xform->cm->compositor->color_transform_id_generator,
|
|
xform->id);
|
|
xform->cm->destroy_color_transform(xform);
|
|
}
|
|
|
|
/**
|
|
* Initializes a newly allocated color transform object
|
|
*
|
|
* This is used only by color managers. They sub-class weston_color_transform.
|
|
*
|
|
* The reference count starts at 1.
|
|
*
|
|
* To destroy a weston_color_transform, use weston_color_transfor_unref().
|
|
*/
|
|
WL_EXPORT void
|
|
weston_color_transform_init(struct weston_color_transform *xform,
|
|
struct weston_color_manager *cm)
|
|
{
|
|
xform->cm = cm;
|
|
xform->ref_count = 1;
|
|
xform->id = weston_idalloc_get_id(cm->compositor->color_transform_id_generator);
|
|
wl_signal_init(&xform->destroy_signal);
|
|
}
|
|
|
|
static const char *
|
|
param_curve_type_to_str(enum weston_color_curve_parametric_type type)
|
|
{
|
|
switch(type) {
|
|
case WESTON_COLOR_CURVE_PARAMETRIC_TYPE_LINPOW:
|
|
return "linpow";
|
|
case WESTON_COLOR_CURVE_PARAMETRIC_TYPE_POWLIN:
|
|
return "powlin";
|
|
}
|
|
return "???";
|
|
}
|
|
|
|
static const char *
|
|
mapping_type_to_str(enum weston_color_mapping_type mapping_type)
|
|
{
|
|
switch (mapping_type) {
|
|
case WESTON_COLOR_MAPPING_TYPE_IDENTITY:
|
|
return "identity";
|
|
case WESTON_COLOR_MAPPING_TYPE_MATRIX:
|
|
return "matrix";
|
|
}
|
|
return "???";
|
|
}
|
|
|
|
static void
|
|
weston_color_curve_details_fprint(FILE *fp,
|
|
int indent,
|
|
const char *step,
|
|
const struct weston_color_curve *curve)
|
|
{
|
|
static const char *chan[] = { "R", "G", "B" };
|
|
const struct weston_color_curve_enum *en;
|
|
const struct weston_color_curve_parametric *par;
|
|
unsigned ch;
|
|
unsigned i;
|
|
|
|
switch (curve->type) {
|
|
case WESTON_COLOR_CURVE_TYPE_IDENTITY:
|
|
break;
|
|
case WESTON_COLOR_CURVE_TYPE_LUT_3x1D:
|
|
break;
|
|
case WESTON_COLOR_CURVE_TYPE_ENUM:
|
|
en = &curve->u.enumerated;
|
|
if (en->tf.info->count_parameters == 0)
|
|
break;
|
|
|
|
fprintf(fp, "%*s%s, %s:\n", indent, "", step, en->tf.info->desc);
|
|
fprintf(fp, "%*s R,G,B", indent, "");
|
|
for (i = 0; i < en->tf.info->count_parameters; i++)
|
|
fprintf(fp, " % .4f", en->tf.params[i]);
|
|
fprintf(fp, "\n");
|
|
break;
|
|
case WESTON_COLOR_CURVE_TYPE_PARAMETRIC:
|
|
par = &curve->u.parametric;
|
|
fprintf(fp, "%*s%s, %s %s:\n", indent, "", step,
|
|
par->clamped_input ? "clamped" : "unlimited",
|
|
param_curve_type_to_str(par->type));
|
|
for (ch = 0; ch < 3; ch++) {
|
|
fprintf(fp, "%*s %s", indent, "", chan[ch]);
|
|
for (i = 0; i < ARRAY_LENGTH(par->params.chan[0].data); i++)
|
|
fprintf(fp, " % .4f", par->params.chan[ch].data[i]);
|
|
fprintf(fp, "\n");
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
weston_color_mapping_details_fprint(FILE *fp,
|
|
int indent,
|
|
const char *step,
|
|
const struct weston_color_mapping *map)
|
|
{
|
|
const struct weston_color_mapping_matrix *mat;
|
|
unsigned r, c;
|
|
|
|
switch (map->type) {
|
|
case WESTON_COLOR_MAPPING_TYPE_IDENTITY:
|
|
break;
|
|
case WESTON_COLOR_MAPPING_TYPE_MATRIX:
|
|
mat = &map->u.mat;
|
|
fprintf(fp, "%*s%s matrix:\n", indent, "", step);
|
|
for (r = 0; r < 3; r++) {
|
|
fprintf(fp, "%*s", indent + 1, "");
|
|
for (c = 0; c < 3; c++)
|
|
fprintf(fp, " %8.4f", mat->matrix.col[c].el[r]);
|
|
fprintf(fp, " %8.4f\n", mat->offset.el[r]);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Print details of the elements of the color transform pipeline to a string
|
|
*
|
|
* \param indent Count of spaces to use for indenting every line.
|
|
* \param xform The color transform.
|
|
* \return The string in which the pipeline is printed, or NULL if there is
|
|
* nothing to print.
|
|
*/
|
|
WL_EXPORT char *
|
|
weston_color_transform_details_string(int indent,
|
|
const struct weston_color_transform *xform)
|
|
{
|
|
FILE *fp;
|
|
char *str = NULL;
|
|
size_t size = 0;
|
|
|
|
if (!xform->steps_valid)
|
|
return NULL;
|
|
|
|
fp = open_memstream(&str, &size);
|
|
abort_oom_if_null(fp);
|
|
|
|
if (xform->pre_curve.type != WESTON_COLOR_CURVE_TYPE_IDENTITY)
|
|
weston_color_curve_details_fprint(fp, indent, "pre-curve", &xform->pre_curve);
|
|
|
|
if (xform->mapping.type != WESTON_COLOR_MAPPING_TYPE_IDENTITY)
|
|
weston_color_mapping_details_fprint(fp, indent, "mapping", &xform->mapping);
|
|
|
|
if (xform->post_curve.type != WESTON_COLOR_CURVE_TYPE_IDENTITY)
|
|
weston_color_curve_details_fprint(fp, indent, "post-curve", &xform->post_curve);
|
|
|
|
fclose(fp);
|
|
abort_oom_if_null(str);
|
|
|
|
return str;
|
|
}
|
|
|
|
static void
|
|
weston_color_curve_fprint(FILE *fp, const struct weston_color_curve *curve)
|
|
{
|
|
switch (curve->type) {
|
|
case WESTON_COLOR_CURVE_TYPE_IDENTITY:
|
|
fprintf(fp, "identity");
|
|
break;
|
|
case WESTON_COLOR_CURVE_TYPE_LUT_3x1D:
|
|
fprintf(fp, "3x1D LUT [%u]", curve->u.lut_3x1d.optimal_len);
|
|
break;
|
|
case WESTON_COLOR_CURVE_TYPE_ENUM:
|
|
fprintf(fp, "(enum) %s%s",
|
|
curve->u.enumerated.tf_direction == WESTON_INVERSE_TF ? "inverse " : "",
|
|
curve->u.enumerated.tf.info->desc);
|
|
break;
|
|
case WESTON_COLOR_CURVE_TYPE_PARAMETRIC:
|
|
fprintf(fp, "(parametric) %s",
|
|
param_curve_type_to_str(curve->u.parametric.type));
|
|
break;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Print the color transform pipeline to a string
|
|
*
|
|
* \param xform The color transform.
|
|
* \return The string in which the pipeline is printed.
|
|
*/
|
|
WL_EXPORT char *
|
|
weston_color_transform_string(const struct weston_color_transform *xform)
|
|
{
|
|
enum weston_color_mapping_type mapping_type = xform->mapping.type;
|
|
enum weston_color_curve_type pre_type = xform->pre_curve.type;
|
|
enum weston_color_curve_type post_type = xform->post_curve.type;
|
|
const char *empty = "";
|
|
const char *sep = empty;
|
|
FILE *fp;
|
|
char *str = NULL;
|
|
size_t size = 0;
|
|
|
|
if (!xform->steps_valid)
|
|
return xstrdup("Pipeline: uses shaper + 3D LUT\n");
|
|
|
|
fp = open_memstream(&str, &size);
|
|
abort_oom_if_null(fp);
|
|
|
|
fprintf(fp, "Pipeline: ");
|
|
|
|
if (pre_type != WESTON_COLOR_CURVE_TYPE_IDENTITY) {
|
|
fprintf(fp, "%spre = ", sep);
|
|
weston_color_curve_fprint(fp, &xform->pre_curve);
|
|
sep = ", ";
|
|
}
|
|
|
|
if (mapping_type != WESTON_COLOR_MAPPING_TYPE_IDENTITY) {
|
|
fprintf(fp, "%smapping = %s", sep, mapping_type_to_str(mapping_type));
|
|
sep = ", ";
|
|
}
|
|
|
|
if (post_type != WESTON_COLOR_CURVE_TYPE_IDENTITY) {
|
|
fprintf(fp, "%spost = ", sep);
|
|
weston_color_curve_fprint(fp, &xform->post_curve);
|
|
sep = ", ";
|
|
}
|
|
|
|
if (sep == empty)
|
|
fprintf(fp, "identity\n");
|
|
else
|
|
fprintf(fp, "\n");
|
|
|
|
fclose(fp);
|
|
abort_oom_if_null(str);
|
|
|
|
return str;
|
|
}
|
|
|
|
/** Deep copy */
|
|
void
|
|
weston_surface_color_transform_copy(struct weston_surface_color_transform *dst,
|
|
const struct weston_surface_color_transform *src)
|
|
{
|
|
*dst = *src;
|
|
dst->transform = weston_color_transform_ref(src->transform);
|
|
}
|
|
|
|
/** Unref contents */
|
|
void
|
|
weston_surface_color_transform_fini(struct weston_surface_color_transform *surf_xform)
|
|
{
|
|
weston_color_transform_unref(surf_xform->transform);
|
|
surf_xform->transform = NULL;
|
|
surf_xform->identity_pipeline = false;
|
|
}
|
|
|
|
/**
|
|
* Ensure that the surface's color transformation for the given output is
|
|
* populated in the paint nodes for all the views.
|
|
*
|
|
* Creates the color transformation description if necessary by calling
|
|
* into the color manager.
|
|
*
|
|
* \param pnode Paint node defining the surface and the output. All
|
|
* paint nodes with the same surface and output will be ensured.
|
|
*/
|
|
void
|
|
weston_paint_node_ensure_color_transform(struct weston_paint_node *pnode)
|
|
{
|
|
struct weston_surface *surface = pnode->surface;
|
|
struct weston_output *output = pnode->output;
|
|
struct weston_color_manager *cm = surface->compositor->color_manager;
|
|
struct weston_surface_color_transform surf_xform = {};
|
|
struct weston_paint_node *it;
|
|
bool ok;
|
|
|
|
/*
|
|
* Invariant: all paint nodes with the same surface+output have the
|
|
* same surf_xform state.
|
|
*/
|
|
if (pnode->surf_xform_valid)
|
|
return;
|
|
|
|
ok = cm->get_surface_color_transform(cm, surface, output, &surf_xform);
|
|
|
|
wl_list_for_each(it, &surface->paint_node_list, surface_link) {
|
|
if (it->output == output) {
|
|
assert(it->surf_xform_valid == false);
|
|
assert(it->surf_xform.transform == NULL);
|
|
weston_surface_color_transform_copy(&it->surf_xform,
|
|
&surf_xform);
|
|
it->surf_xform_valid = ok;
|
|
}
|
|
}
|
|
|
|
weston_surface_color_transform_fini(&surf_xform);
|
|
|
|
if (!ok) {
|
|
if (surface->resource)
|
|
wl_resource_post_no_memory(surface->resource);
|
|
weston_log("Failed to create color transformation for a surface.\n");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Load ICC profile file
|
|
*
|
|
* Loads an ICC profile file, ensures it is fit for use, and returns a
|
|
* new reference to the weston_color_profile. Use weston_color_profile_unref()
|
|
* to free it.
|
|
*
|
|
* \param compositor The compositor instance, identifies the color manager.
|
|
* \param path Path to the ICC file to be open()'d.
|
|
* \return A color profile reference, or NULL on failure.
|
|
*
|
|
* Error messages are printed to libweston log.
|
|
*
|
|
* This function is not meant for loading profiles on behalf of Wayland
|
|
* clients.
|
|
*/
|
|
WL_EXPORT struct weston_color_profile *
|
|
weston_compositor_load_icc_file(struct weston_compositor *compositor,
|
|
const char *path)
|
|
{
|
|
struct weston_color_manager *cm = compositor->color_manager;
|
|
struct weston_color_profile *cprof = NULL;
|
|
int fd;
|
|
struct stat icc_stat;
|
|
void *icc_data;
|
|
size_t len;
|
|
char *errmsg = NULL;
|
|
|
|
fd = open(path, O_RDONLY);
|
|
if (fd == -1) {
|
|
weston_log("Error: Cannot open ICC profile \"%s\" for reading: %s\n",
|
|
path, strerror(errno));
|
|
return NULL;
|
|
}
|
|
|
|
if (fstat(fd, &icc_stat) != 0) {
|
|
weston_log("Error: Cannot fstat ICC profile \"%s\": %s\n",
|
|
path, strerror(errno));
|
|
goto out_close;
|
|
}
|
|
len = icc_stat.st_size;
|
|
if (len < 1) {
|
|
weston_log("Error: ICC profile \"%s\" has no size.\n", path);
|
|
goto out_close;
|
|
}
|
|
|
|
icc_data = mmap(NULL, len, PROT_READ, MAP_PRIVATE, fd, 0);
|
|
if (icc_data == MAP_FAILED) {
|
|
weston_log("Error: Cannot mmap ICC profile \"%s\": %s\n",
|
|
path, strerror(errno));
|
|
goto out_close;
|
|
}
|
|
|
|
if (!cm->get_color_profile_from_icc(cm, icc_data, len,
|
|
path, &cprof, &errmsg)) {
|
|
weston_log("Error: loading ICC profile \"%s\" failed: %s\n",
|
|
path, errmsg);
|
|
free(errmsg);
|
|
}
|
|
|
|
munmap(icc_data, len);
|
|
|
|
out_close:
|
|
close(fd);
|
|
return cprof;
|
|
}
|
|
|
|
/** Get a string naming the EOTF mode for logs
|
|
*
|
|
* \return Static string. "???" for unknown mode.
|
|
*/
|
|
WL_EXPORT const char *
|
|
weston_eotf_mode_to_str(enum weston_eotf_mode e)
|
|
{
|
|
switch (e) {
|
|
case WESTON_EOTF_MODE_NONE: return "(none)";
|
|
case WESTON_EOTF_MODE_SDR: return "SDR";
|
|
case WESTON_EOTF_MODE_TRADITIONAL_HDR: return "traditional gamma HDR";
|
|
case WESTON_EOTF_MODE_ST2084: return "ST2084";
|
|
case WESTON_EOTF_MODE_HLG: return "HLG";
|
|
}
|
|
return "???";
|
|
}
|
|
|
|
/** A list of EOTF modes as a string
|
|
*
|
|
* \param eotf_mask Bitwise-or'd enum weston_eotf_mode values.
|
|
* \return Comma separated names of the listed EOTF modes. Must be free()'d by
|
|
* the caller.
|
|
*/
|
|
WL_EXPORT char *
|
|
weston_eotf_mask_to_str(uint32_t eotf_mask)
|
|
{
|
|
return bits_to_str(eotf_mask, weston_eotf_mode_to_str);
|
|
}
|
|
|
|
static const struct weston_colorimetry_mode_info colorimetry_mode_info_map[] = {
|
|
{ WESTON_COLORIMETRY_MODE_NONE, "(none)", WDRM_COLORSPACE__COUNT },
|
|
{ WESTON_COLORIMETRY_MODE_DEFAULT, "default", WDRM_COLORSPACE_DEFAULT },
|
|
{ WESTON_COLORIMETRY_MODE_BT2020_CYCC, "BT.2020 (cYCC)", WDRM_COLORSPACE_BT2020_CYCC },
|
|
{ WESTON_COLORIMETRY_MODE_BT2020_YCC, "BT.2020 (YCC)", WDRM_COLORSPACE_BT2020_YCC },
|
|
{ WESTON_COLORIMETRY_MODE_BT2020_RGB, "BT.2020 (RGB)", WDRM_COLORSPACE_BT2020_RGB },
|
|
{ WESTON_COLORIMETRY_MODE_P3D65, "DCI-P3 RGB D65", WDRM_COLORSPACE_DCI_P3_RGB_D65 },
|
|
{ WESTON_COLORIMETRY_MODE_P3DCI, "DCI-P3 RGB Theatre", WDRM_COLORSPACE_DCI_P3_RGB_THEATER },
|
|
{ WESTON_COLORIMETRY_MODE_ICTCP, "BT.2100 ICtCp", WDRM_COLORSPACE__COUNT },
|
|
};
|
|
|
|
/** Get information structure of colorimetry mode
|
|
*
|
|
* \internal
|
|
*/
|
|
WL_EXPORT const struct weston_colorimetry_mode_info *
|
|
weston_colorimetry_mode_info_get(enum weston_colorimetry_mode c)
|
|
{
|
|
unsigned i;
|
|
|
|
for (i = 0; i < ARRAY_LENGTH(colorimetry_mode_info_map); i++)
|
|
if (colorimetry_mode_info_map[i].mode == c)
|
|
return &colorimetry_mode_info_map[i];
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/** Get information structure of colorimetry mode from KMS "Colorspace" enum
|
|
*
|
|
* \internal
|
|
*/
|
|
WL_EXPORT const struct weston_colorimetry_mode_info *
|
|
weston_colorimetry_mode_info_get_by_wdrm(enum wdrm_colorspace cs)
|
|
{
|
|
unsigned i;
|
|
|
|
for (i = 0; i < ARRAY_LENGTH(colorimetry_mode_info_map); i++)
|
|
if (colorimetry_mode_info_map[i].wdrm == cs)
|
|
return &colorimetry_mode_info_map[i];
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/** Get a string naming the colorimetry mode for logs
|
|
*
|
|
* \return Static string. "???" for unknown mode.
|
|
*/
|
|
WL_EXPORT const char *
|
|
weston_colorimetry_mode_to_str(enum weston_colorimetry_mode c)
|
|
{
|
|
const struct weston_colorimetry_mode_info *info;
|
|
|
|
info = weston_colorimetry_mode_info_get(c);
|
|
|
|
return info ? info->name : "???";
|
|
}
|
|
|
|
/** A list of colorimetry modes as a string
|
|
*
|
|
* \param colorimetry_mask Bitwise-or'd enum weston_colorimetry_mode values.
|
|
* \return Comma separated names of the listed colorimetry modes.
|
|
* Must be free()'d by the caller.
|
|
*/
|
|
WL_EXPORT char *
|
|
weston_colorimetry_mask_to_str(uint32_t colorimetry_mask)
|
|
{
|
|
return bits_to_str(colorimetry_mask, weston_colorimetry_mode_to_str);
|
|
}
|
|
|
|
static float
|
|
CIExy_to_z(struct weston_CIExy c)
|
|
{
|
|
return 1.0f - (c.x + c.y);
|
|
}
|
|
|
|
static struct weston_vec3f
|
|
CIExy_to_XYZ(struct weston_CIExy c)
|
|
{
|
|
return WESTON_VEC3F(c.x / c.y, 1.0f, CIExy_to_z(c) / c.y);
|
|
}
|
|
|
|
/** Compute normalized primary matrix (NPM) from primaries and white point
|
|
*
|
|
* \param[out] npm The resulting NPM or inverse NPM.
|
|
* \param[in] gamut Primaries and white point in CIE 1931 xy.
|
|
* \param dir Choose NPM (forward) or its inverse.
|
|
* \return True for success. False for failure: either white point y < 0.01, or
|
|
* an intermediate matrix from the primaries is not invertible.
|
|
*
|
|
* The NPM converts device RGB to CIE 1931 XYZ.
|
|
*
|
|
* Based on SMPTE RP 177-1993, "Derivation of Basic Television Color Equations".
|
|
*/
|
|
WL_EXPORT bool
|
|
weston_normalized_primary_matrix_init(struct weston_mat3f *npm,
|
|
const struct weston_color_gamut *gamut,
|
|
enum weston_npm_direction dir)
|
|
{
|
|
struct weston_CIExy r = gamut->primary[0];
|
|
struct weston_CIExy g = gamut->primary[1];
|
|
struct weston_CIExy b = gamut->primary[2];
|
|
struct weston_CIExy w = gamut->white_point;
|
|
struct weston_mat3f P = WESTON_MAT3F(
|
|
r.x, g.x, b.x,
|
|
r.y, g.y, b.y,
|
|
CIExy_to_z(r), CIExy_to_z(g), CIExy_to_z(b)
|
|
);
|
|
struct weston_mat3f Pinv;
|
|
|
|
if (w.y < 0.01f)
|
|
return false;
|
|
|
|
if (!weston_m3f_invert(&Pinv, P))
|
|
return false;
|
|
|
|
struct weston_vec3f c = weston_m3f_mul_v3f(Pinv, CIExy_to_XYZ(w));
|
|
|
|
switch (dir) {
|
|
case WESTON_NPM_FORWARD:
|
|
/* NPM = P * diag(c) */
|
|
*npm = weston_m3f_mul_m3f(P, weston_m3f_diag(c));
|
|
break;
|
|
case WESTON_NPM_INVERSE:
|
|
/* NPM⁻¹ = (P * diag(c))⁻¹ = diag(c)⁻¹ * P⁻¹ */
|
|
c = WESTON_VEC3F(1.0f / c.x, 1.0f / c.y, 1.0f / c.z);
|
|
*npm = weston_m3f_mul_m3f(weston_m3f_diag(c), Pinv);
|
|
break;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/** Compute linearized Bradford transformation
|
|
*
|
|
* \param from Source adapted white point.
|
|
* \param to Destination adapted white point.
|
|
* \return Full adaptation matrix.
|
|
*
|
|
* Based on ICC.1:2022 (ICC v4.4), annex E.
|
|
*/
|
|
WL_EXPORT struct weston_mat3f
|
|
weston_bradford_adaptation(struct weston_CIExy from, struct weston_CIExy to)
|
|
{
|
|
static const struct weston_mat3f bradford = WESTON_MAT3F(
|
|
0.8951, 0.2664, -0.1614,
|
|
-0.7502, 1.7135, 0.0367,
|
|
0.0389, -0.0685, 1.0296
|
|
);
|
|
struct weston_mat3f inv;
|
|
struct weston_vec3f from_cr;
|
|
struct weston_vec3f to_cr;
|
|
struct weston_vec3f r;
|
|
struct weston_mat3f tmp;
|
|
|
|
weston_m3f_invert(&inv, bradford);
|
|
from_cr = weston_m3f_mul_v3f(bradford, CIExy_to_XYZ(from));
|
|
to_cr = weston_m3f_mul_v3f(bradford, CIExy_to_XYZ(to));
|
|
r = WESTON_VEC3F(to_cr.x / from_cr.x,
|
|
to_cr.y / from_cr.y,
|
|
to_cr.z / from_cr.z);
|
|
tmp = weston_m3f_mul_m3f(weston_m3f_diag(r), bradford);
|
|
return weston_m3f_mul_m3f(inv, tmp);
|
|
}
|
|
|
|
/** Get a string naming the color format
|
|
*
|
|
* \internal
|
|
*/
|
|
WL_EXPORT const char *
|
|
weston_color_format_to_str(enum weston_color_format c)
|
|
{
|
|
switch (c) {
|
|
case WESTON_COLOR_FORMAT_AUTO:
|
|
return "AUTO";
|
|
case WESTON_COLOR_FORMAT_RGB:
|
|
return "RGB";
|
|
case WESTON_COLOR_FORMAT_YUV444:
|
|
return "YUV 4:4:4";
|
|
case WESTON_COLOR_FORMAT_YUV422:
|
|
return "YUV 4:2:2";
|
|
case WESTON_COLOR_FORMAT_YUV420:
|
|
return "YUV 4:2:0";
|
|
}
|
|
|
|
return "???";
|
|
}
|
|
|
|
/** A list of color formats as a string
|
|
*
|
|
* \param color_format_mask Bitwise-or'd enum weston_color_format values.
|
|
* \return Comma separated names of the listed color formats.
|
|
* Must be free()'d by the caller.
|
|
*/
|
|
WL_EXPORT char *
|
|
weston_color_format_mask_to_str(uint32_t color_format_mask)
|
|
{
|
|
return bits_to_str(color_format_mask, weston_color_format_to_str);
|
|
}
|