mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-22 15:18:21 +00:00
189 lines
4.8 KiB
C
189 lines
4.8 KiB
C
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <gst/gst.h>
|
|
#include <gst/video/video.h>
|
|
|
|
static gboolean use_postproc;
|
|
static GOptionEntry g_options[] = {
|
|
{"postproc", 'p', 0, G_OPTION_ARG_NONE, &use_postproc,
|
|
"use vaapipostproc to rotate rather than vaapisink", NULL},
|
|
{NULL,}
|
|
};
|
|
|
|
typedef struct _CustomData
|
|
{
|
|
GstElement *pipeline;
|
|
GstElement *rotator;
|
|
GMainLoop *loop;
|
|
gboolean orient_automatic;
|
|
} AppData;
|
|
|
|
static void
|
|
send_rotate_event (AppData * data)
|
|
{
|
|
gboolean res = FALSE;
|
|
GstEvent *event;
|
|
static gint counter = 0;
|
|
const static gchar *tags[] = { "rotate-90", "rotate-180", "rotate-270",
|
|
"rotate-0", "flip-rotate-0", "flip-rotate-90", "flip-rotate-180",
|
|
"flip-rotate-270",
|
|
};
|
|
|
|
event = gst_event_new_tag (gst_tag_list_new (GST_TAG_IMAGE_ORIENTATION,
|
|
tags[counter++ % G_N_ELEMENTS (tags)], NULL));
|
|
|
|
/* Send the event */
|
|
g_print ("Sending event %" GST_PTR_FORMAT ": ", event);
|
|
res = gst_element_send_event (data->pipeline, event);
|
|
g_print ("%s\n", res ? "ok" : "failed");
|
|
|
|
}
|
|
|
|
static void
|
|
keyboard_cb (const gchar * key, AppData * data)
|
|
{
|
|
switch (g_ascii_tolower (key[0])) {
|
|
case 'r':
|
|
send_rotate_event (data);
|
|
break;
|
|
case 's':{
|
|
if (use_postproc) {
|
|
g_object_set (G_OBJECT (data->rotator), "video-direction",
|
|
GST_VIDEO_ORIENTATION_AUTO, NULL);
|
|
} else {
|
|
/* rotation=360 means auto for vaapisnk */
|
|
g_object_set (G_OBJECT (data->rotator), "rotation", 360, NULL);
|
|
}
|
|
break;
|
|
}
|
|
case 'q':
|
|
g_main_loop_quit (data->loop);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
bus_msg (GstBus * bus, GstMessage * msg, gpointer user_data)
|
|
{
|
|
AppData *data = user_data;
|
|
|
|
switch (GST_MESSAGE_TYPE (msg)) {
|
|
case GST_MESSAGE_ELEMENT:
|
|
{
|
|
GstNavigationMessageType mtype = gst_navigation_message_get_type (msg);
|
|
if (mtype == GST_NAVIGATION_MESSAGE_EVENT) {
|
|
GstEvent *ev = NULL;
|
|
|
|
if (gst_navigation_message_parse_event (msg, &ev)) {
|
|
GstNavigationEventType type = gst_navigation_event_get_type (ev);
|
|
if (type == GST_NAVIGATION_EVENT_KEY_PRESS) {
|
|
const gchar *key;
|
|
|
|
if (gst_navigation_event_parse_key_event (ev, &key))
|
|
keyboard_cb (key, data);
|
|
}
|
|
}
|
|
if (ev)
|
|
gst_event_unref (ev);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/* Process keyboard input */
|
|
static gboolean
|
|
handle_keyboard (GIOChannel * source, GIOCondition cond, AppData * data)
|
|
{
|
|
gchar *str = NULL;
|
|
|
|
if (g_io_channel_read_line (source, &str, NULL, NULL,
|
|
NULL) != G_IO_STATUS_NORMAL) {
|
|
return TRUE;
|
|
}
|
|
|
|
keyboard_cb (str, data);
|
|
g_free (str);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
int
|
|
main (int argc, char *argv[])
|
|
{
|
|
AppData data;
|
|
GstStateChangeReturn ret;
|
|
GIOChannel *io_stdin;
|
|
GOptionContext *ctx;
|
|
GError *err = NULL;
|
|
guint srcid;
|
|
|
|
/* Initialize GStreamer */
|
|
ctx = g_option_context_new ("- test options");
|
|
if (!ctx)
|
|
return -1;
|
|
g_option_context_add_group (ctx, gst_init_get_option_group ());
|
|
g_option_context_add_main_entries (ctx, g_options, NULL);
|
|
if (!g_option_context_parse (ctx, &argc, &argv, NULL))
|
|
return -1;
|
|
g_option_context_free (ctx);
|
|
|
|
/* Print usage map */
|
|
g_print ("USAGE: Choose one of the following options, then press enter:\n"
|
|
" 'r' to send image-orientation tag event\n"
|
|
" 's' to set orient-automatic\n" " 'Q' to quit\n");
|
|
|
|
if (use_postproc) {
|
|
data.pipeline =
|
|
gst_parse_launch ("videotestsrc ! vaapipostproc name=pp ! xvimagesink",
|
|
&err);
|
|
} else {
|
|
data.pipeline =
|
|
gst_parse_launch ("videotestsrc ! vaapisink name=sink", &err);
|
|
}
|
|
if (err) {
|
|
g_printerr ("failed to create pipeline: %s\n", err->message);
|
|
g_error_free (err);
|
|
return -1;
|
|
}
|
|
|
|
if (use_postproc)
|
|
data.rotator = gst_bin_get_by_name (GST_BIN (data.pipeline), "pp");
|
|
else
|
|
data.rotator = gst_bin_get_by_name (GST_BIN (data.pipeline), "sink");
|
|
srcid = gst_bus_add_watch (GST_ELEMENT_BUS (data.pipeline), bus_msg, &data);
|
|
|
|
/* Add a keyboard watch so we get notified of keystrokes */
|
|
io_stdin = g_io_channel_unix_new (fileno (stdin));
|
|
g_io_add_watch (io_stdin, G_IO_IN, (GIOFunc) handle_keyboard, &data);
|
|
|
|
/* Start playing */
|
|
ret = gst_element_set_state (data.pipeline, GST_STATE_PLAYING);
|
|
if (ret == GST_STATE_CHANGE_FAILURE) {
|
|
g_printerr ("Unable to set the pipeline to the playing state.\n");
|
|
goto bail;
|
|
}
|
|
|
|
/* Create a GLib Main Loop and set it to run */
|
|
data.loop = g_main_loop_new (NULL, FALSE);
|
|
g_main_loop_run (data.loop);
|
|
|
|
gst_element_set_state (data.pipeline, GST_STATE_NULL);
|
|
|
|
bail:
|
|
/* Free resources */
|
|
g_source_remove (srcid);
|
|
g_main_loop_unref (data.loop);
|
|
g_io_channel_unref (io_stdin);
|
|
|
|
gst_object_unref (data.rotator);
|
|
gst_object_unref (data.pipeline);
|
|
|
|
return 0;
|
|
}
|