mirror of
https://gitlab.freedesktop.org/monado/monado.git
synced 2025-01-19 13:18:32 +00:00
u/sink: Add Bayer format converter
This commit is contained in:
parent
5b6e6bf6a6
commit
eec3446e5b
1
doc/changes/auxiliary/mr.665.md
Normal file
1
doc/changes/auxiliary/mr.665.md
Normal file
|
@ -0,0 +1 @@
|
||||||
|
u/sink: Add Bayer format converter.
|
|
@ -44,6 +44,15 @@ u_sink_create_to_r8g8b8_or_l8(struct xrt_frame_context *xfctx,
|
||||||
struct xrt_frame_sink *downstream,
|
struct xrt_frame_sink *downstream,
|
||||||
struct xrt_frame_sink **out_xfs);
|
struct xrt_frame_sink **out_xfs);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* @relatesalso xrt_frame_sink
|
||||||
|
* @relates xrt_frame_context
|
||||||
|
*/
|
||||||
|
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);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* @relatesalso xrt_frame_sink
|
* @relatesalso xrt_frame_sink
|
||||||
* @relates xrt_frame_context
|
* @relates xrt_frame_context
|
||||||
|
|
|
@ -341,6 +341,44 @@ from_MJPEG_to_YUV888(struct xrt_frame *dst_frame, size_t size, const uint8_t *da
|
||||||
#endif
|
#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)
|
||||||
|
{
|
||||||
|
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.
|
* Misc functions.
|
||||||
|
@ -349,15 +387,16 @@ from_MJPEG_to_YUV888(struct xrt_frame *dst_frame, size_t size, const uint8_t *da
|
||||||
|
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Creates a frame that the conversion should happen to.
|
* Creates a frame that the conversion should happen to, allows to set the size.
|
||||||
*
|
*
|
||||||
* @todo Allocate from a pool of frames.
|
* @todo Allocate from a pool of frames.
|
||||||
*/
|
*/
|
||||||
static bool
|
static bool
|
||||||
create_frame_with_format(struct xrt_frame *xf, enum xrt_format format, struct xrt_frame **out_frame)
|
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;
|
struct xrt_frame *frame = NULL;
|
||||||
u_frame_create_one_off(format, xf->width, xf->height, &frame);
|
u_frame_create_one_off(format, w, h, &frame);
|
||||||
if (frame == NULL) {
|
if (frame == NULL) {
|
||||||
U_LOG_E("Failed to create target frame!");
|
U_LOG_E("Failed to create target frame!");
|
||||||
*out_frame = NULL;
|
*out_frame = NULL;
|
||||||
|
@ -375,6 +414,15 @@ create_frame_with_format(struct xrt_frame *xf, enum xrt_format format, struct xr
|
||||||
return true;
|
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
|
static void
|
||||||
receive_frame_r8g8b8_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf)
|
receive_frame_r8g8b8_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf)
|
||||||
{
|
{
|
||||||
|
@ -385,6 +433,62 @@ receive_frame_r8g8b8_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf)
|
||||||
switch (xf->format) {
|
switch (xf->format) {
|
||||||
case XRT_FORMAT_L8:
|
case XRT_FORMAT_L8:
|
||||||
case XRT_FORMAT_R8G8B8: s->downstream->push_frame(s->downstream, xf); return;
|
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)) {
|
||||||
|
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
|
||||||
|
receive_frame_r8g8b8_bayer_or_l8(struct xrt_frame_sink *xs, struct xrt_frame *xf)
|
||||||
|
{
|
||||||
|
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:
|
case XRT_FORMAT_YUYV422:
|
||||||
if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) {
|
if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) {
|
||||||
return;
|
return;
|
||||||
|
@ -431,6 +535,14 @@ receive_frame_r8g8b8(struct xrt_frame_sink *xs, struct xrt_frame *xf)
|
||||||
|
|
||||||
switch (xf->format) {
|
switch (xf->format) {
|
||||||
case XRT_FORMAT_R8G8B8: s->downstream->push_frame(s->downstream, xf); return;
|
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:
|
case XRT_FORMAT_YUYV422:
|
||||||
if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) {
|
if (!create_frame_with_format(xf, XRT_FORMAT_R8G8B8, &converted)) {
|
||||||
return;
|
return;
|
||||||
|
@ -533,6 +645,27 @@ receive_frame_yuv_or_yuyv(struct xrt_frame_sink *xs, struct xrt_frame *xf)
|
||||||
xrt_frame_reference(&converted, NULL);
|
xrt_frame_reference(&converted, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
XRT_MAYBE_UNUSED static void
|
||||||
|
receive_frame_bayer(struct xrt_frame_sink *xs, struct xrt_frame *xf)
|
||||||
|
{
|
||||||
|
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
|
static void
|
||||||
break_apart(struct xrt_frame_node *node)
|
break_apart(struct xrt_frame_node *node)
|
||||||
{}
|
{}
|
||||||
|
@ -598,6 +731,23 @@ u_sink_create_to_r8g8b8_or_l8(struct xrt_frame_context *xfctx,
|
||||||
*out_xfs = &s->base;
|
*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 = receive_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
|
void
|
||||||
u_sink_create_to_yuv_yuyv_uyvy_or_l8(struct xrt_frame_context *xfctx,
|
u_sink_create_to_yuv_yuyv_uyvy_or_l8(struct xrt_frame_context *xfctx,
|
||||||
struct xrt_frame_sink *downstream,
|
struct xrt_frame_sink *downstream,
|
||||||
|
|
Loading…
Reference in a new issue