ICC profiles can now be specified in weston.ini for each output, or a CMS implementation can optionally loaded from a pluggable module.
parent
7b9195f9d6
commit
b24e48e2ff
@ -0,0 +1,134 @@ |
|||||||
|
/*
|
||||||
|
* Copyright © 2013 Richard Hughes |
||||||
|
* |
||||||
|
* Permission to use, copy, modify, distribute, and sell this software and |
||||||
|
* its documentation for any purpose is hereby granted without fee, provided |
||||||
|
* that the above copyright notice appear in all copies and that both that |
||||||
|
* copyright notice and this permission notice appear in supporting |
||||||
|
* documentation, and that the name of the copyright holders not be used in |
||||||
|
* advertising or publicity pertaining to distribution of the software |
||||||
|
* without specific, written prior permission. The copyright holders make |
||||||
|
* no representations about the suitability of this software for any |
||||||
|
* purpose. It is provided "as is" without express or implied warranty. |
||||||
|
* |
||||||
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS |
||||||
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND |
||||||
|
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY |
||||||
|
* SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER |
||||||
|
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF |
||||||
|
* CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN |
||||||
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
||||||
|
*/ |
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H |
||||||
|
#include <config.h> |
||||||
|
#endif |
||||||
|
|
||||||
|
#include <stdlib.h> |
||||||
|
#include <string.h> |
||||||
|
#include <stdio.h> |
||||||
|
|
||||||
|
#ifdef HAVE_LCMS |
||||||
|
#include <lcms2.h> |
||||||
|
#endif |
||||||
|
|
||||||
|
#include "compositor.h" |
||||||
|
#include "cms-helper.h" |
||||||
|
|
||||||
|
#ifdef HAVE_LCMS |
||||||
|
static void |
||||||
|
weston_cms_gamma_clear(struct weston_output *o) |
||||||
|
{ |
||||||
|
int i; |
||||||
|
uint16_t *red; |
||||||
|
|
||||||
|
if (!o->set_gamma) |
||||||
|
return; |
||||||
|
|
||||||
|
red = calloc(o->gamma_size, sizeof(uint16_t)); |
||||||
|
for (i = 0; i < o->gamma_size; i++) |
||||||
|
red[i] = (uint32_t) 0xffff * (uint32_t) i / (uint32_t) (o->gamma_size - 1); |
||||||
|
o->set_gamma(o, o->gamma_size, red, red, red); |
||||||
|
free(red); |
||||||
|
} |
||||||
|
#endif |
||||||
|
|
||||||
|
void |
||||||
|
weston_cms_set_color_profile(struct weston_output *o, |
||||||
|
struct weston_color_profile *p) |
||||||
|
{ |
||||||
|
#ifdef HAVE_LCMS |
||||||
|
cmsFloat32Number in; |
||||||
|
const cmsToneCurve **vcgt; |
||||||
|
int i; |
||||||
|
int size; |
||||||
|
uint16_t *red = NULL; |
||||||
|
uint16_t *green = NULL; |
||||||
|
uint16_t *blue = NULL; |
||||||
|
|
||||||
|
if (!o->set_gamma) |
||||||
|
return; |
||||||
|
if (!p) { |
||||||
|
weston_cms_gamma_clear(o); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
weston_log("Using ICC profile %s\n", p->filename); |
||||||
|
vcgt = cmsReadTag (p->lcms_handle, cmsSigVcgtTag); |
||||||
|
if (vcgt == NULL || vcgt[0] == NULL) { |
||||||
|
weston_cms_gamma_clear(o); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
size = o->gamma_size; |
||||||
|
red = calloc(size, sizeof(uint16_t)); |
||||||
|
green = calloc(size, sizeof(uint16_t)); |
||||||
|
blue = calloc(size, sizeof(uint16_t)); |
||||||
|
for (i = 0; i < size; i++) { |
||||||
|
in = (cmsFloat32Number) i / (cmsFloat32Number) (size - 1); |
||||||
|
red[i] = cmsEvalToneCurveFloat(vcgt[0], in) * (double) 0xffff; |
||||||
|
green[i] = cmsEvalToneCurveFloat(vcgt[1], in) * (double) 0xffff; |
||||||
|
blue[i] = cmsEvalToneCurveFloat(vcgt[2], in) * (double) 0xffff; |
||||||
|
} |
||||||
|
o->set_gamma(o, size, red, green, blue); |
||||||
|
free(red); |
||||||
|
free(green); |
||||||
|
free(blue); |
||||||
|
#endif |
||||||
|
} |
||||||
|
|
||||||
|
void |
||||||
|
weston_cms_destroy_profile(struct weston_color_profile *p) |
||||||
|
{ |
||||||
|
if (!p) |
||||||
|
return; |
||||||
|
#ifdef HAVE_LCMS |
||||||
|
cmsCloseProfile(p->lcms_handle); |
||||||
|
#endif |
||||||
|
free(p->filename); |
||||||
|
free(p); |
||||||
|
} |
||||||
|
|
||||||
|
struct weston_color_profile * |
||||||
|
weston_cms_create_profile(const char *filename, |
||||||
|
void *lcms_profile) |
||||||
|
{ |
||||||
|
struct weston_color_profile *p; |
||||||
|
p = calloc(1, sizeof(struct weston_color_profile)); |
||||||
|
p->filename = strdup(filename); |
||||||
|
p->lcms_handle = lcms_profile; |
||||||
|
return p; |
||||||
|
} |
||||||
|
|
||||||
|
struct weston_color_profile * |
||||||
|
weston_cms_load_profile(const char *filename) |
||||||
|
{ |
||||||
|
struct weston_color_profile *p = NULL; |
||||||
|
#ifdef HAVE_LCMS |
||||||
|
cmsHPROFILE lcms_profile; |
||||||
|
lcms_profile = cmsOpenProfileFromFile(filename, "r"); |
||||||
|
if (lcms_profile) |
||||||
|
p = weston_cms_create_profile(filename, lcms_profile); |
||||||
|
#endif |
||||||
|
return p; |
||||||
|
} |
@ -0,0 +1,70 @@ |
|||||||
|
/*
|
||||||
|
* Copyright © 2013 Richard Hughes |
||||||
|
* |
||||||
|
* Permission to use, copy, modify, distribute, and sell this software and |
||||||
|
* its documentation for any purpose is hereby granted without fee, provided |
||||||
|
* that the above copyright notice appear in all copies and that both that |
||||||
|
* copyright notice and this permission notice appear in supporting |
||||||
|
* documentation, and that the name of the copyright holders not be used in |
||||||
|
* advertising or publicity pertaining to distribution of the software |
||||||
|
* without specific, written prior permission. The copyright holders make |
||||||
|
* no representations about the suitability of this software for any |
||||||
|
* purpose. It is provided "as is" without express or implied warranty. |
||||||
|
* |
||||||
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS |
||||||
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND |
||||||
|
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY |
||||||
|
* SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER |
||||||
|
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF |
||||||
|
* CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN |
||||||
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
||||||
|
*/ |
||||||
|
|
||||||
|
#ifndef _WESTON_CMS_H_ |
||||||
|
#define _WESTON_CMS_H_ |
||||||
|
|
||||||
|
#include "compositor.h" |
||||||
|
|
||||||
|
/* General overview on how to be a CMS plugin:
|
||||||
|
* |
||||||
|
* First, some nomenclature: |
||||||
|
* |
||||||
|
* CMF: Color management framework, i.e. "Use foo.icc for device $bar" |
||||||
|
* CMM: Color management module that converts pixel colors, which is |
||||||
|
* usually lcms2 on any modern OS. |
||||||
|
* CMS: Color management system that encompasses both a CMF and CMM. |
||||||
|
* ICC: International Color Consortium, the people that define the |
||||||
|
* binary encoding of a .icc file. |
||||||
|
* VCGT: Video Card Gamma Tag. An Apple extension to the ICC specification |
||||||
|
* that allows the calibration state to be stored in the ICC profile |
||||||
|
* Output: Physical port with a display attached, e.g. LVDS1 |
||||||
|
* |
||||||
|
* As a CMF is probably something you don't want or need on an embeded install |
||||||
|
* these functions will not be called if the icc_profile key is set for a |
||||||
|
* specific [output] section in weston.ini |
||||||
|
* |
||||||
|
* Most desktop environments want the CMF to decide what profile to use in |
||||||
|
* different situations, so that displays can be profiled and also so that |
||||||
|
* the ICC profiles can be changed at runtime depending on the task or ambient |
||||||
|
* environment. |
||||||
|
* |
||||||
|
* The CMF can be selected using the 'modules' key in the [core] section. |
||||||
|
*/ |
||||||
|
|
||||||
|
struct weston_color_profile { |
||||||
|
char *filename; |
||||||
|
void *lcms_handle; |
||||||
|
}; |
||||||
|
|
||||||
|
void |
||||||
|
weston_cms_set_color_profile(struct weston_output *o, |
||||||
|
struct weston_color_profile *p); |
||||||
|
struct weston_color_profile * |
||||||
|
weston_cms_create_profile(const char *filename, |
||||||
|
void *lcms_profile); |
||||||
|
struct weston_color_profile * |
||||||
|
weston_cms_load_profile(const char *filename); |
||||||
|
void |
||||||
|
weston_cms_destroy_profile(struct weston_color_profile *p); |
||||||
|
|
||||||
|
#endif |
@ -0,0 +1,174 @@ |
|||||||
|
/*
|
||||||
|
* Copyright © 2013 Richard Hughes |
||||||
|
* |
||||||
|
* Permission to use, copy, modify, distribute, and sell this software and |
||||||
|
* its documentation for any purpose is hereby granted without fee, provided |
||||||
|
* that the above copyright notice appear in all copies and that both that |
||||||
|
* copyright notice and this permission notice appear in supporting |
||||||
|
* documentation, and that the name of the copyright holders not be used in |
||||||
|
* advertising or publicity pertaining to distribution of the software |
||||||
|
* without specific, written prior permission. The copyright holders make |
||||||
|
* no representations about the suitability of this software for any |
||||||
|
* purpose. It is provided "as is" without express or implied warranty. |
||||||
|
* |
||||||
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS |
||||||
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND |
||||||
|
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY |
||||||
|
* SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER |
||||||
|
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF |
||||||
|
* CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN |
||||||
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
||||||
|
*/ |
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H |
||||||
|
#include <config.h> |
||||||
|
#endif |
||||||
|
|
||||||
|
#define _GNU_SOURCE |
||||||
|
#include <stdlib.h> |
||||||
|
#include <string.h> |
||||||
|
|
||||||
|
#include "compositor.h" |
||||||
|
#include "cms-helper.h" |
||||||
|
|
||||||
|
struct cms_static { |
||||||
|
struct weston_compositor *ec; |
||||||
|
struct wl_listener destroy_listener; |
||||||
|
struct wl_listener output_created_listener; |
||||||
|
struct wl_list configured_output_list; |
||||||
|
}; |
||||||
|
|
||||||
|
struct cms_configured_output { |
||||||
|
char *icc_profile; |
||||||
|
char *name; |
||||||
|
struct wl_list link; |
||||||
|
}; |
||||||
|
|
||||||
|
static void |
||||||
|
cms_output_created(struct cms_static *cms, struct weston_output *o) |
||||||
|
{ |
||||||
|
struct cms_configured_output *configured_output; |
||||||
|
struct weston_color_profile *p; |
||||||
|
|
||||||
|
weston_log("cms-static: output %i [%s] created\n", o->id, o->name); |
||||||
|
|
||||||
|
/* find profile from configured list */ |
||||||
|
wl_list_for_each(configured_output, &cms->configured_output_list, link) { |
||||||
|
if (strcmp (o->name, configured_output->name) == 0) { |
||||||
|
p = weston_cms_load_profile(configured_output->icc_profile); |
||||||
|
if (p == NULL) { |
||||||
|
weston_log("cms-static: failed to load %s\n", |
||||||
|
configured_output->icc_profile); |
||||||
|
} else { |
||||||
|
weston_log("cms-static: loading %s for %s\n", |
||||||
|
configured_output->icc_profile, o->name); |
||||||
|
weston_cms_set_color_profile(o, p); |
||||||
|
} |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
static void |
||||||
|
cms_notifier_output_created(struct wl_listener *listener, void *data) |
||||||
|
{ |
||||||
|
struct weston_output *o = (struct weston_output *) data; |
||||||
|
struct cms_static *cms = container_of(listener, struct cms_static, destroy_listener); |
||||||
|
cms_output_created(cms, o); |
||||||
|
} |
||||||
|
|
||||||
|
static void |
||||||
|
cms_module_destroy(struct cms_static *cms) |
||||||
|
{ |
||||||
|
struct cms_configured_output *configured_output, *next_co; |
||||||
|
|
||||||
|
wl_list_for_each_safe(configured_output, next_co, |
||||||
|
&cms->configured_output_list, link) { |
||||||
|
free(configured_output->name); |
||||||
|
free(configured_output->icc_profile); |
||||||
|
free(configured_output); |
||||||
|
} |
||||||
|
free(cms); |
||||||
|
} |
||||||
|
|
||||||
|
static void |
||||||
|
cms_notifier_destroy(struct wl_listener *listener, void *data) |
||||||
|
{ |
||||||
|
struct cms_static *cms = container_of(listener, struct cms_static, destroy_listener); |
||||||
|
cms_module_destroy(cms); |
||||||
|
} |
||||||
|
|
||||||
|
static char *output_icc_profile; |
||||||
|
static char *output_name; |
||||||
|
|
||||||
|
static void |
||||||
|
output_section_done(void *data) |
||||||
|
{ |
||||||
|
struct cms_configured_output *configured_output; |
||||||
|
struct cms_static *cms = (struct cms_static *) data; |
||||||
|
|
||||||
|
if (output_name == NULL || output_icc_profile == NULL) { |
||||||
|
free(output_name); |
||||||
|
free(output_icc_profile); |
||||||
|
output_name = NULL; |
||||||
|
output_icc_profile = NULL; |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
weston_log("cms-static: output %s profile configured as %s\n", |
||||||
|
output_name, output_icc_profile); |
||||||
|
|
||||||
|
/* create an object used to store name<->profile data to avoid parsing
|
||||||
|
* the config file every time a new output is added */ |
||||||
|
configured_output = malloc(sizeof *configured_output); |
||||||
|
memset(configured_output, 0, sizeof *configured_output); |
||||||
|
configured_output->name = output_name; |
||||||
|
configured_output->icc_profile = output_icc_profile; |
||||||
|
wl_list_insert(&cms->configured_output_list, &configured_output->link); |
||||||
|
output_name = NULL; |
||||||
|
output_icc_profile = NULL; |
||||||
|
} |
||||||
|
|
||||||
|
WL_EXPORT int |
||||||
|
module_init(struct weston_compositor *ec, |
||||||
|
int *argc, char *argv[], const char *config_file) |
||||||
|
{ |
||||||
|
struct cms_static *cms; |
||||||
|
struct weston_output *output; |
||||||
|
|
||||||
|
weston_log("cms-static: initialized\n"); |
||||||
|
|
||||||
|
/* create local state object */ |
||||||
|
cms = malloc(sizeof *cms); |
||||||
|
if (cms == NULL) |
||||||
|
return -1; |
||||||
|
memset(cms, 0, sizeof *cms); |
||||||
|
|
||||||
|
wl_list_init(&cms->configured_output_list); |
||||||
|
|
||||||
|
/* parse config file */ |
||||||
|
const struct config_key drm_config_keys[] = { |
||||||
|
{ "name", CONFIG_KEY_STRING, &output_name }, |
||||||
|
{ "icc_profile", CONFIG_KEY_STRING, &output_icc_profile }, |
||||||
|
}; |
||||||
|
|
||||||
|
const struct config_section config_section[] = { |
||||||
|
{ "output", drm_config_keys, |
||||||
|
ARRAY_LENGTH(drm_config_keys), output_section_done }, |
||||||
|
}; |
||||||
|
|
||||||
|
parse_config_file(config_file, config_section, |
||||||
|
ARRAY_LENGTH(config_section), cms); |
||||||
|
|
||||||
|
cms->destroy_listener.notify = cms_notifier_destroy; |
||||||
|
wl_signal_add(&ec->destroy_signal, &cms->destroy_listener); |
||||||
|
|
||||||
|
cms->output_created_listener.notify = cms_notifier_output_created; |
||||||
|
wl_signal_add(&ec->output_created_signal, &cms->output_created_listener); |
||||||
|
|
||||||
|
/* discover outputs */ |
||||||
|
wl_list_for_each(output, &ec->output_list, link) |
||||||
|
cms_output_created(cms, output); |
||||||
|
|
||||||
|
return 0; |
||||||
|
} |
Loading…
Reference in new issue