// Copyright 2019-2021, Collabora, Ltd. // SPDX-License-Identifier: BSL-1.0 /*! * @file * @brief @ref xrt_frame_sink converters and other helpers. * @author Jakob Bornecrantz * @ingroup aux_util */ #include "xrt/xrt_config_have.h" #include "util/u_logging.h" #include "util/u_misc.h" #include "util/u_sink.h" #include "util/u_frame.h" #include "util/u_format.h" #include "util/u_trace_marker.h" #include #ifdef XRT_HAVE_JPEG #include "jpeglib.h" #endif /* * * Structs * */ /*! * An @ref xrt_frame_sink that converts frames. * @implements xrt_frame_sink * @implements xrt_frame_node */ struct u_sink_converter { struct xrt_frame_sink base; struct xrt_frame_node node; struct xrt_frame_sink *downstream; struct xrt_frame_sink *downstream2; enum xrt_format format; }; /* * * L8 functions. * */ static void from_L8_to_R8G8B8(struct xrt_frame *dst_frame, uint32_t w, uint32_t h, size_t stride, const uint8_t *data) { SINK_TRACE_MARKER(); for (uint32_t y = 0; y < h; y++) { for (uint32_t x = 0; x < w; x += 1) { const uint8_t *src = data; uint8_t *dst = dst_frame->data; src = src + (y * stride) + x; dst = dst + (y * dst_frame->stride) + (x * 3); dst[2] = dst[1] = dst[0] = src[0]; } } } /* * * YUV functions. * */ static inline int clamp_to_byte(int v) { if (v < 0) { return 0; } if (v >= 255) { return 255; } return v; } static inline uint32_t YUV444_to_RGBX8888(int y, int u, int v) { int C = y - 16; int D = u - 128; int E = v - 128; int R = clamp_to_byte((298 * C + 409 * E + 128) >> 8); int G = clamp_to_byte((298 * C - 100 * D - 209 * E + 128) >> 8); int B = clamp_to_byte((298 * C + 516 * D + 128) >> 8); return B << 16 | G << 8 | R; } #undef USE_TABLE #ifdef USE_TABLE static uint32_t lookup_YUV_to_RGBX[256][256][256] = {0}; static void generate_lookup_YUV_to_RGBX() { if (lookup_YUV_to_RGBX[255][255][255] != 0) { return; } int y; int u; int v; for (y = 0; y < 256; y++) { for (u = 0; u < 256; u++) { for (v = 0; v < 256; v++) { lookup_YUV_to_RGBX[y][u][v] = YUV444_to_RGBX8888(y, u, v); } } } } #endif XRT_MAYBE_UNUSED inline static void YUYV422_to_R8G8B8X8(const uint8_t *input, uint32_t *rgb1, uint32_t *rgb2) { uint8_t y0 = input[0]; uint8_t u = input[1]; uint8_t y1 = input[2]; uint8_t v = input[3]; #ifdef USE_TABLE *rgb1 = lookup_YUV_to_RGBX[y0][u][v]; *rgb2 = lookup_YUV_to_RGBX[y1][u][v]; #else *rgb1 = YUV444_to_RGBX8888(y0, u, v); *rgb2 = YUV444_to_RGBX8888(y1, u, v); #endif } inline static void YUYV422_to_R8G8B8(const uint8_t *input, uint8_t *dst) { uint8_t y0 = input[0]; uint8_t u = input[1]; uint8_t y1 = input[2]; uint8_t v = input[3]; #ifdef USE_TABLE uint8_t *rgb1 = (uint8_t *)&lookup_YUV_to_RGBX[y0][u][v]; uint8_t *rgb2 = (uint8_t *)&lookup_YUV_to_RGBX[y1][u][v]; #else uint32_t rgb1v = YUV444_to_RGBX8888(y0, u, v); uint32_t rgb2v = YUV444_to_RGBX8888(y1, u, v); uint8_t *rgb1 = (uint8_t *)&rgb1v; uint8_t *rgb2 = (uint8_t *)&rgb2v; #endif dst[0] = rgb1[0]; dst[1] = rgb1[1]; dst[2] = rgb1[2]; dst[3] = rgb2[0]; dst[4] = rgb2[1]; dst[5] = rgb2[2]; } inline static void UYVY422_to_R8G8B8(const uint8_t *input, uint8_t *dst) { uint8_t u = input[0]; uint8_t y0 = input[1]; uint8_t v = input[2]; uint8_t y1 = input[3]; #ifdef USE_TABLE uint8_t *rgb1 = (uint8_t *)&lookup_YUV_to_RGBX[y0][u][v]; uint8_t *rgb2 = (uint8_t *)&lookup_YUV_to_RGBX[y1][u][v]; #else uint32_t rgb1v = YUV444_to_RGBX8888(y0, u, v); uint32_t rgb2v = YUV444_to_RGBX8888(y1, u, v); uint8_t *rgb1 = (uint8_t *)&rgb1v; uint8_t *rgb2 = (uint8_t *)&rgb2v; #endif dst[0] = rgb1[0]; dst[1] = rgb1[1]; dst[2] = rgb1[2]; dst[3] = rgb2[0]; dst[4] = rgb2[1]; dst[5] = rgb2[2]; } inline static void YUV444_to_R8G8B8(const uint8_t *input, uint8_t *dst) { uint8_t y = input[0]; uint8_t u = input[1]; uint8_t v = input[2]; #ifdef USE_TABLE uint8_t *rgb = (uint8_t *)&lookup_YUV_to_RGBX[y][u][v]; #else uint32_t rgbv = YUV444_to_RGBX8888(y, u, v); uint8_t *rgb = (uint8_t *)&rgbv; #endif dst[0] = rgb[0]; dst[1] = rgb[1]; dst[2] = rgb[2]; } static void from_YUYV422_to_R8G8B8(struct xrt_frame *dst_frame, uint32_t w, uint32_t h, size_t stride, const uint8_t *data) { SINK_TRACE_MARKER(); for (uint32_t y = 0; y < h; y++) { for (uint32_t x = 0; x < w; x += 2) { const uint8_t *src = data; uint8_t *dst = dst_frame->data; src = src + (y * stride) + (x * 2); dst = dst + (y * dst_frame->stride) + (x * 3); YUYV422_to_R8G8B8(src, dst); } } } static void from_UYVY422_to_R8G8B8(struct xrt_frame *dst_frame, uint32_t w, uint32_t h, size_t stride, const uint8_t *data) { SINK_TRACE_MARKER(); for (uint32_t y = 0; y < h; y++) { for (uint32_t x = 0; x < w; x += 2) { const uint8_t *src = data; uint8_t *dst = dst_frame->data; src = src + (y * stride) + (x * 2); dst = dst + (y * dst_frame->stride) + (x * 3); UYVY422_to_R8G8B8(src, dst); } } } static void from_YUV888_to_R8G8B8(struct xrt_frame *dst_frame, uint32_t w, uint32_t h, size_t stride, const uint8_t *data) { SINK_TRACE_MARKER(); for (uint32_t y = 0; y < h; y++) { for (uint32_t x = 0; x < w; x++) { const uint8_t *src = data; uint8_t *dst = dst_frame->data; src = src + (y * stride) + (x * 3); dst = dst + (y * dst_frame->stride) + (x * 3); YUV444_to_R8G8B8(src, dst); } } } /* * * MJPEG * */ #ifdef XRT_HAVE_JPEG static bool check_header(size_t size, const uint8_t *data) { if (size < 16) { U_LOG_E("Invalid JPEG file size! %u", (uint32_t)size); return false; } if (data[0] != 0xFF || data[1] != 0xD8) { U_LOG_E("Invalid file header! 0x%02X 0x%02X", data[0], data[1]); return false; } return true; } static bool from_MJPEG_to_R8G8B8(struct xrt_frame *dst_frame, size_t size, const uint8_t *data) { SINK_TRACE_MARKER(); if (!check_header(size, data)) { return false; } struct jpeg_decompress_struct cinfo = {0}; struct jpeg_error_mgr jerr = {0}; cinfo.err = jpeg_std_error(&jerr); jerr.trace_level = 0; jpeg_create_decompress(&cinfo); jpeg_mem_src(&cinfo, data, size); int ret = jpeg_read_header(&cinfo, TRUE); if (ret != JPEG_HEADER_OK) { jpeg_destroy_decompress(&cinfo); return false; } cinfo.out_color_space = JCS_RGB; jpeg_start_decompress(&cinfo); uint8_t *moving_ptr = dst_frame->data; uint32_t scanlines_read = 0; while (scanlines_read < cinfo.image_height) { int read_count = jpeg_read_scanlines(&cinfo, &moving_ptr, 16); moving_ptr += read_count * dst_frame->stride; scanlines_read += read_count; } jpeg_finish_decompress(&cinfo); jpeg_destroy_decompress(&cinfo); return true; } static bool from_MJPEG_to_YUV888(struct xrt_frame *dst_frame, size_t size, const uint8_t *data) { SINK_TRACE_MARKER(); if (!check_header(size, data)) { return false; } struct jpeg_decompress_struct cinfo = {0}; struct jpeg_error_mgr jerr = {0}; cinfo.err = jpeg_std_error(&jerr); jerr.trace_level = 0; jpeg_create_decompress(&cinfo); jpeg_mem_src(&cinfo, data, size); int ret = jpeg_read_header(&cinfo, TRUE); if (ret != JPEG_HEADER_OK) { jpeg_destroy_decompress(&cinfo); return false; } cinfo.out_color_space = JCS_YCbCr; jpeg_start_decompress(&cinfo); uint8_t *moving_ptr = dst_frame->data; uint32_t scanlines_read = 0; while (scanlines_read < cinfo.image_height) { int read_count = jpeg_read_scanlines(&cinfo, &moving_ptr, 16); moving_ptr += read_count * dst_frame->stride; scanlines_read += read_count; } jpeg_finish_decompress(&cinfo); jpeg_destroy_decompress(&cinfo); return true; } #endif /* * * Bayer * */ static void from_BAYER_GR8_to_R8G8B8(struct xrt_frame *dst_frame, uint32_t w, uint32_t h, size_t stride, const uint8_t *data) { SINK_TRACE_MARKER(); const uint8_t *src_data = data; uint32_t src_stride = stride; uint8_t *dst_data = dst_frame->data; uint32_t dst_stride = dst_frame->stride; for (uint32_t y = 0; y < h; y++) { const uint8_t *src0 = src_data + (y * 2) * src_stride; const uint8_t *src1 = src_data + (y * 2 + 1) * src_stride; uint8_t *dst = dst_data + (y * dst_stride); for (uint32_t x = 0; x < w; x++) { uint8_t g0 = src0[0]; uint8_t r = src0[1]; uint8_t b = src1[0]; uint8_t g1 = src1[1]; dst[0] = r; dst[1] = (g0 + g1) / 2; dst[2] = b; src0 += 2; src1 += 2; dst += 3; } } } /* * * Misc functions. * */ /*! * Creates a frame that the conversion should happen to, allows to set the size. * * @todo Allocate from a pool of frames. */ static bool create_frame_with_format_of_size( struct xrt_frame *xf, uint32_t w, uint32_t h, enum xrt_format format, struct xrt_frame **out_frame) { struct xrt_frame *frame = NULL; u_frame_create_one_off(format, w, h, &frame); if (frame == NULL) { U_LOG_E("Failed to create target frame!"); *out_frame = NULL; return false; } frame->timestamp = xf->timestamp; frame->source_timestamp = xf->source_timestamp; frame->source_sequence = xf->source_sequence; frame->source_id = xf->source_id; frame->stereo_format = xf->stereo_format; *out_frame = frame; return true; } /*! * Creates a frame that the conversion should happen to. */ static bool create_frame_with_format(struct xrt_frame *xf, enum xrt_format format, struct xrt_frame **out_frame) { return create_frame_with_format_of_size(xf, xf->width, xf->height, format, out_frame); } static void convert_frame_r8g8b8_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf) { SINK_TRACE_MARKER(); struct u_sink_converter *s = (struct u_sink_converter *)xs; struct xrt_frame *converted = NULL; switch (xf->format) { case XRT_FORMAT_L8: case XRT_FORMAT_R8G8B8: s->downstream->push_frame(s->downstream, xf); return; case XRT_FORMAT_BAYER_GR8:; uint32_t w = xf->width / 2; uint32_t h = xf->height / 2; if (!create_frame_with_format_of_size(xf, w, h, XRT_FORMAT_R8G8B8, &converted)) { return; } from_BAYER_GR8_to_R8G8B8(converted, w, h, xf->stride, xf->data); break; case XRT_FORMAT_YUYV422: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_YUYV422_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; case XRT_FORMAT_UYVY422: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_UYVY422_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; case XRT_FORMAT_YUV888: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_YUV888_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; #ifdef XRT_HAVE_JPEG case XRT_FORMAT_MJPEG: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } if (!from_MJPEG_to_R8G8B8(converted, xf->size, xf->data)) { // Make sure to free frame when we fail to decode. xrt_frame_reference(&converted, NULL); return; } break; #endif default: U_LOG_E("Can not convert from '%s' to R8G8B8 or L8!", u_format_str(xf->format)); return; } s->downstream->push_frame(s->downstream, converted); // Refcount in case it's being held downstream. xrt_frame_reference(&converted, NULL); } static void convert_frame_r8g8b8_bayer_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf) { SINK_TRACE_MARKER(); struct u_sink_converter *s = (struct u_sink_converter *)xs; struct xrt_frame *converted = NULL; switch (xf->format) { case XRT_FORMAT_L8: case XRT_FORMAT_R8G8B8: case XRT_FORMAT_BAYER_GR8:; s->downstream->push_frame(s->downstream, xf); return; case XRT_FORMAT_YUYV422: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_YUYV422_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; case XRT_FORMAT_UYVY422: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_UYVY422_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; case XRT_FORMAT_YUV888: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_YUV888_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; #ifdef XRT_HAVE_JPEG case XRT_FORMAT_MJPEG: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } if (!from_MJPEG_to_R8G8B8(converted, xf->size, xf->data)) { // Make sure to free frame when we fail to decode. xrt_frame_reference(&converted, NULL); return; } break; #endif default: U_LOG_E("Can not convert from '%s' to R8G8B8 or L8!", u_format_str(xf->format)); return; } s->downstream->push_frame(s->downstream, converted); // Refcount in case it's being held downstream. xrt_frame_reference(&converted, NULL); } static void convert_frame_r8g8b8(struct xrt_frame_sink *xs, struct xrt_frame *xf) { SINK_TRACE_MARKER(); struct u_sink_converter *s = (struct u_sink_converter *)xs; struct xrt_frame *converted = NULL; switch (xf->format) { case XRT_FORMAT_R8G8B8: s->downstream->push_frame(s->downstream, xf); return; case XRT_FORMAT_L8: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_L8_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; case XRT_FORMAT_BAYER_GR8:; uint32_t w = xf->width / 2; uint32_t h = xf->height / 2; if (!create_frame_with_format_of_size(xf, w, h, XRT_FORMAT_R8G8B8, &converted)) { return; } from_BAYER_GR8_to_R8G8B8(converted, w, h, xf->stride, xf->data); break; case XRT_FORMAT_YUYV422: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_YUYV422_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; case XRT_FORMAT_UYVY422: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_UYVY422_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; case XRT_FORMAT_YUV888: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } from_YUV888_to_R8G8B8(converted, xf->width, xf->height, xf->stride, xf->data); break; #ifdef XRT_HAVE_JPEG case XRT_FORMAT_MJPEG: if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) { return; } if (!from_MJPEG_to_R8G8B8(converted, xf->size, xf->data)) { // Make sure to free frame when we fail to decode. xrt_frame_reference(&converted, NULL); return; } break; #endif default: U_LOG_E("Can not convert from '%s' to R8G8B8!", u_format_str(xf->format)); return; } s->downstream->push_frame(s->downstream, converted); // Refcount in case it's being held downstream. xrt_frame_reference(&converted, NULL); } static void convert_frame_rgb_yuv_yuyv_uyvy_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf) { SINK_TRACE_MARKER(); struct u_sink_converter *s = (struct u_sink_converter *)xs; struct xrt_frame *converted = NULL; switch (xf->format) { case XRT_FORMAT_R8G8B8: case XRT_FORMAT_L8: case XRT_FORMAT_YUYV422: case XRT_FORMAT_UYVY422: case XRT_FORMAT_YUV888: s->downstream->push_frame(s->downstream, xf); return; #ifdef XRT_HAVE_JPEG case XRT_FORMAT_MJPEG: if (!create_frame_with_format(xf, XRT_FORMAT_YUV888, &converted)) { return; } if (!from_MJPEG_to_YUV888(converted, xf->size, xf->data)) { return; } break; #endif default: U_LOG_E("Can not convert from '%s' to either R8G8B8, YUV, YUYV, UYVY or L8!", u_format_str(xf->format)); return; } s->downstream->push_frame(s->downstream, converted); // Refcount in case it's being held downstream. xrt_frame_reference(&converted, NULL); } static void convert_frame_yuv_yuyv_uyvy_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf) { SINK_TRACE_MARKER(); struct u_sink_converter *s = (struct u_sink_converter *)xs; struct xrt_frame *converted = NULL; switch (xf->format) { case XRT_FORMAT_L8: case XRT_FORMAT_YUYV422: case XRT_FORMAT_UYVY422: case XRT_FORMAT_YUV888: s->downstream->push_frame(s->downstream, xf); return; #ifdef XRT_HAVE_JPEG case XRT_FORMAT_MJPEG: if (!create_frame_with_format(xf, XRT_FORMAT_YUV888, &converted)) { return; } if (!from_MJPEG_to_YUV888(converted, xf->size, xf->data)) { return; } break; #endif default: U_LOG_E( "Can not convert from '%s' to either YUV, YUYV, UYVY or " "L8!", u_format_str(xf->format)); return; } s->downstream->push_frame(s->downstream, converted); // Refcount in case it's being held downstream. xrt_frame_reference(&converted, NULL); } static void convert_frame_yuv_or_yuyv(struct xrt_frame_sink *xs, struct xrt_frame *xf) { SINK_TRACE_MARKER(); struct u_sink_converter *s = (struct u_sink_converter *)xs; struct xrt_frame *converted = NULL; switch (xf->format) { case XRT_FORMAT_YUYV422: case XRT_FORMAT_YUV888: s->downstream->push_frame(s->downstream, xf); return; #ifdef XRT_HAVE_JPEG case XRT_FORMAT_MJPEG: if (!create_frame_with_format(xf, XRT_FORMAT_YUV888, &converted)) { return; } if (!from_MJPEG_to_YUV888(converted, xf->size, xf->data)) { return; } break; #endif default: U_LOG_E("Can not convert from '%s' to either YUV or YUYV!", u_format_str(xf->format)); return; } s->downstream->push_frame(s->downstream, converted); // Refcount in case it's being held downstream. xrt_frame_reference(&converted, NULL); } XRT_MAYBE_UNUSED static void convert_frame_bayer(struct xrt_frame_sink *xs, struct xrt_frame *xf) { SINK_TRACE_MARKER(); struct u_sink_converter *s = (struct u_sink_converter *)xs; uint32_t w = xf->width / 2; uint32_t h = xf->height / 2; struct xrt_frame *converted = NULL; if (!create_frame_with_format_of_size(xf, w, h, XRT_FORMAT_R8G8B8, &converted)) { return; } from_BAYER_GR8_to_R8G8B8(converted, w, h, xf->stride, xf->data); s->downstream->push_frame(s->downstream, converted); // Refcount in case it's being held downstream. xrt_frame_reference(&converted, NULL); } static void break_apart(struct xrt_frame_node *node) {} static void destroy(struct xrt_frame_node *node) { struct u_sink_converter *s = container_of(node, struct u_sink_converter, node); free(s); } /* * * "Exported" functions. * */ void u_sink_create_format_converter(struct xrt_frame_context *xfctx, enum xrt_format f, struct xrt_frame_sink *downstream, struct xrt_frame_sink **out_xfs) { if (f != XRT_FORMAT_R8G8B8) { U_LOG_E("Format '%s' not supported", u_format_str(f)); return; } #ifdef USE_TABLE generate_lookup_YUV_to_RGBX(); #endif struct u_sink_converter *s = U_TYPED_CALLOC(struct u_sink_converter); s->base.push_frame = convert_frame_r8g8b8; s->node.break_apart = break_apart; s->node.destroy = destroy; s->downstream = downstream; xrt_frame_context_add(xfctx, &s->node); *out_xfs = &s->base; } void u_sink_create_to_r8g8b8_or_l8(struct xrt_frame_context *xfctx, struct xrt_frame_sink *downstream, struct xrt_frame_sink **out_xfs) { struct u_sink_converter *s = U_TYPED_CALLOC(struct u_sink_converter); s->base.push_frame = convert_frame_r8g8b8_or_l8; s->node.break_apart = break_apart; s->node.destroy = destroy; s->downstream = downstream; #ifdef USE_TABLE generate_lookup_YUV_to_RGBX(); #endif xrt_frame_context_add(xfctx, &s->node); *out_xfs = &s->base; } void u_sink_create_to_r8g8b8_bayer_or_l8(struct xrt_frame_context *xfctx, struct xrt_frame_sink *downstream, struct xrt_frame_sink **out_xfs) { struct u_sink_converter *s = U_TYPED_CALLOC(struct u_sink_converter); s->base.push_frame = convert_frame_r8g8b8_bayer_or_l8; s->node.break_apart = break_apart; s->node.destroy = destroy; s->downstream = downstream; xrt_frame_context_add(xfctx, &s->node); *out_xfs = &s->base; } void u_sink_create_to_rgb_yuv_yuyv_uyvy_or_l8(struct xrt_frame_context *xfctx, struct xrt_frame_sink *downstream, struct xrt_frame_sink **out_xfs) { struct u_sink_converter *s = U_TYPED_CALLOC(struct u_sink_converter); s->base.push_frame = convert_frame_rgb_yuv_yuyv_uyvy_or_l8; s->node.break_apart = break_apart; s->node.destroy = destroy; s->downstream = downstream; xrt_frame_context_add(xfctx, &s->node); *out_xfs = &s->base; } void u_sink_create_to_yuv_yuyv_uyvy_or_l8(struct xrt_frame_context *xfctx, struct xrt_frame_sink *downstream, struct xrt_frame_sink **out_xfs) { struct u_sink_converter *s = U_TYPED_CALLOC(struct u_sink_converter); s->base.push_frame = convert_frame_yuv_yuyv_uyvy_or_l8; s->node.break_apart = break_apart; s->node.destroy = destroy; s->downstream = downstream; xrt_frame_context_add(xfctx, &s->node); *out_xfs = &s->base; } void u_sink_create_to_yuv_or_yuyv(struct xrt_frame_context *xfctx, struct xrt_frame_sink *downstream, struct xrt_frame_sink **out_xfs) { struct u_sink_converter *s = U_TYPED_CALLOC(struct u_sink_converter); s->base.push_frame = convert_frame_yuv_or_yuyv; s->node.break_apart = break_apart; s->node.destroy = destroy; s->downstream = downstream; xrt_frame_context_add(xfctx, &s->node); *out_xfs = &s->base; }