2022-02-19 01:29:27 +00:00
|
|
|
// Copyright 2019-2022, Collabora, Ltd.
|
2021-02-02 18:21:14 +00:00
|
|
|
// SPDX-License-Identifier: BSL-1.0
|
|
|
|
/*!
|
|
|
|
* @file
|
|
|
|
* @brief An @ref xrt_frame_sink that does gst things.
|
|
|
|
* @author Jakob Bornecrantz <jakob@collabora.com>
|
|
|
|
* @author Aaron Boxer <aaron.boxer@collabora.com>
|
2022-02-19 01:29:27 +00:00
|
|
|
* @author Moses Turner <moses@collabora.com>
|
2021-02-02 18:21:14 +00:00
|
|
|
* @ingroup aux_util
|
|
|
|
*/
|
|
|
|
|
2022-02-19 01:29:27 +00:00
|
|
|
#include "util/u_trace_marker.h"
|
2021-02-02 18:21:14 +00:00
|
|
|
#include "util/u_misc.h"
|
|
|
|
#include "util/u_debug.h"
|
|
|
|
#include "util/u_format.h"
|
|
|
|
|
|
|
|
#include "gstreamer/gst_sink.h"
|
|
|
|
#include "gstreamer/gst_pipeline.h"
|
|
|
|
#include "gstreamer/gst_internal.h"
|
2022-02-19 01:29:27 +00:00
|
|
|
#include "gst/video/video-format.h"
|
|
|
|
#include "gst/video/gstvideometa.h"
|
2021-02-02 18:21:14 +00:00
|
|
|
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Internal sink functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
|
|
|
wrapped_buffer_destroy(gpointer data)
|
|
|
|
{
|
|
|
|
struct xrt_frame *xf = (struct xrt_frame *)data;
|
|
|
|
|
|
|
|
U_LOG_T("Called");
|
|
|
|
|
|
|
|
xrt_frame_reference(&xf, NULL);
|
|
|
|
}
|
|
|
|
|
2022-02-19 01:29:27 +00:00
|
|
|
static GstVideoFormat
|
|
|
|
gst_fmt_from_xf_format(enum xrt_format format_in)
|
|
|
|
{
|
|
|
|
switch (format_in) {
|
|
|
|
case XRT_FORMAT_R8G8B8: return GST_VIDEO_FORMAT_RGB;
|
|
|
|
case XRT_FORMAT_R8G8B8A8: return GST_VIDEO_FORMAT_RGBA;
|
|
|
|
case XRT_FORMAT_R8G8B8X8: return GST_VIDEO_FORMAT_RGBx;
|
|
|
|
case XRT_FORMAT_YUYV422: return GST_VIDEO_FORMAT_YUY2;
|
|
|
|
case XRT_FORMAT_L8: return GST_VIDEO_FORMAT_GRAY8;
|
2022-03-09 11:46:43 +00:00
|
|
|
default: assert(false); return GST_VIDEO_FORMAT_UNKNOWN;
|
2022-02-19 01:29:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-11 23:45:00 +00:00
|
|
|
static void
|
|
|
|
complain_if_wrong_image_size(struct xrt_frame *xf)
|
|
|
|
{
|
2022-06-14 12:58:54 +00:00
|
|
|
// libx264 is the actual source of this requirement; it refuses to handle odd widths/heights when encoding I420
|
|
|
|
// subsampled content. OpenH264 should work, but it's easy enough to just force all users of this code to
|
|
|
|
// provide normal-sized inputs.
|
2022-06-11 23:45:00 +00:00
|
|
|
if (xf->width % 2 == 1) {
|
|
|
|
U_LOG_W("Image width needs to be divisible by 2!");
|
|
|
|
}
|
|
|
|
if (xf->height % 2 == 1) {
|
|
|
|
U_LOG_W("Image height needs to be divisible by 2!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-02 18:21:14 +00:00
|
|
|
static void
|
|
|
|
push_frame(struct xrt_frame_sink *xfs, struct xrt_frame *xf)
|
|
|
|
{
|
2022-02-19 01:29:27 +00:00
|
|
|
SINK_TRACE_MARKER();
|
2021-02-02 18:21:14 +00:00
|
|
|
struct gstreamer_sink *gs = (struct gstreamer_sink *)xfs;
|
2022-06-11 23:45:00 +00:00
|
|
|
|
|
|
|
complain_if_wrong_image_size(xf);
|
|
|
|
|
2021-02-02 18:21:14 +00:00
|
|
|
GstBuffer *buffer;
|
|
|
|
GstFlowReturn ret;
|
|
|
|
|
|
|
|
U_LOG_T(
|
|
|
|
"Called"
|
|
|
|
"\n\tformat: %s"
|
|
|
|
"\n\twidth: %u"
|
|
|
|
"\n\theight: %u",
|
|
|
|
u_format_str(xf->format), xf->width, xf->height);
|
|
|
|
|
|
|
|
/* We need to take a reference on the frame to keep it alive. */
|
|
|
|
struct xrt_frame *taken = NULL;
|
|
|
|
xrt_frame_reference(&taken, xf);
|
|
|
|
|
|
|
|
/* Wrap the frame that we now hold a reference to. */
|
|
|
|
buffer = gst_buffer_new_wrapped_full( //
|
|
|
|
0, // GstMemoryFlags flags
|
|
|
|
(gpointer)xf->data, // gpointer data
|
|
|
|
taken->size, // gsize maxsize
|
|
|
|
0, // gsize offset
|
|
|
|
taken->size, // gsize size
|
|
|
|
taken, // gpointer user_data
|
|
|
|
wrapped_buffer_destroy); // GDestroyNotify notify
|
|
|
|
|
2022-02-19 01:29:27 +00:00
|
|
|
int stride = xf->stride;
|
|
|
|
|
|
|
|
gsize offsets[4] = {0, 0, 0, 0};
|
|
|
|
gint strides[4] = {stride, 0, 0, 0};
|
|
|
|
gst_buffer_add_video_meta_full(buffer, GST_VIDEO_FRAME_FLAG_NONE, gst_fmt_from_xf_format(xf->format), xf->width,
|
|
|
|
xf->height, 1, offsets, strides);
|
|
|
|
|
2021-02-02 18:21:14 +00:00
|
|
|
//! Get the timestampe from the frame.
|
|
|
|
uint64_t xtimestamp_ns = xf->timestamp;
|
|
|
|
|
|
|
|
// Use the first frame as offset.
|
|
|
|
if (gs->offset_ns == 0) {
|
|
|
|
gs->offset_ns = xtimestamp_ns;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Need to be offset or gstreamer becomes sad.
|
|
|
|
GST_BUFFER_PTS(buffer) = xtimestamp_ns - gs->offset_ns;
|
|
|
|
|
|
|
|
// Duration is measured from last time stamp.
|
|
|
|
GST_BUFFER_DURATION(buffer) = xtimestamp_ns - gs->timestamp_ns;
|
|
|
|
gs->timestamp_ns = xtimestamp_ns;
|
|
|
|
|
|
|
|
// All done, send it to the gstreamer pipeline.
|
|
|
|
ret = gst_app_src_push_buffer((GstAppSrc *)gs->appsrc, buffer);
|
|
|
|
if (ret != GST_FLOW_OK) {
|
|
|
|
U_LOG_E("Got GST error '%i'", ret);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
enough_data(GstElement *appsrc, gpointer udata)
|
|
|
|
{
|
|
|
|
// Debugging code.
|
|
|
|
U_LOG_T("Called");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
break_apart(struct xrt_frame_node *node)
|
|
|
|
{
|
|
|
|
struct gstreamer_sink *gs = container_of(node, struct gstreamer_sink, node);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This function is called when we are shutting down, after returning
|
|
|
|
* from this function you are not allowed to call any other nodes in the
|
|
|
|
* graph. But it must be safe for other nodes to call any normal
|
|
|
|
* functions on us. Once the context is done calling break_aprt on all
|
|
|
|
* objects it will call destroy on them.
|
|
|
|
*/
|
|
|
|
|
|
|
|
(void)gs;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
destroy(struct xrt_frame_node *node)
|
|
|
|
{
|
|
|
|
struct gstreamer_sink *gs = container_of(node, struct gstreamer_sink, node);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* All of the nodes has been broken apart and none of our functions will
|
|
|
|
* be called, it's now safe to destroy and free ourselves.
|
|
|
|
*/
|
|
|
|
|
|
|
|
free(gs);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Exported functions.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
void
|
|
|
|
gstreamer_sink_send_eos(struct gstreamer_sink *gs)
|
|
|
|
{
|
|
|
|
gst_element_send_event(gs->appsrc, gst_event_new_eos());
|
|
|
|
}
|
|
|
|
|
2021-09-29 23:28:09 +00:00
|
|
|
uint64_t
|
|
|
|
gstreamer_sink_get_timestamp_offset(struct gstreamer_sink *gs)
|
|
|
|
{
|
|
|
|
return gs->offset_ns;
|
|
|
|
}
|
|
|
|
|
2021-02-02 18:21:14 +00:00
|
|
|
void
|
|
|
|
gstreamer_sink_create_with_pipeline(struct gstreamer_pipeline *gp,
|
|
|
|
uint32_t width,
|
|
|
|
uint32_t height,
|
|
|
|
enum xrt_format format,
|
|
|
|
const char *appsrc_name,
|
|
|
|
struct gstreamer_sink **out_gs,
|
|
|
|
struct xrt_frame_sink **out_xfs)
|
|
|
|
{
|
|
|
|
const char *format_str = NULL;
|
|
|
|
switch (format) {
|
|
|
|
case XRT_FORMAT_R8G8B8: format_str = "RGB"; break;
|
2022-02-19 01:29:27 +00:00
|
|
|
case XRT_FORMAT_R8G8B8A8: format_str = "RGBA"; break;
|
|
|
|
case XRT_FORMAT_R8G8B8X8: format_str = "RGBx"; break;
|
2021-02-02 18:21:14 +00:00
|
|
|
case XRT_FORMAT_YUYV422: format_str = "YUY2"; break;
|
|
|
|
case XRT_FORMAT_L8: format_str = "GRAY8"; break;
|
|
|
|
default: assert(false); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct gstreamer_sink *gs = U_TYPED_CALLOC(struct gstreamer_sink);
|
|
|
|
gs->base.push_frame = push_frame;
|
|
|
|
gs->node.break_apart = break_apart;
|
|
|
|
gs->node.destroy = destroy;
|
|
|
|
gs->gp = gp;
|
|
|
|
gs->appsrc = gst_bin_get_by_name(GST_BIN(gp->pipeline), appsrc_name);
|
|
|
|
|
|
|
|
|
|
|
|
GstCaps *caps = gst_caps_new_simple( //
|
|
|
|
"video/x-raw", //
|
|
|
|
"format", G_TYPE_STRING, format_str, //
|
|
|
|
"width", G_TYPE_INT, width, //
|
|
|
|
"height", G_TYPE_INT, height, //
|
|
|
|
"framerate", GST_TYPE_FRACTION, 0, 1, //
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_object_set(G_OBJECT(gs->appsrc), //
|
|
|
|
"caps", caps, //
|
|
|
|
"stream-type", GST_APP_STREAM_TYPE_STREAM, //
|
|
|
|
"format", GST_FORMAT_TIME, //
|
|
|
|
"is-live", TRUE, //
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
g_signal_connect(G_OBJECT(gs->appsrc), "enough-data", G_CALLBACK(enough_data), gs);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Add ourselves to the context so we are destroyed.
|
|
|
|
* This is done once we know everything is completed.
|
|
|
|
*/
|
|
|
|
xrt_frame_context_add(gp->xfctx, &gs->node);
|
|
|
|
|
|
|
|
*out_gs = gs;
|
|
|
|
*out_xfs = &gs->base;
|
|
|
|
}
|