monado/src/xrt/auxiliary/util/u_sink_deinterleaver.c

149 lines
3.1 KiB
C
Raw Normal View History

2021-07-09 20:36:57 +00:00
// Copyright 2019-2021, Collabora, Ltd.
2020-01-18 20:59:38 +00:00
// SPDX-License-Identifier: BSL-1.0
/*!
* @file
2020-06-03 16:43:30 +00:00
* @brief An @ref xrt_frame_sink that deinterleaves stereo frames.
2020-01-18 20:59:38 +00:00
* @author Pete Black <pblack@collabora.com>
* @author Jakob Bornecrantz <jakob@collabora.com>
* @ingroup aux_util
*/
#include "util/u_misc.h"
#include "util/u_sink.h"
#include "util/u_frame.h"
2021-07-09 20:36:57 +00:00
#include "util/u_trace_marker.h"
2020-01-18 20:59:38 +00:00
2020-06-03 16:43:30 +00:00
/*!
* An @ref xrt_frame_sink that deinterleaves stereo frames.
* @implements xrt_frame_sink
* @implements xrt_frame_node
*/
2020-01-18 20:59:38 +00:00
struct u_sink_deinterleaver
{
struct xrt_frame_sink base;
struct xrt_frame_node node;
struct xrt_frame_sink *downstream;
};
/*
*
* Helpers
*
*/
inline static void
L8_interleaved_to_L8(const uint8_t *input, uint8_t *l8a, uint8_t *l8b)
{
*l8a = input[0];
*l8b = input[1];
}
static void
2021-01-14 14:13:48 +00:00
from_L8_interleaved_to_L8(struct xrt_frame *frame, uint32_t w, uint32_t h, size_t stride, const uint8_t *data)
2020-01-18 20:59:38 +00:00
{
2021-07-09 20:36:57 +00:00
SINK_TRACE_MARKER();
2020-01-18 20:59:38 +00:00
uint32_t half_w = w / 2;
for (uint32_t y = 0; y < h; y++) {
const uint8_t *src = data + (y * stride);
uint8_t *dst = frame->data + (y * frame->stride);
for (uint32_t x = 0; x < half_w; x++) {
L8_interleaved_to_L8(src, dst, dst + half_w);
dst += 1;
src += 2;
}
}
}
/*
*
* Helpers
*
*/
static void
2021-07-09 20:36:57 +00:00
deinterleave_frame(struct xrt_frame_sink *xfs, struct xrt_frame *xf)
2020-01-18 20:59:38 +00:00
{
2021-07-09 20:36:57 +00:00
SINK_TRACE_MARKER();
2020-01-18 20:59:38 +00:00
struct u_sink_deinterleaver *de = (struct u_sink_deinterleaver *)xfs;
if (xf->stereo_format != XRT_STEREO_FORMAT_INTERLEAVED) {
de->downstream->push_frame(de->downstream, xf);
return;
}
if (xf->format != XRT_FORMAT_L8) {
de->downstream->push_frame(de->downstream, xf);
return;
}
enum xrt_format format = XRT_FORMAT_L8;
uint32_t w = xf->width;
uint32_t h = xf->height;
size_t stride = xf->stride;
const uint8_t *data = xf->data;
struct xrt_frame *frame = NULL;
u_frame_create_one_off(format, w, h, &frame);
// Copy directly from original frame.
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 = XRT_STEREO_FORMAT_SBS;
// Copy the data.
from_L8_interleaved_to_L8(frame, w, h, stride, data);
// Push downstream.
de->downstream->push_frame(de->downstream, frame);
// Refcount in case it's being held downstream.
xrt_frame_reference(&frame, NULL);
}
static void
2021-07-09 20:36:57 +00:00
deinterleave_break_apart(struct xrt_frame_node *node)
{
// Noop
}
2020-01-18 20:59:38 +00:00
static void
2021-07-09 20:36:57 +00:00
deinterleave_destroy(struct xrt_frame_node *node)
2020-01-18 20:59:38 +00:00
{
2021-01-14 14:13:48 +00:00
struct u_sink_deinterleaver *de = container_of(node, struct u_sink_deinterleaver, node);
2020-01-18 20:59:38 +00:00
free(de);
}
/*
*
* Exported functions.
*
*/
void
u_sink_deinterleaver_create(struct xrt_frame_context *xfctx,
struct xrt_frame_sink *downstream,
struct xrt_frame_sink **out_xfs)
{
2021-01-14 14:13:48 +00:00
struct u_sink_deinterleaver *de = U_TYPED_CALLOC(struct u_sink_deinterleaver);
2020-01-18 20:59:38 +00:00
2021-07-09 20:36:57 +00:00
de->base.push_frame = deinterleave_frame;
de->node.break_apart = deinterleave_break_apart;
de->node.destroy = deinterleave_destroy;
2020-01-18 20:59:38 +00:00
de->downstream = downstream;
*out_xfs = &de->base;
}