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;
|
|
|
|
|
2021-08-23 15:28:30 +00:00
|
|
|
xrt_frame_context_add(xfctx, &de->node);
|
|
|
|
|
2020-01-18 20:59:38 +00:00
|
|
|
*out_xfs = &de->base;
|
|
|
|
}
|