'color_characteristics_config_error' test ensures that all code paths in parse_color_characteristics() and wet_output_set_color_characteristics() get exercised. The return value and logged error messages are checked. Other cases test the weston_hdr_metadata_type1 validation. These are for the sake of test coverage, but also an example of how to test a function from main.c, and how to capture messages from weston_log(). Signed-off-by: Pekka Paalanen <pekka.paalanen@collabora.com>dev
parent
e13e64c4e0
commit
ccb4c383d7
@ -0,0 +1,338 @@ |
||||
/*
|
||||
* Copyright 2022 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 <string.h> |
||||
#include <assert.h> |
||||
#include <math.h> |
||||
|
||||
#include "weston-test-client-helper.h" |
||||
#include "weston-test-fixture-compositor.h" |
||||
|
||||
#include "weston-private.h" |
||||
#include "libweston-internal.h" |
||||
#include "backend.h" |
||||
#include "color.h" |
||||
|
||||
struct config_testcase { |
||||
bool has_characteristics_key; |
||||
const char *output_characteristics_name; |
||||
const char *characteristics_name; |
||||
const char *red_x; |
||||
const char *green_y; |
||||
const char *white_y; |
||||
const char *min_L; |
||||
int expected_retval; |
||||
const char *expected_error; |
||||
}; |
||||
|
||||
static const struct config_testcase config_cases[] = { |
||||
{ |
||||
false, "fred", "fred", "red_x=0.9", "green_y=0.8", "white_y=0.323", "min_L=1e-4", 0, |
||||
"" |
||||
}, |
||||
{ |
||||
true, "fred", "fred", "red_x=0.9", "green_y= 0.8 ", "white_y=0.323", "min_L=1e-4", 0, |
||||
"" |
||||
}, |
||||
{ |
||||
true, "fred", "fred", "red_x=0.9", "green_y= 0.8 ", "white_y=0.323", "", 0, |
||||
"" |
||||
}, |
||||
{ |
||||
true, "notexisting", "fred", "red_x=0.9", "green_y=0.8", "white_y=0.323", "min_L=1e-4", -1, |
||||
"Config error in weston.ini, output mockoutput: no [color_characteristics] section with 'name=notexisting' found.\n" |
||||
}, |
||||
{ |
||||
true, "fr:ed", "fr:ed", "red_x=0.9", "green_y=0.8", "white_y=0.323", "min_L=1e-4", -1, |
||||
"Config error in weston.ini [color_characteristics] name=fr:ed: reserved name. Do not use ':' character in the name.\n" |
||||
}, |
||||
{ |
||||
true, "fred", "fred", "red_x=-5", "green_y=1.01", "white_y=0.323", "min_L=1e-4", -1, |
||||
"Config error in weston.ini [color_characteristics] name=fred: red_x value -5.000000 is outside of the range 0.000000 - 1.000000.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: green_y value 1.010000 is outside of the range 0.000000 - 1.000000.\n" |
||||
}, |
||||
{ |
||||
true, "fred", "fred", "red_x=haahaa", "green_y=-", "white_y=0.323", "min_L=1e-4", -1, |
||||
"Config error in weston.ini [color_characteristics] name=fred: failed to parse the value of key red_x.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: failed to parse the value of key green_y.\n" |
||||
}, |
||||
{ |
||||
true, "fred", "fred", "", "", "white_y=0.323", "min_L=1e-4", -1, |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 1 key red_x is missing. You must set either none or all keys of a group.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 1 key red_y is set. You must set either none or all keys of a group.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 1 key green_x is set. You must set either none or all keys of a group.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 1 key green_y is missing. You must set either none or all keys of a group.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 1 key blue_x is set. You must set either none or all keys of a group.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 1 key blue_y is set. You must set either none or all keys of a group.\n" |
||||
}, |
||||
{ |
||||
true, "fred", "fred", "red_x=0.9", "green_y=0.8", "", "min_L=1e-4", -1, |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 2 key white_x is set. You must set either none or all keys of a group.\n" |
||||
"Config error in weston.ini [color_characteristics] name=fred: group 2 key white_y is missing. You must set either none or all keys of a group.\n" |
||||
}, |
||||
}; |
||||
|
||||
static FILE *logfile; |
||||
|
||||
static int |
||||
logger(const char *fmt, va_list arg) |
||||
{ |
||||
return vfprintf(logfile, fmt, arg); |
||||
} |
||||
|
||||
static int |
||||
no_logger(const char *fmt, va_list arg) |
||||
{ |
||||
return 0; |
||||
} |
||||
|
||||
static struct weston_config * |
||||
create_config(const struct config_testcase *t) |
||||
{ |
||||
struct compositor_setup setup; |
||||
struct weston_config *wc; |
||||
|
||||
compositor_setup_defaults(&setup); |
||||
weston_ini_setup(&setup, |
||||
cfgln("[output]"), |
||||
cfgln("name=mockoutput"), |
||||
t->has_characteristics_key ? |
||||
cfgln("color_characteristics=%s", t->output_characteristics_name) : |
||||
cfgln(""), |
||||
cfgln("eotf-mode=st2084"), |
||||
|
||||
cfgln("[color_characteristics]"), |
||||
cfgln("name=%s", t->characteristics_name), |
||||
cfgln("maxFALL=1000"), |
||||
cfgln("%s", t->red_x), |
||||
cfgln("red_y=0.3"), |
||||
cfgln("blue_x=0.1"), |
||||
cfgln("blue_y=0.11"), |
||||
cfgln("green_x=0.1771"), |
||||
cfgln("%s", t->green_y), |
||||
cfgln("white_x=0.313"), |
||||
cfgln("%s", t->white_y), |
||||
cfgln("%s", t->min_L), |
||||
cfgln("max_L=65535.0"), |
||||
|
||||
cfgln("[core]"), |
||||
cfgln("color-management=true")); |
||||
|
||||
wc = weston_config_parse(setup.config_file); |
||||
free(setup.config_file); |
||||
|
||||
return wc; |
||||
} |
||||
|
||||
/*
|
||||
* Manufacture various weston.ini and check what |
||||
* wet_output_set_color_characteristics() says. Tests for the return value and |
||||
* the error messages logged. |
||||
*/ |
||||
TEST_P(color_characteristics_config_error, config_cases) |
||||
{ |
||||
const struct config_testcase *t = data; |
||||
struct weston_config *wc; |
||||
struct weston_config_section *section; |
||||
int retval; |
||||
char *logbuf; |
||||
size_t logsize; |
||||
struct weston_output mock_output = {}; |
||||
|
||||
weston_output_init(&mock_output, NULL, "mockoutput"); |
||||
|
||||
logfile = open_memstream(&logbuf, &logsize); |
||||
weston_log_set_handler(logger, logger); |
||||
|
||||
wc = create_config(t); |
||||
section = weston_config_get_section(wc, "output", "name", "mockoutput"); |
||||
assert(section); |
||||
|
||||
retval = wet_output_set_color_characteristics(&mock_output, wc, section); |
||||
|
||||
assert(fclose(logfile) == 0); |
||||
logfile = NULL; |
||||
|
||||
testlog("retval %d, logs:\n%s\n", retval, logbuf); |
||||
|
||||
assert(retval == t->expected_retval); |
||||
assert(strcmp(logbuf, t->expected_error) == 0); |
||||
|
||||
weston_config_destroy(wc); |
||||
free(logbuf); |
||||
weston_output_release(&mock_output); |
||||
} |
||||
|
||||
/* Setting NULL resets group_mask */ |
||||
TEST(weston_output_set_color_characteristics_null) |
||||
{ |
||||
struct weston_output mock_output = {}; |
||||
|
||||
weston_output_init(&mock_output, NULL, "mockoutput"); |
||||
|
||||
mock_output.color_characteristics.group_mask = 1; |
||||
weston_output_set_color_characteristics(&mock_output, NULL); |
||||
assert(mock_output.color_characteristics.group_mask == 0); |
||||
|
||||
weston_output_release(&mock_output); |
||||
} |
||||
|
||||
struct value_testcase { |
||||
unsigned field_index; |
||||
float value; |
||||
bool retval; |
||||
}; |
||||
|
||||
static const struct value_testcase value_cases[] = { |
||||
{ 0, 0.0, true }, |
||||
{ 0, 1.0, true }, |
||||
{ 0, -0.001, false }, |
||||
{ 0, 1.01, false }, |
||||
{ 0, NAN, false }, |
||||
{ 0, HUGE_VALF, false }, |
||||
{ 0, -HUGE_VALF, false }, |
||||
{ 1, -1.0, false }, |
||||
{ 2, 2.0, false }, |
||||
{ 3, 2.0, false }, |
||||
{ 4, 2.0, false }, |
||||
{ 5, 2.0, false }, |
||||
{ 6, 2.0, false }, |
||||
{ 7, 2.0, false }, |
||||
{ 8, 0.99, false }, |
||||
{ 8, 65535.1, false }, |
||||
{ 9, 0.000099, false }, |
||||
{ 9, 6.55351, false }, |
||||
{ 10, 0.99, false }, |
||||
{ 10, 65535.1, false }, |
||||
{ 11, 0.99, false }, |
||||
{ 11, 65535.1, false }, |
||||
}; |
||||
|
||||
struct mock_color_manager { |
||||
struct weston_color_manager base; |
||||
struct weston_hdr_metadata_type1 *test_hdr_meta; |
||||
}; |
||||
|
||||
static struct weston_output_color_outcome * |
||||
mock_create_output_color_outcome(struct weston_color_manager *cm_base, |
||||
struct weston_output *output) |
||||
{ |
||||
struct mock_color_manager *cm = container_of(cm_base, typeof(*cm), base); |
||||
struct weston_output_color_outcome *co; |
||||
|
||||
co = zalloc(sizeof *co); |
||||
assert(co); |
||||
|
||||
co->hdr_meta = *cm->test_hdr_meta; |
||||
|
||||
return co; |
||||
} |
||||
|
||||
/*
|
||||
* Modify one value in a known good metadata structure, and see how |
||||
* validation reacts to it. |
||||
*/ |
||||
TEST_P(hdr_metadata_type1_errors, value_cases) |
||||
{ |
||||
struct value_testcase *t = data; |
||||
struct weston_hdr_metadata_type1 meta = { |
||||
.group_mask = WESTON_HDR_METADATA_TYPE1_GROUP_ALL_MASK, |
||||
.primary[0] = { 0.6650, 0.3261 }, |
||||
.primary[1] = { 0.2890, 0.6435 }, |
||||
.primary[2] = { 0.1491, 0.0507 }, |
||||
.white = { 0.3134, 0.3291 }, |
||||
.maxDML = 600.0, |
||||
.minDML = 0.0001, |
||||
.maxCLL = 600.0, |
||||
.maxFALL = 400.0, |
||||
}; |
||||
float *fields[] = { |
||||
&meta.primary[0].x, &meta.primary[0].y, |
||||
&meta.primary[1].x, &meta.primary[1].y, |
||||
&meta.primary[2].x, &meta.primary[2].y, |
||||
&meta.white.x, &meta.white.y, |
||||
&meta.maxDML, &meta.minDML, |
||||
&meta.maxCLL, &meta.maxFALL, |
||||
}; |
||||
struct mock_color_manager mock_cm = { |
||||
.base.create_output_color_outcome = mock_create_output_color_outcome, |
||||
.test_hdr_meta = &meta, |
||||
}; |
||||
struct weston_compositor mock_compositor = { |
||||
.color_manager = &mock_cm.base, |
||||
}; |
||||
struct weston_output mock_output = {}; |
||||
bool ret; |
||||
|
||||
weston_log_set_handler(no_logger, no_logger); |
||||
|
||||
weston_output_init(&mock_output, &mock_compositor, "mockoutput"); |
||||
|
||||
assert(t->field_index < ARRAY_LENGTH(fields)); |
||||
*fields[t->field_index] = t->value; |
||||
ret = weston_output_set_color_outcome(&mock_output); |
||||
assert(ret == t->retval); |
||||
|
||||
weston_output_color_outcome_destroy(&mock_output.color_outcome); |
||||
weston_output_release(&mock_output); |
||||
} |
||||
|
||||
/* Unflagged members are ignored in validity check */ |
||||
TEST(hdr_metadata_type1_ignore_unflagged) |
||||
{ |
||||
/* All values invalid, but also empty mask so none actually used. */ |
||||
struct weston_hdr_metadata_type1 meta = { |
||||
.group_mask = 0, |
||||
.primary[0] = { -1.0, -1.0 }, |
||||
.primary[1] = { -1.0, -1.0 }, |
||||
.primary[2] = { -1.0, -1.0 }, |
||||
.white = { -1.0, -1.0 }, |
||||
.maxDML = -1.0, |
||||
.minDML = -1.0, |
||||
.maxCLL = -1.0, |
||||
.maxFALL = -1.0, |
||||
}; |
||||
struct mock_color_manager mock_cm = { |
||||
.base.create_output_color_outcome = mock_create_output_color_outcome, |
||||
.test_hdr_meta = &meta, |
||||
}; |
||||
struct weston_compositor mock_compositor = { |
||||
.color_manager = &mock_cm.base, |
||||
}; |
||||
struct weston_output mock_output = {}; |
||||
bool ret; |
||||
|
||||
weston_log_set_handler(no_logger, no_logger); |
||||
|
||||
weston_output_init(&mock_output, &mock_compositor, "mockoutput"); |
||||
|
||||
ret = weston_output_set_color_outcome(&mock_output); |
||||
assert(ret); |
||||
|
||||
weston_output_color_outcome_destroy(&mock_output.color_outcome); |
||||
weston_output_release(&mock_output); |
||||
} |
Loading…
Reference in new issue