weston/libweston/color.c
Pekka Paalanen 120b88aa0a color: run vec3 through weston_color_curve_sample()
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>
2025-10-21 14:23:37 +03:00

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, &params->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, &params->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);
}