9002667aa0
Until now, all the curves would be represented with 3x1D LUT's. Now we support LINPOW and POWLIN curves (arbitrary names that we've picked). We can use these curves to represent LittleCMS curves type 1, 4 and their inverses -1, -4. The reason why we want that is because we gain precision using the parametric curves (compared to the LUT's); Surprisingly we had to increase the tolerance of the sRGB->adobeRGB MAT test. Our analysis is that the inverse EOTF power-law curve with exponent 1.0 / 2.2 amplifies errors more than the LUT, specially for input (optical) values closer to zero. That makes sense, because this curve is more sensible to input values closer to zero (i.e. little input variation results in lots of output variation). And this model makes sense, as humans are more capable of perceiving changes of light intensity in the dark. But the downside of all that is that for input values closer to zero, a little bit of noise increases significantly the error. Signed-off-by: Leandro Ribeiro <leandro.ribeiro@collabora.com>
323 lines
8.5 KiB
C
323 lines
8.5 KiB
C
/*
|
|
* Copyright © 2019 Collabora, Ltd.
|
|
* Copyright © 2019 Harish Krupo
|
|
* Copyright © 2019 Intel Corporation
|
|
* Copyright 2021 Advanced Micro Devices, Inc.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#ifndef GL_RENDERER_INTERNAL_H
|
|
#define GL_RENDERER_INTERNAL_H
|
|
|
|
#include <stdbool.h>
|
|
#include <time.h>
|
|
|
|
#include <wayland-util.h>
|
|
#include <GLES2/gl2.h>
|
|
#include <GLES2/gl2ext.h>
|
|
#include "shared/weston-egl-ext.h" /* for PFN* stuff */
|
|
#include "shared/helpers.h"
|
|
|
|
/* Keep the following in sync with vertex.glsl. */
|
|
enum gl_shader_texcoord_input {
|
|
SHADER_TEXCOORD_INPUT_ATTRIB = 0,
|
|
SHADER_TEXCOORD_INPUT_SURFACE,
|
|
};
|
|
|
|
enum gl_shader_texture_variant {
|
|
SHADER_VARIANT_NONE = 0,
|
|
/* Keep the following in sync with fragment.glsl. */
|
|
SHADER_VARIANT_RGBX,
|
|
SHADER_VARIANT_RGBA,
|
|
SHADER_VARIANT_Y_U_V,
|
|
SHADER_VARIANT_Y_UV,
|
|
SHADER_VARIANT_Y_XUXV,
|
|
SHADER_VARIANT_XYUV,
|
|
SHADER_VARIANT_SOLID,
|
|
SHADER_VARIANT_EXTERNAL,
|
|
};
|
|
|
|
/* Keep the following in sync with fragment.glsl. */
|
|
enum gl_shader_color_curve {
|
|
SHADER_COLOR_CURVE_IDENTITY = 0,
|
|
SHADER_COLOR_CURVE_LUT_3x1D,
|
|
SHADER_COLOR_CURVE_LINPOW,
|
|
SHADER_COLOR_CURVE_POWLIN,
|
|
};
|
|
|
|
/* Keep the following in sync with fragment.glsl. */
|
|
enum gl_shader_color_mapping {
|
|
SHADER_COLOR_MAPPING_IDENTITY = 0,
|
|
SHADER_COLOR_MAPPING_3DLUT,
|
|
SHADER_COLOR_MAPPING_MATRIX,
|
|
};
|
|
|
|
/** GL shader requirements key
|
|
*
|
|
* This structure is used as a binary blob key for building and searching
|
|
* shaders. Therefore it must not contain any bytes or bits the C compiler
|
|
* would be free to leave undefined e.g. after struct initialization,
|
|
* struct assignment, or member operations.
|
|
*
|
|
* Use 'pahole' from package 'dwarves' to inspect this structure.
|
|
*/
|
|
struct gl_shader_requirements
|
|
{
|
|
unsigned texcoord_input:1; /* enum gl_shader_texcoord_input */
|
|
|
|
unsigned variant:4; /* enum gl_shader_texture_variant */
|
|
bool input_is_premult:1;
|
|
bool green_tint:1;
|
|
|
|
unsigned color_pre_curve:2; /* enum gl_shader_color_curve */
|
|
unsigned color_mapping:2; /* enum gl_shader_color_mapping */
|
|
unsigned color_post_curve:2; /* enum gl_shader_color_curve */
|
|
/*
|
|
* The total size of all bitfields plus pad_bits_ must fill up exactly
|
|
* how many bytes the compiler allocates for them together.
|
|
*/
|
|
unsigned pad_bits_:19;
|
|
};
|
|
static_assert(sizeof(struct gl_shader_requirements) ==
|
|
4 /* total bitfield size in bytes */,
|
|
"struct gl_shader_requirements must not contain implicit padding");
|
|
|
|
struct gl_shader;
|
|
struct weston_color_transform;
|
|
|
|
#define GL_SHADER_INPUT_TEX_MAX 3
|
|
struct gl_shader_config {
|
|
struct gl_shader_requirements req;
|
|
|
|
struct weston_matrix projection;
|
|
struct weston_matrix surface_to_buffer;
|
|
float view_alpha;
|
|
GLfloat unicolor[4];
|
|
GLint input_tex_filter; /* GL_NEAREST or GL_LINEAR */
|
|
GLuint input_tex[GL_SHADER_INPUT_TEX_MAX];
|
|
|
|
union {
|
|
struct {
|
|
GLuint tex;
|
|
GLfloat scale_offset[2];
|
|
} lut_3x1d;
|
|
struct {
|
|
GLfloat params[3][10];
|
|
GLboolean clamped_input;
|
|
} parametric;
|
|
} color_pre_curve;
|
|
|
|
union {
|
|
struct {
|
|
GLuint tex;
|
|
GLfloat scale_offset[2];
|
|
} lut3d;
|
|
GLfloat matrix[9];
|
|
} color_mapping;
|
|
|
|
union {
|
|
struct {
|
|
GLuint tex;
|
|
GLfloat scale_offset[2];
|
|
} lut_3x1d;
|
|
struct {
|
|
GLfloat params[3][10];
|
|
GLboolean clamped_input;
|
|
} parametric;
|
|
} color_post_curve;
|
|
};
|
|
|
|
struct gl_renderer {
|
|
struct weston_renderer base;
|
|
struct weston_compositor *compositor;
|
|
struct weston_log_scope *renderer_scope;
|
|
|
|
bool fragment_shader_debug;
|
|
bool fan_debug;
|
|
struct weston_binding *fragment_binding;
|
|
struct weston_binding *fan_binding;
|
|
|
|
EGLenum platform;
|
|
EGLDisplay egl_display;
|
|
EGLContext egl_context;
|
|
EGLConfig egl_config;
|
|
|
|
uint32_t gl_version;
|
|
|
|
struct wl_array vertices;
|
|
struct wl_array vtxcnt;
|
|
|
|
EGLDeviceEXT egl_device;
|
|
const char *drm_device;
|
|
|
|
struct weston_drm_format_array supported_formats;
|
|
|
|
PFNGLEGLIMAGETARGETTEXTURE2DOESPROC image_target_texture_2d;
|
|
PFNGLTEXIMAGE3DOESPROC tex_image_3d;
|
|
PFNEGLCREATEIMAGEKHRPROC create_image;
|
|
PFNEGLDESTROYIMAGEKHRPROC destroy_image;
|
|
PFNEGLSWAPBUFFERSWITHDAMAGEEXTPROC swap_buffers_with_damage;
|
|
|
|
PFNEGLGETPLATFORMDISPLAYEXTPROC get_platform_display;
|
|
PFNEGLCREATEPLATFORMWINDOWSURFACEEXTPROC create_platform_window;
|
|
bool has_platform_base;
|
|
|
|
PFNEGLBINDWAYLANDDISPLAYWL bind_display;
|
|
PFNEGLUNBINDWAYLANDDISPLAYWL unbind_display;
|
|
PFNEGLQUERYWAYLANDBUFFERWL query_buffer;
|
|
bool has_bind_display;
|
|
|
|
bool has_context_priority;
|
|
|
|
bool has_egl_image_external;
|
|
|
|
bool has_egl_buffer_age;
|
|
bool has_egl_partial_update;
|
|
PFNEGLSETDAMAGEREGIONKHRPROC set_damage_region;
|
|
|
|
bool has_configless_context;
|
|
|
|
bool has_surfaceless_context;
|
|
|
|
bool has_dmabuf_import;
|
|
struct wl_list dmabuf_images;
|
|
struct wl_list dmabuf_formats;
|
|
|
|
bool has_texture_type_2_10_10_10_rev;
|
|
bool has_gl_texture_rg;
|
|
bool has_texture_norm16;
|
|
bool has_texture_storage;
|
|
bool has_pack_reverse;
|
|
bool has_rgb8_rgba8;
|
|
|
|
bool has_pbo;
|
|
GLenum pbo_usage;
|
|
PFNGLMAPBUFFERRANGEEXTPROC map_buffer_range;
|
|
PFNGLUNMAPBUFFEROESPROC unmap_buffer;
|
|
|
|
struct wl_list pending_capture_list;
|
|
|
|
struct gl_shader *current_shader;
|
|
struct gl_shader *fallback_shader;
|
|
|
|
struct wl_signal destroy_signal;
|
|
|
|
bool has_dmabuf_import_modifiers;
|
|
PFNEGLQUERYDMABUFFORMATSEXTPROC query_dmabuf_formats;
|
|
PFNEGLQUERYDMABUFMODIFIERSEXTPROC query_dmabuf_modifiers;
|
|
|
|
bool has_device_query;
|
|
PFNEGLQUERYDISPLAYATTRIBEXTPROC query_display_attrib;
|
|
PFNEGLQUERYDEVICESTRINGEXTPROC query_device_string;
|
|
|
|
bool has_native_fence_sync;
|
|
PFNEGLCREATESYNCKHRPROC create_sync;
|
|
PFNEGLDESTROYSYNCKHRPROC destroy_sync;
|
|
PFNEGLDUPNATIVEFENCEFDANDROIDPROC dup_native_fence_fd;
|
|
|
|
bool has_wait_sync;
|
|
PFNEGLWAITSYNCKHRPROC wait_sync;
|
|
|
|
bool has_disjoint_timer_query;
|
|
PFNGLGENQUERIESEXTPROC gen_queries;
|
|
PFNGLDELETEQUERIESEXTPROC delete_queries;
|
|
PFNGLBEGINQUERYEXTPROC begin_query;
|
|
PFNGLENDQUERYEXTPROC end_query;
|
|
#if !defined(NDEBUG)
|
|
PFNGLGETQUERYOBJECTIVEXTPROC get_query_object_iv;
|
|
#endif
|
|
PFNGLGETQUERYOBJECTUI64VEXTPROC get_query_object_ui64v;
|
|
|
|
bool gl_supports_color_transforms;
|
|
|
|
/** Shader program cache in most recently used order
|
|
*
|
|
* Uses struct gl_shader::link.
|
|
*/
|
|
struct wl_list shader_list;
|
|
struct weston_log_scope *shader_scope;
|
|
};
|
|
|
|
static inline struct gl_renderer *
|
|
get_renderer(struct weston_compositor *ec)
|
|
{
|
|
return (struct gl_renderer *)ec->renderer;
|
|
}
|
|
|
|
void
|
|
gl_renderer_print_egl_error_state(void);
|
|
|
|
void
|
|
gl_renderer_log_extensions(struct gl_renderer *gr,
|
|
const char *name, const char *extensions);
|
|
|
|
void
|
|
log_egl_config_info(EGLDisplay egldpy, EGLConfig eglconfig);
|
|
|
|
EGLConfig
|
|
gl_renderer_get_egl_config(struct gl_renderer *gr,
|
|
EGLint egl_surface_type,
|
|
const struct pixel_format_info *const *formats,
|
|
unsigned formats_count);
|
|
|
|
int
|
|
gl_renderer_setup_egl_display(struct gl_renderer *gr, void *native_display);
|
|
|
|
int
|
|
gl_renderer_setup_egl_client_extensions(struct gl_renderer *gr);
|
|
|
|
int
|
|
gl_renderer_setup_egl_extensions(struct weston_compositor *ec);
|
|
|
|
GLenum
|
|
gl_shader_texture_variant_get_target(enum gl_shader_texture_variant v);
|
|
|
|
bool
|
|
gl_shader_texture_variant_can_be_premult(enum gl_shader_texture_variant v);
|
|
|
|
void
|
|
gl_shader_destroy(struct gl_renderer *gr, struct gl_shader *shader);
|
|
|
|
void
|
|
gl_renderer_shader_list_destroy(struct gl_renderer *gr);
|
|
|
|
struct gl_shader *
|
|
gl_renderer_create_fallback_shader(struct gl_renderer *gr);
|
|
|
|
void
|
|
gl_renderer_garbage_collect_programs(struct gl_renderer *gr);
|
|
|
|
bool
|
|
gl_renderer_use_program(struct gl_renderer *gr,
|
|
const struct gl_shader_config *sconf);
|
|
|
|
struct weston_log_scope *
|
|
gl_shader_scope_create(struct gl_renderer *gr);
|
|
|
|
bool
|
|
gl_shader_config_set_color_transform(struct gl_renderer *gr,
|
|
struct gl_shader_config *sconf,
|
|
struct weston_color_transform *xform);
|
|
|
|
#endif /* GL_RENDERER_INTERNAL_H */
|