Skip to content

Commit 2cd0a24

Browse files
committed
tests: lib: pixel: add unit tests for format conversion
The tests use data generated by the ffmpeg command line utilty in order to avoid bias by having the same person implementing the tests and the source code. Signed-off-by: Josuah Demangeon <me@josuah.net>
1 parent d4a4f00 commit 2cd0a24

File tree

5 files changed

+380
-0
lines changed

5 files changed

+380
-0
lines changed

tests/lib/pixel/CMakeLists.txt

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
# SPDX-License-Identifier: Apache-2.0
2+
3+
cmake_minimum_required(VERSION 3.20.0)
4+
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
5+
project(pixel)
6+
7+
FILE(GLOB app_sources src/*.c)
8+
target_sources(app PRIVATE ${app_sources})

tests/lib/pixel/prj.conf

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
CONFIG_PIXEL_LOG_LEVEL_DBG=y
2+
CONFIG_ZTEST=y
3+
CONFIG_PIXEL=y

tests/lib/pixel/src/pixel_format.c

+230
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,230 @@
1+
/*
2+
* Copyright (c) 2025 tinyVision.ai Inc.
3+
*
4+
* SPDX-License-Identifier: Apache-2.0
5+
*/
6+
#include <zephyr/kernel.h>
7+
#include <zephyr/random/random.h>
8+
#include <zephyr/ztest.h>
9+
#include <zephyr/pixel/formats.h>
10+
#include <zephyr/pixel/print.h>
11+
12+
#define WIDTH 16
13+
#define HEIGHT 16
14+
15+
#define ERROR_MARGIN 10
16+
17+
/*
18+
* To get YUV BT.709 test data:
19+
*
20+
* ffmpeg -y -f lavfi -colorspace bt709 -i color=#rrggbb:2x2:d=3,format=rgb24 \
21+
* -f rawvideo -pix_fmt yuyv422 - | hexdump -C
22+
*
23+
* To get RGB565 test data:
24+
*
25+
* ffmpeg -y -f lavfi -i color=$rgb:2x2:d=3,format=rgb24 \
26+
* -f rawvideo -pix_fmt rgb565 - | hexdump -C
27+
*/
28+
29+
const struct color_ref {
30+
uint8_t rgb24[3];
31+
uint8_t rgb565[2];
32+
uint8_t rgb332[1];
33+
uint8_t yuv24_bt709[3];
34+
uint8_t yuv24_bt601[3];
35+
} reference_data[] = {
36+
37+
/* Primary colors */
38+
{{0x00, 0x00, 0x00}, {0x00, 0x00}, {0x00}, {0x10, 0x80, 0x80}, {0x10, 0x80, 0x80}},
39+
{{0x00, 0x00, 0xff}, {0x00, 0x1f}, {0x03}, {0x20, 0xf0, 0x76}, {0x29, 0xf1, 0x6e}},
40+
{{0x00, 0xff, 0x00}, {0x07, 0xe0}, {0x1c}, {0xad, 0x2a, 0x1a}, {0x9a, 0x2a, 0x35}},
41+
{{0x00, 0xff, 0xff}, {0x07, 0xff}, {0x1f}, {0xbc, 0x9a, 0x10}, {0xb4, 0xa0, 0x23}},
42+
{{0xff, 0x00, 0x00}, {0xf8, 0x00}, {0xe0}, {0x3f, 0x66, 0xf0}, {0x50, 0x5b, 0xee}},
43+
{{0xff, 0x00, 0xff}, {0xf8, 0x1f}, {0xe3}, {0x4e, 0xd6, 0xe6}, {0x69, 0xcb, 0xdc}},
44+
{{0xff, 0xff, 0x00}, {0xff, 0xe0}, {0xfc}, {0xdb, 0x10, 0x8a}, {0xd0, 0x0a, 0x93}},
45+
{{0xff, 0xff, 0xff}, {0xff, 0xff}, {0xff}, {0xeb, 0x80, 0x80}, {0xeb, 0x80, 0x80}},
46+
47+
/* Arbitrary colors */
48+
{{0x00, 0x70, 0xc5}, {0x03, 0x98}, {0x0f}, {0x61, 0xb1, 0x4b}, {0x5e, 0xb5, 0x4d}},
49+
{{0x33, 0x8d, 0xd1}, {0x3c, 0x7a}, {0x33}, {0x7d, 0xa7, 0x56}, {0x7b, 0xab, 0x57}},
50+
{{0x66, 0xa9, 0xdc}, {0x6d, 0x5b}, {0x77}, {0x98, 0x9d, 0x61}, {0x96, 0xa0, 0x61}},
51+
{{0x7d, 0xd2, 0xf7}, {0x86, 0x9e}, {0x7b}, {0xb7, 0x99, 0x59}, {0xb3, 0x9d, 0x5a}},
52+
{{0x97, 0xdb, 0xf9}, {0x9e, 0xde}, {0x9b}, {0xc2, 0x94, 0x61}, {0xbf, 0x97, 0x62}},
53+
{{0xb1, 0xe4, 0xfa}, {0xb7, 0x3f}, {0xbf}, {0xcc, 0x8f, 0x69}, {0xca, 0x91, 0x69}},
54+
{{0x79, 0x29, 0xd2}, {0x79, 0x5a}, {0x67}, {0x4c, 0xc2, 0x9c}, {0x57, 0xbf, 0x96}},
55+
{{0x94, 0x54, 0xdb}, {0x9a, 0xbb}, {0x8b}, {0x6c, 0xb5, 0x97}, {0x75, 0xb3, 0x92}},
56+
{{0xaf, 0x7f, 0xe4}, {0xb3, 0xfc}, {0xaf}, {0x8c, 0xa8, 0x91}, {0x93, 0xa6, 0x8d}},
57+
};
58+
59+
static uint8_t line_in[WIDTH * 4];
60+
static uint8_t line_out[WIDTH * 4];
61+
62+
void test_conversion(const uint8_t *pix_in, size_t pix_in_size, size_t pix_in_step,
63+
const uint8_t *pix_out, size_t pix_out_size, size_t pix_out_step,
64+
void (*fn)(const uint8_t *in, uint8_t *out, uint16_t width))
65+
{
66+
bool done = false;
67+
68+
/* Fill the input line as much as possible */
69+
for (size_t w = 0; w < WIDTH; w += pix_in_step) {
70+
memcpy(&line_in[w * pix_in_size], pix_in, pix_in_size * pix_in_step);
71+
}
72+
73+
/* Perform the conversion to test */
74+
fn(line_in, line_out, WIDTH);
75+
76+
printf("out:");
77+
for (int i = 0; i < pix_out_step * pix_out_size; i++) {
78+
printf(" %02x", line_out[i]);
79+
}
80+
printf("\n");
81+
printf("ref:");
82+
for (int i = 0; i < pix_out_step * pix_out_size; i++) {
83+
printf(" %02x", pix_out[i]);
84+
}
85+
printf("\n");
86+
87+
/* Scan the result against the reference output pixel to make sure it worked */
88+
for (size_t w = 0; w < WIDTH; w += pix_out_step) {
89+
for (int i = 0; w * pix_out_size + i < (w + pix_out_step) * pix_out_size; i++) {
90+
zassert_within(line_out[w * pix_out_size + i], pix_out[i], 9,
91+
"at %u: value 0x%02x, reference 0x%02x",
92+
i, line_out[w * pix_out_size + i], pix_out[i]);
93+
}
94+
95+
/* Make sure we visited that loop */
96+
done = true;
97+
}
98+
99+
zassert_true(done);
100+
}
101+
102+
ZTEST(lib_pixel_format, test_pixel_format_line)
103+
{
104+
for (size_t i = 0; i < ARRAY_SIZE(reference_data); i++) {
105+
/* The current color we are testing */
106+
const struct color_ref *ref = &reference_data[i];
107+
108+
/* Generate very small buffers out of the reference tables */
109+
const uint8_t rgb24[] = {
110+
ref->rgb24[0],
111+
ref->rgb24[1],
112+
ref->rgb24[2],
113+
};
114+
const uint8_t rgb565be[] = {
115+
ref->rgb565[0],
116+
ref->rgb565[1],
117+
};
118+
const uint8_t rgb565le[] = {
119+
ref->rgb565[1],
120+
ref->rgb565[0],
121+
};
122+
const uint8_t rgb332[] = {
123+
ref->rgb332[0],
124+
};
125+
const uint8_t yuyv_bt709[] = {
126+
ref->yuv24_bt709[0],
127+
ref->yuv24_bt709[1],
128+
ref->yuv24_bt709[0],
129+
ref->yuv24_bt709[2],
130+
};
131+
132+
printf("\ncolor #%02x%02x%02x\n", ref->rgb24[0], ref->rgb24[1], ref->rgb24[2]);
133+
134+
test_conversion(rgb24, 3, 1, rgb565be, 2, 1, &pixel_rgb24line_to_rgb565beline);
135+
printf("rgb24 in |");
136+
pixel_print_rgb24frame_truecolor(line_in, sizeof(line_in), WIDTH / 2, 2);
137+
printf("rgb565 out|");
138+
pixel_print_rgb565beframe_truecolor(line_out, sizeof(line_out), WIDTH / 2, 2);
139+
140+
test_conversion(rgb24, 3, 1, rgb565le, 2, 1, &pixel_rgb24line_to_rgb565leline);
141+
printf("rgb24 in |");
142+
pixel_print_rgb24frame_truecolor(line_in, sizeof(line_in), WIDTH / 2, 2);
143+
printf("rgb565 out|");
144+
pixel_print_rgb565leframe_truecolor(line_out, sizeof(line_out), WIDTH / 2, 2);
145+
146+
test_conversion(rgb24, 3, 1, rgb332, 1, 1, &pixel_rgb24line_to_rgb332line);
147+
printf("rgb24 in |");
148+
pixel_print_rgb24frame_truecolor(line_in, sizeof(line_in), WIDTH / 2, 2);
149+
printf("rgb332 out|");
150+
pixel_print_rgb332frame_truecolor(line_out, sizeof(line_out), WIDTH / 2, 2);
151+
152+
test_conversion(rgb565be, 2, 1, rgb24, 3, 1, &pixel_rgb565beline_to_rgb24line);
153+
printf("rgb565 in |");
154+
pixel_print_rgb565beframe_truecolor(line_in, sizeof(line_in), WIDTH / 2, 2);
155+
printf("rgb24 out |");
156+
pixel_print_rgb24frame_truecolor(line_out, sizeof(line_out), WIDTH / 2, 2);
157+
158+
test_conversion(rgb565le, 2, 1, rgb24, 3, 1, &pixel_rgb565leline_to_rgb24line);
159+
printf("rgb565 in |");
160+
pixel_print_rgb565leframe_truecolor(line_in, sizeof(line_in), WIDTH / 2, 2);
161+
printf("rgb24 out |");
162+
pixel_print_rgb24frame_truecolor(line_out, sizeof(line_out), WIDTH / 2, 2);
163+
164+
test_conversion(rgb24, 3, 1, yuyv_bt709, 2, 2, &pixel_rgb24line_to_yuyvline_bt709);
165+
printf("rgb24 in |");
166+
pixel_print_rgb24frame_truecolor(line_in, sizeof(line_in), WIDTH / 2, 2);
167+
printf("bt709 out |");
168+
pixel_print_yuyvframe_bt709_truecolor(line_out, sizeof(line_out), WIDTH / 2, 2);
169+
170+
test_conversion(yuyv_bt709, 2, 2, rgb24, 3, 1, &pixel_yuyvline_to_rgb24line_bt709);
171+
printf("bt709 in |");
172+
pixel_print_yuyvframe_bt709_truecolor(line_in, sizeof(line_in), WIDTH / 2, 2);
173+
printf("rgb24 out |");
174+
pixel_print_rgb24frame_truecolor(line_out, sizeof(line_out), WIDTH / 2, 2);
175+
}
176+
}
177+
178+
/* From RGB24 */
179+
static PIXEL_RGB24STREAM_TO_RGB565BESTREAM(step_rgb24_to_rgb565be, WIDTH, HEIGHT);
180+
static PIXEL_RGB24STREAM_TO_RGB565LESTREAM(step_rgb24_to_rgb565le, WIDTH, HEIGHT);
181+
static PIXEL_RGB24STREAM_TO_YUYVSTREAM_BT709(step_rgb24_to_yuyv, WIDTH, HEIGHT);
182+
183+
/* To RGB24 */
184+
static PIXEL_RGB565BESTREAM_TO_RGB24STREAM(step_rgb565be_to_rgb24, WIDTH, HEIGHT);
185+
static PIXEL_RGB565LESTREAM_TO_RGB24STREAM(step_rgb565le_to_rgb24, WIDTH, HEIGHT);
186+
static PIXEL_YUYVSTREAM_TO_RGB24STREAM_BT709(step_yuyv_to_rgb24, WIDTH, HEIGHT);
187+
188+
static uint8_t rgb24frame_in[WIDTH * HEIGHT * 3];
189+
static uint8_t rgb24frame_out[WIDTH * HEIGHT * 3];
190+
191+
ZTEST(lib_pixel_format, test_pixel_format_stream)
192+
{
193+
struct pixel_stream root = {0};
194+
struct pixel_stream *step = &root;
195+
struct pixel_stream step_export = {
196+
.ring = RING_BUF_INIT(rgb24frame_out, sizeof(rgb24frame_out)),
197+
.pitch = WIDTH * 3,
198+
.height = HEIGHT,
199+
.name = "[export]",
200+
};
201+
202+
/* Generate test input data */
203+
for (size_t i = 0; i < sizeof(rgb24frame_in); i++) {
204+
rgb24frame_in[i] = i / 3;
205+
}
206+
207+
/* Build and run the pipeline */
208+
step = step->next = &step_rgb24_to_rgb565le;
209+
step = step->next = &step_rgb565le_to_rgb24;
210+
step = step->next = &step_rgb24_to_rgb565be;
211+
step = step->next = &step_rgb565be_to_rgb24;
212+
step = step->next = &step_rgb24_to_yuyv;
213+
step = step->next = &step_yuyv_to_rgb24;
214+
step = step->next = &step_export;
215+
pixel_stream_load(root.next, rgb24frame_in, sizeof(rgb24frame_in));
216+
217+
printf("input:\n");
218+
pixel_print_rgb24frame_truecolor(rgb24frame_in, sizeof(rgb24frame_in), WIDTH, HEIGHT);
219+
220+
printf("output:\n");
221+
pixel_print_rgb24frame_truecolor(rgb24frame_out, sizeof(rgb24frame_out), WIDTH, HEIGHT);
222+
223+
for (int i = 0; i < sizeof(rgb24frame_out); i++) {
224+
/* Precision is not 100% as some conversions steps are lossy */
225+
zassert_within(rgb24frame_in[i], rgb24frame_out[i], ERROR_MARGIN,
226+
"Testing position %u", i);
227+
}
228+
}
229+
230+
ZTEST_SUITE(lib_pixel_format, NULL, NULL, NULL, NULL, NULL);

tests/lib/pixel/src/pixel_resize.c

+132
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,132 @@
1+
/*
2+
* Copyright (c) 2025 tinyVision.ai Inc.
3+
*
4+
* SPDX-License-Identifier: Apache-2.0
5+
*/
6+
#include <zephyr/kernel.h>
7+
#include <zephyr/random/random.h>
8+
#include <zephyr/ztest.h>
9+
#include <zephyr/pixel/resize.h>
10+
#include <zephyr/pixel/formats.h>
11+
#include <zephyr/pixel/print.h>
12+
13+
#define WIDTH_IN 32
14+
#define HEIGHT_IN 16
15+
#define PITCH_IN (WIDTH_IN * 3)
16+
17+
#define WIDTH_OUT 24
18+
#define HEIGHT_OUT 24
19+
#define PITCH_OUT (WIDTH_OUT * 3)
20+
21+
#define ERROR_MARGIN 9
22+
23+
/* Input/output buffers */
24+
static uint8_t rgb24frame_in[WIDTH_IN * HEIGHT_IN * 3];
25+
static uint8_t rgb24frame_out[WIDTH_OUT * HEIGHT_OUT * 3];
26+
27+
/* Stream conversion steps */
28+
static PIXEL_RGB24STREAM_TO_RGB565BESTREAM(step_rgb24_to_rgb565be, WIDTH_IN, HEIGHT_IN);
29+
static PIXEL_RGB24STREAM_TO_RGB565LESTREAM(step_rgb24_to_rgb565le, WIDTH_IN, HEIGHT_IN);
30+
static PIXEL_RGB565BESTREAM_TO_RGB24STREAM(step_rgb565be_to_rgb24, WIDTH_OUT, HEIGHT_OUT);
31+
static PIXEL_RGB565LESTREAM_TO_RGB24STREAM(step_rgb565le_to_rgb24, WIDTH_OUT, HEIGHT_OUT);
32+
static PIXEL_SUBSAMPLE_RGB24STREAM(step_subsample_rgb24, WIDTH_IN, HEIGHT_IN);
33+
static PIXEL_SUBSAMPLE_RGB565STREAM(step_subsample_rgb565, WIDTH_IN, HEIGHT_IN);
34+
35+
struct pixel_stream root = {0};
36+
37+
static void test_resize(void)
38+
{
39+
/* Run the pipeline */
40+
pixel_stream_load(root.next, rgb24frame_in, sizeof(rgb24frame_in));
41+
42+
printf("input:\n");
43+
pixel_print_rgb24frame_truecolor(rgb24frame_in, sizeof(rgb24frame_in), WIDTH_IN, HEIGHT_IN);
44+
45+
printf("output:\n");
46+
pixel_print_rgb24frame_truecolor(rgb24frame_out, sizeof(rgb24frame_out), WIDTH_OUT,
47+
HEIGHT_OUT);
48+
49+
size_t w = WIDTH_OUT;
50+
size_t h = HEIGHT_OUT;
51+
size_t p = PITCH_OUT;
52+
53+
/* Test top left quadramt */
54+
zassert_within(rgb24frame_out[(0) * p + (0) * 3 + 0], 0x00, ERROR_MARGIN);
55+
zassert_within(rgb24frame_out[(0) * p + (0) * 3 + 1], 0x00, ERROR_MARGIN);
56+
zassert_within(rgb24frame_out[(0) * p + (0) * 3 + 2], 0x7f, ERROR_MARGIN);
57+
zassert_within(rgb24frame_out[(h / 2 - 1) * p + (w / 2 - 1) * 3 + 0], 0x00, ERROR_MARGIN);
58+
zassert_within(rgb24frame_out[(h / 2 - 1) * p + (w / 2 - 1) * 3 + 1], 0x00, ERROR_MARGIN);
59+
zassert_within(rgb24frame_out[(h / 2 - 1) * p + (w / 2 - 1) * 3 + 2], 0x7f, ERROR_MARGIN);
60+
61+
/* Test bottom left quadrant */
62+
zassert_within(rgb24frame_out[(h - 1) * p + (0) * 3 + 0], 0x00, ERROR_MARGIN);
63+
zassert_within(rgb24frame_out[(h - 1) * p + (0) * 3 + 1], 0xff, ERROR_MARGIN);
64+
zassert_within(rgb24frame_out[(h - 1) * p + (0) * 3 + 2], 0x7f, ERROR_MARGIN);
65+
zassert_within(rgb24frame_out[(h / 2 + 1) * p + (w / 2 - 1) * 3 + 0], 0x00, ERROR_MARGIN);
66+
zassert_within(rgb24frame_out[(h / 2 + 1) * p + (w / 2 - 1) * 3 + 1], 0xff, ERROR_MARGIN);
67+
zassert_within(rgb24frame_out[(h / 2 + 1) * p + (w / 2 - 1) * 3 + 2], 0x7f, ERROR_MARGIN);
68+
69+
/* Test top right quadrant */
70+
zassert_within(rgb24frame_out[(0) * p + (w - 1) * 3 + 0], 0xff, ERROR_MARGIN);
71+
zassert_within(rgb24frame_out[(0) * p + (w - 1) * 3 + 1], 0x00, ERROR_MARGIN);
72+
zassert_within(rgb24frame_out[(0) * p + (w - 1) * 3 + 2], 0x7f, ERROR_MARGIN);
73+
zassert_within(rgb24frame_out[(h / 2 - 1) * p + (w / 2 + 1) * 3 + 0], 0xff, ERROR_MARGIN);
74+
zassert_within(rgb24frame_out[(h / 2 - 1) * p + (w / 2 + 1) * 3 + 1], 0x00, ERROR_MARGIN);
75+
zassert_within(rgb24frame_out[(h / 2 - 1) * p + (w / 2 + 1) * 3 + 2], 0x7f, ERROR_MARGIN);
76+
77+
/* Test bottom right quadrant */
78+
zassert_within(rgb24frame_out[(h - 1) * p + (w - 1) * 3 + 0], 0xff, ERROR_MARGIN);
79+
zassert_within(rgb24frame_out[(h - 1) * p + (w - 1) * 3 + 1], 0xff, ERROR_MARGIN);
80+
zassert_within(rgb24frame_out[(h - 1) * p + (w - 1) * 3 + 2], 0x7f, ERROR_MARGIN);
81+
zassert_within(rgb24frame_out[(h / 2 + 1) * p + (w / 2 + 1) * 3 + 0], 0xff, ERROR_MARGIN);
82+
zassert_within(rgb24frame_out[(h / 2 + 1) * p + (w / 2 + 1) * 3 + 1], 0xff, ERROR_MARGIN);
83+
zassert_within(rgb24frame_out[(h / 2 + 1) * p + (w / 2 + 1) * 3 + 2], 0x7f, ERROR_MARGIN);
84+
}
85+
86+
ZTEST(lib_pixel_resize, test_pixel_resize_stream)
87+
{
88+
struct pixel_stream step_export_rgb24 = {
89+
.ring = RING_BUF_INIT(rgb24frame_out, sizeof(rgb24frame_out)),
90+
.pitch = WIDTH_OUT * 3,
91+
.width = WIDTH_OUT,
92+
.height = HEIGHT_OUT,
93+
.name = "[export]",
94+
};
95+
struct pixel_stream *step;
96+
97+
/* Generate test input data */
98+
for (uint16_t h = 0; h <= HEIGHT_IN; h++) {
99+
for (uint16_t w = 0; w < WIDTH_IN; w++) {
100+
rgb24frame_in[h * PITCH_IN + w * 3 + 0] = w < WIDTH_IN / 2 ? 0x00 : 0xff;
101+
rgb24frame_in[h * PITCH_IN + w * 3 + 1] = h < HEIGHT_IN / 2 ? 0x00 : 0xff;
102+
rgb24frame_in[h * PITCH_IN + w * 3 + 2] = 0x7f;
103+
}
104+
}
105+
106+
/* RGB24 */
107+
step = &root;
108+
step = step->next = &step_subsample_rgb24;
109+
step = step->next = &step_export_rgb24;
110+
test_resize();
111+
pixel_stream_get_all_input(&step_export_rgb24);
112+
113+
/* RGB565LE */
114+
step = &root;
115+
step = step->next = &step_rgb24_to_rgb565le;
116+
step = step->next = &step_subsample_rgb565;
117+
step = step->next = &step_rgb565le_to_rgb24;
118+
step = step->next = &step_export_rgb24;
119+
test_resize();
120+
pixel_stream_get_all_input(&step_export_rgb24);
121+
122+
/* RGB565BE */
123+
step = &root;
124+
step = step->next = &step_rgb24_to_rgb565be;
125+
step = step->next = &step_subsample_rgb565;
126+
step = step->next = &step_rgb565be_to_rgb24;
127+
step = step->next = &step_export_rgb24;
128+
test_resize();
129+
pixel_stream_get_all_input(&step_export_rgb24);
130+
}
131+
132+
ZTEST_SUITE(lib_pixel_resize, NULL, NULL, NULL, NULL, NULL);

tests/lib/pixel/testcase.yaml

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
tests:
2+
libraries.pixel:
3+
tags:
4+
- pixel
5+
integration_platforms:
6+
- qemu_cortex_m3
7+
- native_sim

0 commit comments

Comments
 (0)