Skip to content

Commit d4a4f00

Browse files
committed
samples: lib: pixel: show usage of the pixel library
The newly introduced lib/pixel features utilities that help composing video pipelines together for the purpose of stream processing, as well as debug utilities. Signed-off-by: Josuah Demangeon <me@josuah.net>
1 parent e81c2a0 commit d4a4f00

22 files changed

+518
-0
lines changed

samples/lib/lib.rst

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
.. zephyr:code-sample-category:: lib
2+
:name: Libraries
3+
:show-listing:
4+
:live-search:
5+
6+
These samples demonstrate how to use the libraries present in Zephyr.

samples/lib/pixel/pixel.rst

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
.. zephyr:code-sample-category:: lib_pixel
2+
:name: Pixel Library
3+
:show-listing:
4+
:live-search:
5+
6+
These samples demonstrate how to use the Pixel processing library of Zephyr.
7+
8+
These samples can be used as starting point for test benches that print an input image,
9+
perform some custom processing, and print the color image back along with the logs directly
10+
on the terminal.
11+
12+
The color rendering of the ``truecolor`` printing functions will be accurate RGB output.
13+
14+
This helps debugging individual functions before integrating it into a larger streams.
+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+
5+
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
6+
project(pixel)
7+
8+
target_sources(app PRIVATE src/main.c)

samples/lib/pixel/print/README.rst

+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
.. zephyr:code-sample:: lib_pixel_print
2+
:name: Pixel Printiing Library
3+
4+
Print images on the console.
5+
6+
Overview
7+
********
8+
9+
A sample showcasing how to make use of the pixel library to visualize an image or histogram data
10+
by printing it out on the console using `ANSI escape codes`_.
11+
12+
This allow interleaving debug logs with small previews of the image as a way to debug image data.
13+
14+
.. _ANSI escape codes: https://en.wikipedia.org/wiki/ANSI_escape_code
15+
16+
Building and Running
17+
********************
18+
19+
This application can be built and executed on QEMU as follows:
20+
21+
.. zephyr-app-commands::
22+
:zephyr-app: samples/lib/pixel/print
23+
:host-os: unix
24+
:board: native_sim
25+
:goals: run
26+
:compact:
27+
28+
To build for another board, change "native_sim" above to that board's name.
29+
30+
Sample Output
31+
=============
32+
33+
.. code-block:: console

samples/lib/pixel/print/prj.conf

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
CONFIG_PIXEL=y
2+
CONFIG_ASSERT=y
3+
CONFIG_LOG=y
4+
5+
# Required to make sure the test harnesses catch the log output
6+
CONFIG_LOG_MODE_IMMEDIATE=y

samples/lib/pixel/print/sample.yaml

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
sample:
2+
description: Pixel Print sample, print images in the terminal for debug purpose
3+
name: pixel print
4+
common:
5+
min_ram: 32
6+
tags: pixel
7+
integration_platforms:
8+
- native_sim
9+
harness: console
10+
harness_config:
11+
type: one_line
12+
regex:
13+
- "truecolor:"
14+
- "256color:"
15+
- "hexdump:"
16+
- "histogram"
17+
tests:
18+
sample.pixel.print:
19+
tags: pixel

samples/lib/pixel/print/src/main.c

+67
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
/*
2+
* Copyright (c) 2025 tinyVision.ai Inc.
3+
*
4+
* SPDX-License-Identifier: Apache-2.0
5+
*/
6+
7+
#include <stdint.h>
8+
9+
#include <zephyr/sys/util.h>
10+
#include <zephyr/pixel/print.h>
11+
#include <zephyr/logging/log.h>
12+
13+
LOG_MODULE_REGISTER(app, LOG_LEVEL_INF);
14+
15+
static uint8_t rgb24frame[16 * 32 * 3];
16+
17+
void print_image(void)
18+
{
19+
const uint8_t beg[] = {0x00, 0x70, 0xc5};
20+
const uint8_t end[] = {0x79, 0x29, 0xd2};
21+
22+
/* Generate an image with a gradient of the two colors above */
23+
for (size_t i = 0, size = sizeof(rgb24frame); i + 3 <= size; i += 3) {
24+
rgb24frame[i + 0] = (beg[0] * (size - i) + end[0] * i) / size;
25+
rgb24frame[i + 1] = (beg[1] * (size - i) + end[1] * i) / size;
26+
rgb24frame[i + 2] = (beg[2] * (size - i) + end[2] * i) / size;
27+
}
28+
29+
LOG_INF("Printing the gradient #%02x%02x%02x -> #%02x%02x%02x",
30+
beg[0], beg[1], beg[2], end[0], end[1], end[2]);
31+
32+
LOG_INF("hexdump:");
33+
pixel_print_rgb24frame_hex(rgb24frame, sizeof(rgb24frame), 16, 32);
34+
35+
LOG_INF("truecolor:");
36+
pixel_print_rgb24frame_truecolor(rgb24frame, sizeof(rgb24frame), 16, 32);
37+
38+
LOG_INF("256color:");
39+
pixel_print_rgb24frame_256color(rgb24frame, sizeof(rgb24frame), 16, 32);
40+
}
41+
42+
void print_histogram(void)
43+
{
44+
static const uint16_t rgb24hist[] = {
45+
9, 4, 7, 1, 0, 5, 1, 0, 0, 2, 2, 3, 0, 1, 3, 0,
46+
7, 6, 5, 1, 1, 4, 2, 0, 1, 2, 3, 4, 1, 1, 2, 2,
47+
8, 4, 7, 4, 2, 3, 1, 2, 2, 2, 2, 2, 0, 0, 1, 1,
48+
};
49+
50+
static const uint16_t y8hist[] = {
51+
8, 5, 6, 2, 1, 4, 1, 1, 1, 2, 3, 3, 1, 1, 2, 1,
52+
};
53+
54+
LOG_INF("Printing a histogram of %zu RGB buckets", ARRAY_SIZE(rgb24hist));
55+
pixel_print_rgb24hist(rgb24hist, ARRAY_SIZE(rgb24hist), 8);
56+
57+
LOG_INF("Printing a histogram of %zu Y (luma) buckets", ARRAY_SIZE(y8hist));
58+
pixel_print_y8hist(y8hist, ARRAY_SIZE(y8hist), 8);
59+
}
60+
61+
int main(void)
62+
{
63+
print_image();
64+
print_histogram();
65+
66+
return 0;
67+
}
+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+
5+
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
6+
project(pixel)
7+
8+
target_sources(app PRIVATE src/main.c)

samples/lib/pixel/resize/README.rst

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
.. zephyr:code-sample:: lib_pixel_resize
2+
:name: Pixel Resizing Library
3+
4+
Resize an image using subsampling.
5+
6+
Overview
7+
********
8+
9+
A sample showcasing how to make use of the pixel library to resize an input image to a smaller or
10+
bigger output image, using the subsampling method. This helps generating a smaller preview of an
11+
input image.
12+
13+
The input and output are printed as preview images on the console.
14+
15+
Building and Running
16+
********************
17+
18+
This application can be built and executed on the native simulator as follows:
19+
20+
.. zephyr-app-commands::
21+
:zephyr-app: samples/lib/pixel/resize
22+
:host-os: unix
23+
:board: native_sim
24+
:goals: run
25+
:compact:
26+
27+
To build for another board, change "native_sim" above to that board's name.
28+
29+
Sample Output
30+
=============
31+
32+
.. code-block:: console

samples/lib/pixel/resize/prj.conf

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
CONFIG_PIXEL=y
2+
CONFIG_ASSERT=y
3+
CONFIG_LOG=y
4+
5+
# Required to make sure the test harnesses catch the log output
6+
CONFIG_LOG_MODE_IMMEDIATE=y

samples/lib/pixel/resize/sample.yaml

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
sample:
2+
description: Pixel Resize sample, down-scale/up-scale an image
3+
name: pixel resize
4+
common:
5+
min_ram: 32
6+
tags: pixel
7+
integration_platforms:
8+
- native_sim
9+
harness: console
10+
harness_config:
11+
type: one_line
12+
regex:
13+
- "output image, bigger,"
14+
- "output image, smaller,"
15+
tests:
16+
sample.pixel.resize:
17+
tags: pixel

samples/lib/pixel/resize/src/main.c

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
/*
2+
* Copyright (c) 2025 tinyVision.ai Inc.
3+
*
4+
* SPDX-License-Identifier: Apache-2.0
5+
*/
6+
7+
#include <stdint.h>
8+
9+
#include <zephyr/sys/util.h>
10+
#include <zephyr/pixel/resize.h>
11+
#include <zephyr/pixel/print.h>
12+
#include <zephyr/logging/log.h>
13+
14+
LOG_MODULE_REGISTER(app, LOG_LEVEL_INF);
15+
16+
static void gradient(uint8_t *rgb24buf, size_t size, const uint8_t beg[3], const uint8_t end[3])
17+
{
18+
for (int i = 0; i + 3 <= size; i += 3) {
19+
rgb24buf[i + 0] = (beg[0] * (size - i) + end[0] * i) / size;
20+
rgb24buf[i + 1] = (beg[1] * (size - i) + end[1] * i) / size;
21+
rgb24buf[i + 2] = (beg[2] * (size - i) + end[2] * i) / size;
22+
}
23+
}
24+
25+
static uint8_t rgb24frame0[32 * 16 * 3];
26+
static uint8_t rgb24frame1[120 * 20 * 3];
27+
static uint8_t rgb24frame2[10 * 10 * 3];
28+
29+
int main(void)
30+
{
31+
const uint8_t beg[] = {0x00, 0x70, 0xc5};
32+
const uint8_t end[] = {0x79, 0x29, 0xd2};
33+
34+
LOG_INF("input image, 32x16, %zu bytes:", sizeof(rgb24frame0));
35+
gradient(rgb24frame0, sizeof(rgb24frame0), beg, end);
36+
pixel_print_rgb24frame_truecolor(rgb24frame0, sizeof(rgb24frame0), 32, 16);
37+
38+
LOG_INF("output image, bigger, 120x16, %zu bytes:", sizeof(rgb24frame1));
39+
pixel_subsample_rgb24frame(rgb24frame0, 32, 16, rgb24frame1, 120, 20);
40+
pixel_print_rgb24frame_truecolor(rgb24frame1, sizeof(rgb24frame1), 120, 20);
41+
42+
LOG_INF("output image, smaller, 10x10, %zu bytes:", sizeof(rgb24frame2));
43+
pixel_subsample_rgb24frame(rgb24frame0, 32, 16, rgb24frame2, 10, 10);
44+
pixel_print_rgb24frame_truecolor(rgb24frame2, sizeof(rgb24frame2), 10, 10);
45+
46+
return 0;
47+
}
+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+
5+
find_package(Zephyr REQUIRED HINTS $ENV{ZEPHYR_BASE})
6+
project(pixel)
7+
8+
target_sources(app PRIVATE src/main.c)

samples/lib/pixel/stats/README.rst

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
.. zephyr:code-sample:: lib_pixel_stats
2+
:name: Pixel Statistics Library
3+
4+
Collect statistics of an image.
5+
6+
Overview
7+
********
8+
9+
A sample showcasing how to make use of the pixel library to collect statistics of an input image
10+
buffer and display both the image and statistics out on the console.
11+
12+
Building and Running
13+
********************
14+
15+
This application can be built and executed on the native simulator as follows:
16+
17+
.. zephyr-app-commands::
18+
:zephyr-app: samples/lib/pixel/stats
19+
:host-os: unix
20+
:board: native_sim
21+
:goals: run
22+
:compact:
23+
24+
To build for another board, change "native_sim" above to that board's name.
25+
26+
Sample Output
27+
=============
28+
29+
.. code-block:: console

samples/lib/pixel/stats/prj.conf

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
CONFIG_PIXEL=y
2+
CONFIG_ASSERT=y
3+
CONFIG_LOG=y
4+
5+
# Required to make sure the test harnesses catch the log output
6+
CONFIG_LOG_MODE_IMMEDIATE=y

samples/lib/pixel/stats/sample.yaml

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
sample:
2+
description: Pixel Stats sample, collect statistics of an input buffer
3+
name: pixel stats
4+
common:
5+
min_ram: 32
6+
tags: pixel
7+
integration_platforms:
8+
- native_sim
9+
harness: console
10+
harness_config:
11+
type: one_line
12+
regex:
13+
- "RGB histogram of the image"
14+
- "RGB channel averages of the image"
15+
tests:
16+
sample.pixel.stats:
17+
tags: pixel

samples/lib/pixel/stats/src/main.c

+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
/*
2+
* Copyright (c) 2025 tinyVision.ai Inc.
3+
*
4+
* SPDX-License-Identifier: Apache-2.0
5+
*/
6+
7+
#include <stdint.h>
8+
9+
#include <zephyr/sys/util.h>
10+
#include <zephyr/pixel/stats.h>
11+
#include <zephyr/pixel/print.h>
12+
#include <zephyr/logging/log.h>
13+
14+
LOG_MODULE_REGISTER(app, LOG_LEVEL_INF);
15+
16+
#define NVAL 100
17+
18+
static const uint8_t image_rgb24[20 * 4 * 3] = {
19+
0x47, 0x84, 0xee, 0x46, 0x84, 0xee, 0x47, 0x84, 0xee, 0x46, 0x83, 0xee, 0x47, 0x84, 0xee,
20+
0x78, 0xaa, 0xec, 0x74, 0xb2, 0xe0, 0x67, 0xaa, 0xdd, 0x78, 0xb2, 0xef, 0x39, 0x8c, 0xf1,
21+
0x3a, 0x8c, 0xf2, 0x39, 0x8b, 0xf1, 0x3a, 0x8b, 0xf1, 0x3a, 0x8b, 0xf1, 0x3a, 0x8b, 0xf1,
22+
0x3a, 0x8b, 0xf1, 0x3b, 0x8b, 0xf1, 0x3b, 0x8b, 0xf1, 0x3b, 0x8b, 0xf1, 0x3b, 0x8a, 0xf1,
23+
0x47, 0x82, 0xed, 0x47, 0x82, 0xed, 0x47, 0x82, 0xee, 0x47, 0x82, 0xed, 0x47, 0x82, 0xed,
24+
0x47, 0x82, 0xed, 0x5d, 0x93, 0xed, 0x5f, 0x9d, 0xeb, 0x3a, 0x8a, 0xf1, 0x3a, 0x8a, 0xf0,
25+
0x3a, 0x8a, 0xf1, 0x3a, 0x8a, 0xf0, 0x3b, 0x8a, 0xf0, 0x3b, 0x8a, 0xf0, 0x3b, 0x8a, 0xf0,
26+
0x3b, 0x89, 0xf0, 0x3b, 0x8a, 0xf0, 0x3b, 0x89, 0xf0, 0x3c, 0x89, 0xf0, 0x3c, 0x89, 0xf0,
27+
0x49, 0x82, 0xee, 0x49, 0x82, 0xed, 0x49, 0x82, 0xee, 0x48, 0x82, 0xed, 0x49, 0x82, 0xee,
28+
0x49, 0x82, 0xed, 0x73, 0x92, 0xe9, 0x50, 0x65, 0xd4, 0x4c, 0x93, 0xf2, 0x3c, 0x8a, 0xf1,
29+
0x3c, 0x8a, 0xf1, 0x3c, 0x8a, 0xf0, 0x3c, 0x8a, 0xf1, 0x3c, 0x89, 0xf0, 0x3d, 0x8a, 0xf1,
30+
0x3d, 0x89, 0xf0, 0x3d, 0x89, 0xf0, 0x3d, 0x89, 0xf0, 0x3e, 0x89, 0xf0, 0x3d, 0x89, 0xf0,
31+
0x4a, 0x81, 0xed, 0x49, 0x81, 0xed, 0x4a, 0x81, 0xed, 0x49, 0x81, 0xed, 0x49, 0x81, 0xed,
32+
0x71, 0x8c, 0xe5, 0x3e, 0x4c, 0xcc, 0x3d, 0x4c, 0xcb, 0x65, 0x85, 0xe1, 0x3d, 0x89, 0xf0,
33+
0x3d, 0x89, 0xf0, 0x3d, 0x88, 0xf0, 0x3d, 0x89, 0xf0, 0x3d, 0x88, 0xf0, 0x3e, 0x88, 0xf0,
34+
0x3e, 0x88, 0xf0, 0x3e, 0x88, 0xf0, 0x3e, 0x88, 0xf0, 0x3f, 0x88, 0xf0, 0x3e, 0x88, 0xef,
35+
};
36+
37+
static uint16_t rgb24hist[3 * 64];
38+
static uint8_t rgb24avg[3];
39+
40+
int main(void)
41+
{
42+
LOG_INF("Input image preview:");
43+
pixel_print_rgb24frame_truecolor(image_rgb24, sizeof(image_rgb24), 20, 4);
44+
45+
LOG_INF("RGB histogram of the image");
46+
pixel_rgb24frame_to_rgb24hist(image_rgb24, sizeof(image_rgb24),
47+
rgb24hist, ARRAY_SIZE(rgb24hist), NVAL);
48+
pixel_print_rgb24hist(rgb24hist, ARRAY_SIZE(rgb24hist), 16);
49+
50+
LOG_INF("RGB channel averages of the image");
51+
pixel_rgb24frame_to_rgb24avg(image_rgb24, sizeof(image_rgb24), rgb24avg, NVAL);
52+
LOG_INF("- R: 0x%02x/0xff", rgb24avg[0]);
53+
LOG_INF("- G: 0x%02x/0xff", rgb24avg[1]);
54+
LOG_INF("- B: 0x%02x/0xff", rgb24avg[2]);
55+
56+
return 0;
57+
}

0 commit comments

Comments
 (0)